/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2012 Jean-Pierre Charras, jp.charras at wanadoo.fr * Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck * Copyright (C) 2012 Wayne Stambaugh * Copyright (C) 1992-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 #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 #include #include "../scripting/python_scripting.h" /* Data to build the layer pair indicator button */ static std::unique_ptr LayerPairBitmap; #define BM_LAYERICON_SIZE 24 static const char s_BitmapLayerIcon[BM_LAYERICON_SIZE][BM_LAYERICON_SIZE] = { // 0 = draw pixel with white // 1 = draw pixel with black // 2 = draw pixel with top layer from router pair // 3 = draw pixel with bottom layer from router pair { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, { 2, 2, 2, 2, 2, 0, 1, 0, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3 }, }; static COLOR4D ICON_WHITE { 0.86, 0.86, 0.86, 1.0 }; static COLOR4D ICON_BLACK { 0.28, 0.28, 0.28, 1.0 }; void PCB_EDIT_FRAME::PrepareLayerIndicator( bool aForceRebuild ) { int ii, jj; COLOR4D top_color, bottom_color, background_color; bool change = aForceRebuild; int requested_scale = KiIconScale( this ); if( m_prevIconVal.previous_requested_scale != requested_scale ) { m_prevIconVal.previous_requested_scale = requested_scale; change = true; } top_color = GetColorSettings()->GetColor( GetScreen()->m_Route_Layer_TOP ); if( m_prevIconVal.previous_Route_Layer_TOP_color != top_color ) { m_prevIconVal.previous_Route_Layer_TOP_color = top_color; change = true; } bottom_color = GetColorSettings()->GetColor( GetScreen()->m_Route_Layer_BOTTOM ); if( m_prevIconVal.previous_Route_Layer_BOTTOM_color != bottom_color ) { m_prevIconVal.previous_Route_Layer_BOTTOM_color = bottom_color; change = true; } background_color = GetColorSettings()->GetColor( LAYER_PCB_BACKGROUND ); if( m_prevIconVal.previous_background_color != background_color ) { m_prevIconVal.previous_background_color = background_color; change = true; } if( change || !LayerPairBitmap ) { LayerPairBitmap = std::make_unique( 24, 24 ); // Draw the icon, with colors according to the router's layer pair wxMemoryDC iconDC; iconDC.SelectObject( *LayerPairBitmap ); wxBrush brush; wxPen pen; int buttonColor = -1; brush.SetStyle( wxBRUSHSTYLE_SOLID ); brush.SetColour( background_color.WithAlpha(1.0).ToColour() ); iconDC.SetBrush( brush ); iconDC.DrawRectangle( 0, 0, BM_LAYERICON_SIZE, BM_LAYERICON_SIZE ); for( ii = 0; ii < BM_LAYERICON_SIZE; ii++ ) { for( jj = 0; jj < BM_LAYERICON_SIZE; jj++ ) { if( s_BitmapLayerIcon[ii][jj] != buttonColor ) { switch( s_BitmapLayerIcon[ii][jj] ) { default: case 0: pen.SetColour( ICON_WHITE.ToColour() ); break; case 1: pen.SetColour( ICON_BLACK.ToColour() ); break; case 2: pen.SetColour( top_color.ToColour() ); break; case 3: pen.SetColour( bottom_color.ToColour() ); break; } buttonColor = s_BitmapLayerIcon[ii][jj]; iconDC.SetPen( pen ); } iconDC.DrawPoint( jj, ii ); } } // Deselect the bitmap from the DC in order to delete the MemoryDC safely without // deleting the bitmap iconDC.SelectObject( wxNullBitmap ); // Scale the bitmap const int scale = ( requested_scale <= 0 ) ? KiIconScale( this ) : requested_scale; wxImage image = LayerPairBitmap->ConvertToImage(); // "NEAREST" causes less mixing of colors image.Rescale( scale * image.GetWidth() / 4, scale * image.GetHeight() / 4, wxIMAGE_QUALITY_NEAREST ); LayerPairBitmap = std::make_unique( image ); if( m_auxiliaryToolBar ) { m_auxiliaryToolBar->SetToolBitmap( PCB_ACTIONS::selectLayerPair, *LayerPairBitmap ); m_auxiliaryToolBar->Refresh(); } } } void PCB_EDIT_FRAME::ReCreateHToolbar() { // Note: // To rebuild the aui toolbar, the more easy way is to clear ( calling m_mainToolBar.Clear() ) // all wxAuiToolBarItems. // However the wxAuiToolBarItems are not the owners of controls managed by // them and therefore do not delete them // So we do not recreate them after clearing the tools. wxWindowUpdateLocker dummy( this ); if( m_mainToolBar ) { m_mainToolBar->ClearToolbar(); } else { m_mainToolBar = new ACTION_TOOLBAR( this, ID_H_TOOLBAR, wxDefaultPosition, wxDefaultSize, KICAD_AUI_TB_STYLE | wxAUI_TB_HORZ_LAYOUT | wxAUI_TB_HORIZONTAL ); m_mainToolBar->SetAuiManager( &m_auimgr ); } // Set up toolbar if( Kiface().IsSingle() ) { m_mainToolBar->Add( ACTIONS::doNew ); m_mainToolBar->Add( ACTIONS::open ); } m_mainToolBar->Add( ACTIONS::save ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( PCB_ACTIONS::boardSetup ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( ACTIONS::pageSettings ); m_mainToolBar->Add( ACTIONS::print ); m_mainToolBar->Add( ACTIONS::plot ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( ACTIONS::undo ); m_mainToolBar->Add( ACTIONS::redo ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( ACTIONS::find ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( ACTIONS::zoomRedraw ); m_mainToolBar->Add( ACTIONS::zoomInCenter ); m_mainToolBar->Add( ACTIONS::zoomOutCenter ); m_mainToolBar->Add( ACTIONS::zoomFitScreen ); m_mainToolBar->Add( ACTIONS::zoomFitObjects ); m_mainToolBar->Add( ACTIONS::zoomTool, ACTION_TOOLBAR::TOGGLE, ACTION_TOOLBAR::CANCEL ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( PCB_ACTIONS::rotateCcw ); m_mainToolBar->Add( PCB_ACTIONS::rotateCw ); m_mainToolBar->Add( PCB_ACTIONS::mirrorV ); m_mainToolBar->Add( PCB_ACTIONS::mirrorH ); m_mainToolBar->Add( PCB_ACTIONS::group ); m_mainToolBar->Add( PCB_ACTIONS::ungroup ); m_mainToolBar->Add( PCB_ACTIONS::lock ); m_mainToolBar->Add( PCB_ACTIONS::unlock ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( ACTIONS::showFootprintEditor ); m_mainToolBar->Add( ACTIONS::showFootprintBrowser ); m_mainToolBar->Add( ACTIONS::show3DViewer ); m_mainToolBar->AddScaledSeparator( this ); if( !Kiface().IsSingle() ) m_mainToolBar->Add( ACTIONS::updatePcbFromSchematic ); else m_mainToolBar->Add( PCB_ACTIONS::importNetlist ); m_mainToolBar->Add( PCB_ACTIONS::runDRC ); m_mainToolBar->AddScaledSeparator( this ); m_mainToolBar->Add( PCB_ACTIONS::showEeschema ); // Add SWIG and API plugins bool scriptingAvailable = SCRIPTING::IsWxAvailable(); bool haveApiPlugins = !Pgm().GetPluginManager().GetActionsForScope( PLUGIN_ACTION_SCOPE::PCB ).empty(); if( scriptingAvailable || haveApiPlugins ) { m_mainToolBar->AddScaledSeparator( this ); if( scriptingAvailable ) { m_mainToolBar->Add( PCB_ACTIONS::showPythonConsole, ACTION_TOOLBAR::TOGGLE ); AddActionPluginTools(); } if( haveApiPlugins ) AddApiPluginTools(); } // after adding the buttons to the toolbar, must call Realize() to reflect the changes m_mainToolBar->KiRealize(); } void PCB_EDIT_FRAME::AddApiPluginTools() { // TODO: Add user control over visibility and order API_PLUGIN_MANAGER& mgr = Pgm().GetPluginManager(); mgr.ButtonBindings().clear(); std::vector actions = mgr.GetActionsForScope( PLUGIN_ACTION_SCOPE::PCB ); for( auto& action : actions ) { if( !action->show_button ) continue; const wxBitmapBundle& icon = KIPLATFORM::UI::IsDarkTheme() && action->icon_dark.IsOk() ? action->icon_dark : action->icon_light; wxAuiToolBarItem* button = m_mainToolBar->AddTool( wxID_ANY, wxEmptyString, icon, action->name ); Connect( button->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( PCB_EDIT_FRAME::OnApiPluginButton ) ); mgr.ButtonBindings().insert( { button->GetId(), action->identifier } ); } } void PCB_EDIT_FRAME::ReCreateOptToolbar() { // Note: // To rebuild the aui toolbar, the more easy way is to clear ( calling m_mainToolBar.Clear() ) // all wxAuiToolBarItems. // However the wxAuiToolBarItems are not the owners of controls managed by // them and therefore do not delete them // So we do not recreate them after clearing the tools. wxWindowUpdateLocker dummy( this ); if( m_optionsToolBar ) { m_optionsToolBar->ClearToolbar(); } else { m_optionsToolBar = new ACTION_TOOLBAR( this, ID_OPT_TOOLBAR, wxDefaultPosition, wxDefaultSize, KICAD_AUI_TB_STYLE | wxAUI_TB_VERTICAL ); m_optionsToolBar->SetAuiManager( &m_auimgr ); } m_optionsToolBar->Add( ACTIONS::toggleGrid, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( ACTIONS::toggleGridOverrides, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::togglePolarCoords, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( ACTIONS::inchesUnits, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( ACTIONS::milsUnits, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( ACTIONS::millimetersUnits, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( ACTIONS::toggleCursorStyle, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( PCB_ACTIONS::toggleHV45Mode, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( PCB_ACTIONS::showRatsnest, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::ratsnestLineMode, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( ACTIONS::highContrastMode, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::toggleNetHighlight, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( PCB_ACTIONS::zoneDisplayFilled, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::zoneDisplayOutline, ACTION_TOOLBAR::TOGGLE ); if( ADVANCED_CFG::GetCfg().m_ExtraZoneDisplayModes ) { m_optionsToolBar->Add( PCB_ACTIONS::zoneDisplayFractured, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::zoneDisplayTriangulated, ACTION_TOOLBAR::TOGGLE ); } m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( PCB_ACTIONS::padDisplayMode, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::viaDisplayMode, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::trackDisplayMode, ACTION_TOOLBAR::TOGGLE ); if( ADVANCED_CFG::GetCfg().m_DrawBoundingBoxes ) m_optionsToolBar->Add( ACTIONS::toggleBoundingBoxes, ACTION_TOOLBAR::TOGGLE ); // Tools to show/hide toolbars: m_optionsToolBar->AddScaledSeparator( this ); m_optionsToolBar->Add( PCB_ACTIONS::showLayersManager, ACTION_TOOLBAR::TOGGLE ); m_optionsToolBar->Add( PCB_ACTIONS::showProperties, ACTION_TOOLBAR::TOGGLE ); PCB_SELECTION_TOOL* selTool = m_toolManager->GetTool(); std::unique_ptr gridMenu = std::make_unique( false, selTool ); gridMenu->Add( ACTIONS::gridProperties ); gridMenu->Add( ACTIONS::gridOrigin ); m_optionsToolBar->AddToolContextMenu( ACTIONS::toggleGrid, std::move( gridMenu ) ); m_optionsToolBar->KiRealize(); } void PCB_EDIT_FRAME::ReCreateVToolbar() { wxWindowUpdateLocker dummy( this ); if( m_drawToolBar ) { m_drawToolBar->ClearToolbar(); } else { m_drawToolBar = new ACTION_TOOLBAR( this, ID_V_TOOLBAR, wxDefaultPosition, wxDefaultSize, KICAD_AUI_TB_STYLE | wxAUI_TB_VERTICAL ); m_drawToolBar->SetAuiManager( &m_auimgr ); } // Groups contained on this toolbar static ACTION_GROUP* dimensionGroup = nullptr; static ACTION_GROUP* originGroup = nullptr; static ACTION_GROUP* routingGroup = nullptr; static ACTION_GROUP* tuneGroup = nullptr; if( !dimensionGroup ) { dimensionGroup = new ACTION_GROUP( "group.pcbDimensions", { &PCB_ACTIONS::drawAlignedDimension, &PCB_ACTIONS::drawOrthogonalDimension, &PCB_ACTIONS::drawCenterDimension, &PCB_ACTIONS::drawRadialDimension, &PCB_ACTIONS::drawLeader } ); } if( !originGroup ) { originGroup = new ACTION_GROUP( "group.pcbOrigins", { &PCB_ACTIONS::gridSetOrigin, &PCB_ACTIONS::drillOrigin } ); } if( !routingGroup ) { routingGroup = new ACTION_GROUP( "group.pcbRouting", { &PCB_ACTIONS::routeSingleTrack, &PCB_ACTIONS::routeDiffPair } ); } if( !tuneGroup ) { tuneGroup = new ACTION_GROUP( "group.pcbTune", { &PCB_ACTIONS::tuneSingleTrack, &PCB_ACTIONS::tuneDiffPair, &PCB_ACTIONS::tuneSkew } ); } m_drawToolBar->Add( ACTIONS::selectionTool, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::localRatsnestTool, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddScaledSeparator( this ); m_drawToolBar->Add( PCB_ACTIONS::placeFootprint, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddGroup( routingGroup, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddGroup( tuneGroup, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawVia, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawZone, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawRuleArea, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddScaledSeparator( this ); m_drawToolBar->Add( PCB_ACTIONS::drawLine, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawArc, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawRectangle, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawCircle, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawPolygon, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::placeReferenceImage, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::placeText, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawTextBox, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( PCB_ACTIONS::drawTable, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddGroup( dimensionGroup, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( ACTIONS::deleteTool, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->AddScaledSeparator( this ); m_drawToolBar->AddGroup( originGroup, ACTION_TOOLBAR::TOGGLE ); m_drawToolBar->Add( ACTIONS::measureTool, ACTION_TOOLBAR::TOGGLE ); PCB_SELECTION_TOOL* selTool = m_toolManager->GetTool(); auto makeArcMenu = [&]() { std::unique_ptr arcMenu = std::make_unique( false, selTool ); arcMenu->Add( PCB_ACTIONS::pointEditorArcKeepCenter, ACTION_MENU::CHECK ); arcMenu->Add( PCB_ACTIONS::pointEditorArcKeepEndpoint, ACTION_MENU::CHECK ); return arcMenu; }; m_drawToolBar->AddToolContextMenu( PCB_ACTIONS::drawArc, makeArcMenu() ); auto makeRouteMenu = [&]() { std::unique_ptr routeMenu = std::make_unique( false, selTool ); routeMenu->Add( PCB_ACTIONS::routerHighlightMode, ACTION_MENU::CHECK ); routeMenu->Add( PCB_ACTIONS::routerShoveMode, ACTION_MENU::CHECK ); routeMenu->Add( PCB_ACTIONS::routerWalkaroundMode, ACTION_MENU::CHECK ); routeMenu->AppendSeparator(); routeMenu->Add( PCB_ACTIONS::routerSettingsDialog ); return routeMenu; }; m_drawToolBar->AddToolContextMenu( PCB_ACTIONS::routeSingleTrack, makeRouteMenu() ); m_drawToolBar->AddToolContextMenu( PCB_ACTIONS::routeDiffPair, makeRouteMenu() ); std::unique_ptr zoneMenu = std::make_unique( false, selTool ); zoneMenu->Add( PCB_ACTIONS::zoneFillAll ); zoneMenu->Add( PCB_ACTIONS::zoneUnfillAll ); m_drawToolBar->AddToolContextMenu( PCB_ACTIONS::drawZone, std::move( zoneMenu ) ); std::unique_ptr lineMenu = std::make_unique( false, selTool ); m_drawToolBar->AddToolContextMenu( PCB_ACTIONS::drawLine, std::move( lineMenu ) ); m_drawToolBar->KiRealize(); } void PCB_EDIT_FRAME::ReCreateAuxiliaryToolbar() { wxWindowUpdateLocker dummy( this ); if( m_auxiliaryToolBar ) { m_auxiliaryToolBar->ClearToolbar(); } else { m_auxiliaryToolBar = new ACTION_TOOLBAR( this, ID_AUX_TOOLBAR, wxDefaultPosition, wxDefaultSize, KICAD_AUI_TB_STYLE | wxAUI_TB_HORZ_LAYOUT ); // The layer indicator is special, so we register a callback directly that will // regenerate the bitmap instead of using the conditions system. auto layerIndicatorUpdate = [this] ( wxUpdateUIEvent& ) { PrepareLayerIndicator(); }; Bind( wxEVT_UPDATE_UI, layerIndicatorUpdate, PCB_ACTIONS::selectLayerPair.GetUIId() ); m_auxiliaryToolBar->SetAuiManager( &m_auimgr ); } /* Set up toolbar items */ // Creates box to display and choose tracks widths: if( m_SelTrackWidthBox == nullptr ) m_SelTrackWidthBox = new wxChoice( m_auxiliaryToolBar, ID_AUX_TOOLBAR_PCB_TRACK_WIDTH, wxDefaultPosition, wxDefaultSize, 0, nullptr ); UpdateTrackWidthSelectBox( m_SelTrackWidthBox, true, true ); m_auxiliaryToolBar->AddControl( m_SelTrackWidthBox ); m_SelTrackWidthBox->SetToolTip( _( "Select the default width for new tracks. Note that this " "width can be overridden by the board minimum width, or by " "the width of an existing track if the 'Use Existing Track " "Width' feature is enabled." ) ); m_auxiliaryToolBar->AddTool( ID_AUX_TOOLBAR_PCB_SELECT_AUTO_WIDTH, wxEmptyString, KiScaledBitmap( BITMAPS::auto_track_width, this ), _( "When routing from an existing track use its width instead " "of the current width setting" ), wxITEM_CHECK ); m_auxiliaryToolBar->AddScaledSeparator( this ); // Creates box to display and choose vias diameters: if( m_SelViaSizeBox == nullptr ) m_SelViaSizeBox = new wxChoice( m_auxiliaryToolBar, ID_AUX_TOOLBAR_PCB_VIA_SIZE, wxDefaultPosition, wxDefaultSize, 0, nullptr ); UpdateViaSizeSelectBox( m_SelViaSizeBox, true, true ); m_auxiliaryToolBar->AddControl( m_SelViaSizeBox ); m_auxiliaryToolBar->AddScaledSeparator( this ); if( m_SelLayerBox == nullptr ) { m_SelLayerBox = new PCB_LAYER_BOX_SELECTOR( m_auxiliaryToolBar, ID_TOOLBARH_PCB_SELECT_LAYER ); m_SelLayerBox->SetBoardFrame( this ); } ReCreateLayerBox( false ); m_auxiliaryToolBar->AddControl( m_SelLayerBox ); m_auxiliaryToolBar->Add( PCB_ACTIONS::selectLayerPair ); PrepareLayerIndicator( true ); // Force rebuild of the bitmap with the active layer colors // Add the box to display and select the current grid size: m_auxiliaryToolBar->AddScaledSeparator( this ); if( m_gridSelectBox == nullptr ) m_gridSelectBox = new wxChoice( m_auxiliaryToolBar, ID_ON_GRID_SELECT, wxDefaultPosition, wxDefaultSize, 0, nullptr ); UpdateGridSelectBox(); m_auxiliaryToolBar->AddControl( m_gridSelectBox ); // Add the box to display and select the current Zoom m_auxiliaryToolBar->AddScaledSeparator( this ); if( m_zoomSelectBox == nullptr ) m_zoomSelectBox = new wxChoice( m_auxiliaryToolBar, ID_ON_ZOOM_SELECT, wxDefaultPosition, wxDefaultSize, 0, nullptr ); UpdateZoomSelectBox(); m_auxiliaryToolBar->AddControl( m_zoomSelectBox ); // Go through and ensure the comboboxes are the correct size, since the strings in the // box could have changed widths. m_auxiliaryToolBar->UpdateControlWidth( ID_AUX_TOOLBAR_PCB_TRACK_WIDTH ); m_auxiliaryToolBar->UpdateControlWidth( ID_AUX_TOOLBAR_PCB_VIA_SIZE ); m_auxiliaryToolBar->UpdateControlWidth( ID_ON_ZOOM_SELECT ); m_auxiliaryToolBar->UpdateControlWidth( ID_ON_GRID_SELECT ); m_auxiliaryToolBar->UpdateControlWidth( ID_TOOLBARH_PCB_SELECT_LAYER ); // after adding the buttons to the toolbar, must call Realize() m_auxiliaryToolBar->KiRealize(); } void PCB_EDIT_FRAME::UpdateToolbarControlSizes() { if( m_mainToolBar ) { // Update the item widths } if( m_auxiliaryToolBar ) { // Update the item widths m_auxiliaryToolBar->UpdateControlWidth( ID_AUX_TOOLBAR_PCB_TRACK_WIDTH ); m_auxiliaryToolBar->UpdateControlWidth( ID_AUX_TOOLBAR_PCB_VIA_SIZE ); m_auxiliaryToolBar->UpdateControlWidth( ID_ON_ZOOM_SELECT ); m_auxiliaryToolBar->UpdateControlWidth( ID_ON_GRID_SELECT ); m_auxiliaryToolBar->UpdateControlWidth( ID_TOOLBARH_PCB_SELECT_LAYER ); } } static wxString ComboBoxUnits( EDA_UNITS aUnits, double aValue, bool aIncludeLabel = true ) { wxString text; const wxChar* format; switch( aUnits ) { default: wxASSERT_MSG( false, wxT( "Invalid unit" ) ); KI_FALLTHROUGH; case EDA_UNITS::UNSCALED: format = wxT( "%.0f" ); break; case EDA_UNITS::MILLIMETRES: format = wxT( "%.3f" ); break; case EDA_UNITS::MILS: format = wxT( "%.2f" ); break; case EDA_UNITS::INCHES: format = wxT( "%.5f" ); break; } text.Printf( format, EDA_UNIT_UTILS::UI::ToUserUnit( pcbIUScale, aUnits, aValue ) ); if( aIncludeLabel ) text += EDA_UNIT_UTILS::GetText( aUnits, EDA_DATA_TYPE::DISTANCE ); return text; } void PCB_EDIT_FRAME::UpdateTrackWidthSelectBox( wxChoice* aTrackWidthSelectBox, bool aShowNetclass, bool aShowEdit ) { if( aTrackWidthSelectBox == nullptr ) return; EDA_UNITS primaryUnit; EDA_UNITS secondaryUnit; GetUnitPair( primaryUnit, secondaryUnit ); wxString msg; aTrackWidthSelectBox->Clear(); if( aShowNetclass ) aTrackWidthSelectBox->Append( _( "Track: use netclass width" ) ); for( unsigned ii = 1; ii < GetDesignSettings().m_TrackWidthList.size(); ii++ ) { int size = GetDesignSettings().m_TrackWidthList[ii]; msg.Printf( _( "Track: %s (%s)" ), ComboBoxUnits( primaryUnit, size ), ComboBoxUnits( secondaryUnit, size ) ); aTrackWidthSelectBox->Append( msg ); } if( aShowEdit ) { aTrackWidthSelectBox->Append( wxT( "---" ) ); aTrackWidthSelectBox->Append( _( "Edit Pre-defined Sizes..." ) ); } if( GetDesignSettings().GetTrackWidthIndex() >= GetDesignSettings().m_TrackWidthList.size() ) GetDesignSettings().SetTrackWidthIndex( 0 ); aTrackWidthSelectBox->SetSelection( GetDesignSettings().GetTrackWidthIndex() ); } void PCB_EDIT_FRAME::UpdateViaSizeSelectBox( wxChoice* aViaSizeSelectBox, bool aShowNetclass, bool aShowEdit ) { if( aViaSizeSelectBox == nullptr ) return; aViaSizeSelectBox->Clear(); COMMON_TOOLS* cmnTool = m_toolManager->GetTool(); EDA_UNITS primaryUnit = GetUserUnits(); EDA_UNITS secondaryUnit = EDA_UNITS::MILS; if( EDA_UNIT_UTILS::IsImperialUnit( primaryUnit ) ) { if( cmnTool ) secondaryUnit = cmnTool->GetLastMetricUnits(); else secondaryUnit = EDA_UNITS::MILLIMETRES; } else { if( cmnTool ) secondaryUnit = cmnTool->GetLastImperialUnits(); else secondaryUnit = EDA_UNITS::MILS; } if( aShowNetclass ) aViaSizeSelectBox->Append( _( "Via: use netclass sizes" ) ); for( unsigned ii = 1; ii < GetDesignSettings().m_ViasDimensionsList.size(); ii++ ) { VIA_DIMENSION viaDimension = GetDesignSettings().m_ViasDimensionsList[ii]; wxString msg, priStr, secStr; double diam = viaDimension.m_Diameter; double hole = viaDimension.m_Drill; if( hole > 0 ) { priStr = ComboBoxUnits( primaryUnit, diam, false ) + wxT( " / " ) + ComboBoxUnits( primaryUnit, hole, true ); secStr = ComboBoxUnits( secondaryUnit, diam, false ) + wxT( " / " ) + ComboBoxUnits( secondaryUnit, hole, true ); } else { priStr = ComboBoxUnits( primaryUnit, diam, true ); secStr = ComboBoxUnits( secondaryUnit, diam, true ); } msg.Printf( _( "Via: %s (%s)" ), priStr, secStr ); aViaSizeSelectBox->Append( msg ); } if( aShowEdit ) { aViaSizeSelectBox->Append( wxT( "---" ) ); aViaSizeSelectBox->Append( _( "Edit Pre-defined Sizes..." ) ); } if( GetDesignSettings().GetViaSizeIndex() >= GetDesignSettings().m_ViasDimensionsList.size() ) GetDesignSettings().SetViaSizeIndex( 0 ); aViaSizeSelectBox->SetSelection( GetDesignSettings().GetViaSizeIndex() ); } void PCB_EDIT_FRAME::ReCreateLayerBox( bool aForceResizeToolbar ) { if( m_SelLayerBox == nullptr || m_auxiliaryToolBar == nullptr ) return; m_SelLayerBox->SetToolTip( _( "+/- to switch" ) ); m_SelLayerBox->Resync(); if( aForceResizeToolbar ) UpdateToolbarControlSizes(); } void PCB_EDIT_FRAME::ToggleLayersManager() { PCBNEW_SETTINGS* settings = GetPcbNewSettings(); wxAuiPaneInfo& layersManager = m_auimgr.GetPane( "LayersManager" ); wxAuiPaneInfo& selectionFilter = m_auimgr.GetPane( "SelectionFilter" ); // show auxiliary Vertical layers and visibility manager toolbar m_show_layer_manager_tools = !m_show_layer_manager_tools; layersManager.Show( m_show_layer_manager_tools ); selectionFilter.Show( m_show_layer_manager_tools ); if( m_show_layer_manager_tools ) { SetAuiPaneSize( m_auimgr, layersManager, settings->m_AuiPanels.right_panel_width, -1 ); } else { settings->m_AuiPanels.right_panel_width = m_appearancePanel->GetSize().x; m_auimgr.Update(); } } void PCB_EDIT_FRAME::ToggleNetInspector() { PCBNEW_SETTINGS* settings = GetPcbNewSettings(); wxAuiPaneInfo& netInspectorPanel = m_auimgr.GetPane( NetInspectorPanelName() ); m_show_net_inspector = !m_show_net_inspector; netInspectorPanel.Show( m_show_net_inspector ); if( m_show_net_inspector ) { SetAuiPaneSize( m_auimgr, netInspectorPanel, settings->m_AuiPanels.net_inspector_width, -1 ); m_netInspectorPanel->OnShowPanel(); } else { m_netInspectorPanel->SaveSettings(); settings->m_AuiPanels.net_inspector_width = m_netInspectorPanel->GetSize().x; m_auimgr.Update(); } } void PCB_EDIT_FRAME::ToggleSearch() { PCBNEW_SETTINGS* settings = GetPcbNewSettings(); // Ensure m_show_search is up to date (the pane can be closed outside the menu) m_show_search = m_auimgr.GetPane( SearchPaneName() ).IsShown(); m_show_search = !m_show_search; wxAuiPaneInfo& searchPaneInfo = m_auimgr.GetPane( SearchPaneName() ); searchPaneInfo.Show( m_show_search ); if( m_show_search ) { searchPaneInfo.Direction( settings->m_AuiPanels.search_panel_dock_direction ); if( settings->m_AuiPanels.search_panel_dock_direction == wxAUI_DOCK_TOP || settings->m_AuiPanels.search_panel_dock_direction == wxAUI_DOCK_BOTTOM ) { SetAuiPaneSize( m_auimgr, searchPaneInfo, -1, settings->m_AuiPanels.search_panel_height ); } else if( settings->m_AuiPanels.search_panel_dock_direction == wxAUI_DOCK_LEFT || settings->m_AuiPanels.search_panel_dock_direction == wxAUI_DOCK_RIGHT ) { SetAuiPaneSize( m_auimgr, searchPaneInfo, settings->m_AuiPanels.search_panel_width, -1 ); } m_searchPane->FocusSearch(); } else { settings->m_AuiPanels.search_panel_height = m_searchPane->GetSize().y; settings->m_AuiPanels.search_panel_width = m_searchPane->GetSize().x; settings->m_AuiPanels.search_panel_dock_direction = searchPaneInfo.dock_direction; m_auimgr.Update(); } } void PCB_EDIT_FRAME::OnUpdateSelectTrackWidth( wxUpdateUIEvent& aEvent ) { if( aEvent.GetId() == ID_AUX_TOOLBAR_PCB_TRACK_WIDTH ) { BOARD_DESIGN_SETTINGS& bds = GetDesignSettings(); int sel; if( bds.UseCustomTrackViaSize() ) sel = wxNOT_FOUND; else sel = bds.GetTrackWidthIndex(); if( m_SelTrackWidthBox->GetSelection() != sel ) m_SelTrackWidthBox->SetSelection( sel ); } } void PCB_EDIT_FRAME::OnUpdateSelectViaSize( wxUpdateUIEvent& aEvent ) { if( aEvent.GetId() == ID_AUX_TOOLBAR_PCB_VIA_SIZE ) { BOARD_DESIGN_SETTINGS& bds = GetDesignSettings(); int sel = 0; if( bds.UseCustomTrackViaSize() ) sel = wxNOT_FOUND; else sel = bds.GetViaSizeIndex(); if( m_SelViaSizeBox->GetSelection() != sel ) m_SelViaSizeBox->SetSelection( sel ); } } void PCB_EDIT_FRAME::OnUpdateSelectAutoWidth( wxUpdateUIEvent& aEvent ) { BOARD_DESIGN_SETTINGS& bds = GetDesignSettings(); aEvent.Check( bds.m_UseConnectedTrackWidth ); } void PCB_EDIT_FRAME::OnUpdateLayerSelectBox( wxUpdateUIEvent& aEvent ) { if( m_SelLayerBox->GetLayerSelection() != GetActiveLayer() ) m_SelLayerBox->SetLayerSelection( GetActiveLayer() ); }