mirror of
https://gitlab.com/kicad/code/kicad.git
synced 2025-09-14 18:23:15 +02:00
Action plugin authors can specify alternative icon path for dark theme. If it's not specified then standard icon will be used for both light and dark themes. If neither icon is specified then the default puzzle piece icon is used. Fixes #7984
1755 lines
58 KiB
C++
1755 lines
58 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2018 Jean-Pierre Charras, jp.charras at wanadoo.fr
|
|
* Copyright (C) 2013 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
|
|
* Copyright (C) 2013 Wayne Stambaugh <stambaughw@gmail.com>
|
|
* Copyright (C) 2013-2021 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <kiface_i.h>
|
|
#include <kiway.h>
|
|
#include <pgm_base.h>
|
|
#include <pcb_edit_frame.h>
|
|
#include <3d_viewer/eda_3d_viewer.h>
|
|
#include <fp_lib_table.h>
|
|
#include <bitmaps.h>
|
|
#include <trace_helpers.h>
|
|
#include <pcbnew_id.h>
|
|
#include <pcbnew_settings.h>
|
|
#include <pcb_layer_box_selector.h>
|
|
#include <footprint_edit_frame.h>
|
|
#include <dialog_plot.h>
|
|
#include <dialog_footprint_properties.h>
|
|
#include <dialogs/dialog_exchange_footprints.h>
|
|
#include <dialog_board_setup.h>
|
|
#include <convert_to_biu.h>
|
|
#include <invoke_pcb_dialog.h>
|
|
#include <board.h>
|
|
#include <footprint.h>
|
|
#include <drawing_sheet/ds_proxy_view_item.h>
|
|
#include <connectivity/connectivity_data.h>
|
|
#include <wildcards_and_files_ext.h>
|
|
#include <pcb_draw_panel_gal.h>
|
|
#include <functional>
|
|
#include <project/project_file.h>
|
|
#include <project/project_local_settings.h>
|
|
#include <project/net_settings.h>
|
|
#include <settings/common_settings.h>
|
|
#include <settings/settings_manager.h>
|
|
#include <tool/tool_manager.h>
|
|
#include <tool/tool_dispatcher.h>
|
|
#include <tool/action_toolbar.h>
|
|
#include <tool/common_control.h>
|
|
#include <tool/common_tools.h>
|
|
#include <tool/selection.h>
|
|
#include <tool/zoom_tool.h>
|
|
#include <tools/pcb_selection_tool.h>
|
|
#include <tools/pcb_picker_tool.h>
|
|
#include <tools/pcb_point_editor.h>
|
|
#include <tools/edit_tool.h>
|
|
#include <tools/group_tool.h>
|
|
#include <tools/drc_tool.h>
|
|
#include <tools/global_edit_tool.h>
|
|
#include <tools/convert_tool.h>
|
|
#include <tools/drawing_tool.h>
|
|
#include <tools/pcb_control.h>
|
|
#include <tools/board_editor_control.h>
|
|
#include <tools/board_inspection_tool.h>
|
|
#include <tools/pcb_editor_conditions.h>
|
|
#include <tools/pcb_viewer_tools.h>
|
|
#include <tools/board_reannotate_tool.h>
|
|
#include <tools/placement_tool.h>
|
|
#include <tools/pad_tool.h>
|
|
#include <microwave/microwave_tool.h>
|
|
#include <tools/position_relative_tool.h>
|
|
#include <tools/zone_filler_tool.h>
|
|
#include <tools/pcb_actions.h>
|
|
#include <router/router_tool.h>
|
|
#include <router/length_tuner_tool.h>
|
|
#include <autorouter/autoplace_tool.h>
|
|
#include <gestfich.h>
|
|
#include <executable_names.h>
|
|
#include <netlist_reader/board_netlist_updater.h>
|
|
#include <netlist_reader/netlist_reader.h>
|
|
#include <netlist_reader/pcb_netlist.h>
|
|
#include <wx/wupdlock.h>
|
|
#include <dialog_drc.h> // for DIALOG_DRC_WINDOW_NAME definition
|
|
#include <ratsnest/ratsnest_view_item.h>
|
|
#include <widgets/appearance_controls.h>
|
|
#include <widgets/panel_selection_filter.h>
|
|
#include <kiplatform/app.h>
|
|
|
|
|
|
#include <widgets/infobar.h>
|
|
|
|
#if defined(KICAD_SCRIPTING) || defined(KICAD_SCRIPTING_WXPYTHON)
|
|
#include <python_scripting.h>
|
|
#if defined(KICAD_SCRIPTING_ACTION_MENU)
|
|
#include <action_plugin.h>
|
|
#endif
|
|
#endif
|
|
|
|
|
|
using namespace std::placeholders;
|
|
|
|
|
|
BEGIN_EVENT_TABLE( PCB_EDIT_FRAME, PCB_BASE_FRAME )
|
|
EVT_SOCKET( ID_EDA_SOCKET_EVENT_SERV, PCB_EDIT_FRAME::OnSockRequestServer )
|
|
EVT_SOCKET( ID_EDA_SOCKET_EVENT, PCB_EDIT_FRAME::OnSockRequest )
|
|
|
|
EVT_CHOICE( ID_ON_ZOOM_SELECT, PCB_EDIT_FRAME::OnSelectZoom )
|
|
EVT_CHOICE( ID_ON_GRID_SELECT, PCB_EDIT_FRAME::OnSelectGrid )
|
|
|
|
EVT_SIZE( PCB_EDIT_FRAME::OnSize )
|
|
|
|
EVT_TOOL( ID_MENU_RECOVER_BOARD_AUTOSAVE, PCB_EDIT_FRAME::Files_io )
|
|
|
|
// Menu Files:
|
|
EVT_MENU( ID_MAIN_MENUBAR, PCB_EDIT_FRAME::Process_Special_Functions )
|
|
|
|
EVT_MENU( ID_IMPORT_NON_KICAD_BOARD, PCB_EDIT_FRAME::Files_io )
|
|
EVT_MENU_RANGE( ID_FILE1, ID_FILEMAX, PCB_EDIT_FRAME::OnFileHistory )
|
|
EVT_MENU( ID_FILE_LIST_CLEAR, PCB_EDIT_FRAME::OnClearFileHistory )
|
|
|
|
EVT_MENU( ID_GEN_EXPORT_FILE_GENCADFORMAT, PCB_EDIT_FRAME::ExportToGenCAD )
|
|
EVT_MENU( ID_GEN_EXPORT_FILE_VRML, PCB_EDIT_FRAME::OnExportVRML )
|
|
EVT_MENU( ID_GEN_EXPORT_FILE_IDF3, PCB_EDIT_FRAME::OnExportIDF3 )
|
|
EVT_MENU( ID_GEN_EXPORT_FILE_STEP, PCB_EDIT_FRAME::OnExportSTEP )
|
|
EVT_MENU( ID_GEN_EXPORT_FILE_HYPERLYNX, PCB_EDIT_FRAME::OnExportHyperlynx )
|
|
|
|
EVT_MENU( ID_MENU_EXPORT_FOOTPRINTS_TO_LIBRARY, PCB_EDIT_FRAME::Process_Special_Functions )
|
|
EVT_MENU( ID_MENU_EXPORT_FOOTPRINTS_TO_NEW_LIBRARY, PCB_EDIT_FRAME::Process_Special_Functions )
|
|
|
|
EVT_MENU( wxID_EXIT, PCB_EDIT_FRAME::OnQuit )
|
|
EVT_MENU( wxID_CLOSE, PCB_EDIT_FRAME::OnQuit )
|
|
|
|
// menu Config
|
|
EVT_MENU( ID_GRID_SETTINGS, PCB_EDIT_FRAME::OnGridSettings )
|
|
|
|
// menu Postprocess
|
|
EVT_MENU( ID_PCB_GEN_CMP_FILE, PCB_EDIT_FRAME::RecreateCmpFileFromBoard )
|
|
|
|
// Horizontal toolbar
|
|
EVT_TOOL( ID_GEN_PLOT_SVG, PCB_EDIT_FRAME::ExportSVG )
|
|
EVT_TOOL( ID_AUX_TOOLBAR_PCB_SELECT_AUTO_WIDTH, PCB_EDIT_FRAME::Tracks_and_Vias_Size_Event )
|
|
EVT_COMBOBOX( ID_TOOLBARH_PCB_SELECT_LAYER, PCB_EDIT_FRAME::Process_Special_Functions )
|
|
EVT_CHOICE( ID_AUX_TOOLBAR_PCB_TRACK_WIDTH, PCB_EDIT_FRAME::Tracks_and_Vias_Size_Event )
|
|
EVT_CHOICE( ID_AUX_TOOLBAR_PCB_VIA_SIZE, PCB_EDIT_FRAME::Tracks_and_Vias_Size_Event )
|
|
|
|
|
|
#if defined(KICAD_SCRIPTING) && defined(KICAD_SCRIPTING_ACTION_MENU)
|
|
EVT_TOOL( ID_TOOLBARH_PCB_ACTION_PLUGIN_REFRESH, PCB_EDIT_FRAME::OnActionPluginRefresh )
|
|
EVT_TOOL( ID_TOOLBARH_PCB_ACTION_PLUGIN_SHOW_FOLDER, PCB_EDIT_FRAME::OnActionPluginShowFolder )
|
|
#endif
|
|
|
|
// Tracks and vias sizes general options
|
|
EVT_MENU_RANGE( ID_POPUP_PCB_SELECT_WIDTH_START_RANGE, ID_POPUP_PCB_SELECT_WIDTH_END_RANGE,
|
|
PCB_EDIT_FRAME::Tracks_and_Vias_Size_Event )
|
|
|
|
// User interface update event handlers.
|
|
EVT_UPDATE_UI( ID_TOOLBARH_PCB_SELECT_LAYER, PCB_EDIT_FRAME::OnUpdateLayerSelectBox )
|
|
EVT_UPDATE_UI( ID_AUX_TOOLBAR_PCB_TRACK_WIDTH, PCB_EDIT_FRAME::OnUpdateSelectTrackWidth )
|
|
EVT_UPDATE_UI( ID_AUX_TOOLBAR_PCB_VIA_SIZE, PCB_EDIT_FRAME::OnUpdateSelectViaSize )
|
|
EVT_UPDATE_UI( ID_AUX_TOOLBAR_PCB_SELECT_AUTO_WIDTH, PCB_EDIT_FRAME::OnUpdateSelectAutoWidth )
|
|
EVT_UPDATE_UI_RANGE( ID_POPUP_PCB_SELECT_WIDTH1, ID_POPUP_PCB_SELECT_WIDTH8,
|
|
PCB_EDIT_FRAME::OnUpdateSelectTrackWidth )
|
|
EVT_UPDATE_UI_RANGE( ID_POPUP_PCB_SELECT_VIASIZE1, ID_POPUP_PCB_SELECT_VIASIZE8,
|
|
PCB_EDIT_FRAME::OnUpdateSelectViaSize )
|
|
END_EVENT_TABLE()
|
|
|
|
|
|
PCB_EDIT_FRAME::PCB_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent ) :
|
|
PCB_BASE_EDIT_FRAME( aKiway, aParent, FRAME_PCB_EDITOR, wxT( "PCB Editor" ), wxDefaultPosition,
|
|
wxDefaultSize, KICAD_DEFAULT_DRAWFRAME_STYLE, PCB_EDIT_FRAME_NAME )
|
|
{
|
|
m_maximizeByDefault = true;
|
|
m_showBorderAndTitleBlock = true; // true to display sheet references
|
|
m_SelTrackWidthBox = NULL;
|
|
m_SelViaSizeBox = NULL;
|
|
m_SelLayerBox = NULL;
|
|
m_show_layer_manager_tools = true;
|
|
m_hasAutoSave = true;
|
|
|
|
// We don't know what state board was in when it was lasat saved, so we have to
|
|
// assume dirty
|
|
m_ZoneFillsDirty = true;
|
|
|
|
m_rotationAngle = 900;
|
|
m_aboutTitle = _( "KiCad PCB Editor" );
|
|
|
|
// Create GAL canvas
|
|
auto canvas = new PCB_DRAW_PANEL_GAL( this, -1, wxPoint( 0, 0 ), m_frameSize,
|
|
GetGalDisplayOptions(),
|
|
EDA_DRAW_PANEL_GAL::GAL_FALLBACK );
|
|
|
|
SetCanvas( canvas );
|
|
|
|
SetBoard( new BOARD() );
|
|
|
|
wxIcon icon;
|
|
wxIconBundle icon_bundle;
|
|
|
|
icon.CopyFromBitmap( KiBitmap( BITMAPS::icon_pcbnew ) );
|
|
icon_bundle.AddIcon( icon );
|
|
icon.CopyFromBitmap( KiBitmap( BITMAPS::icon_pcbnew_32 ) );
|
|
icon_bundle.AddIcon( icon );
|
|
icon.CopyFromBitmap( KiBitmap( BITMAPS::icon_pcbnew_16 ) );
|
|
icon_bundle.AddIcon( icon );
|
|
|
|
SetIcons( icon_bundle );
|
|
|
|
// LoadSettings() *after* creating m_LayersManager, because LoadSettings()
|
|
// initialize parameters in m_LayersManager
|
|
LoadSettings( config() );
|
|
|
|
SetScreen( new PCB_SCREEN( GetPageSettings().GetSizeIU() ) );
|
|
|
|
// PCB drawings start in the upper left corner.
|
|
GetScreen()->m_Center = false;
|
|
|
|
setupTools();
|
|
setupUIConditions();
|
|
|
|
ReCreateMenuBar();
|
|
ReCreateHToolbar();
|
|
ReCreateAuxiliaryToolbar();
|
|
ReCreateVToolbar();
|
|
ReCreateOptToolbar();
|
|
|
|
m_selectionFilterPanel = new PANEL_SELECTION_FILTER( this );
|
|
|
|
m_appearancePanel = new APPEARANCE_CONTROLS( this, GetCanvas() );
|
|
|
|
m_auimgr.SetManagedWindow( this );
|
|
|
|
CreateInfoBar();
|
|
|
|
unsigned int auiFlags = wxAUI_MGR_DEFAULT;
|
|
#if !defined( _WIN32 )
|
|
// Windows cannot redraw the UI fast enough during a live resize and may lead to all kinds of graphical glitches
|
|
auiFlags |= wxAUI_MGR_LIVE_RESIZE;
|
|
#endif
|
|
m_auimgr.SetFlags( auiFlags );
|
|
|
|
// Rows; layers 4 - 6
|
|
m_auimgr.AddPane( m_mainToolBar, EDA_PANE().HToolbar().Name( "MainToolbar" )
|
|
.Top().Layer( 6 ) );
|
|
m_auimgr.AddPane( m_auxiliaryToolBar, EDA_PANE().HToolbar().Name( "AuxToolbar" )
|
|
.Top().Layer( 5 ) );
|
|
m_auimgr.AddPane( m_messagePanel, EDA_PANE().Messages().Name( "MsgPanel" )
|
|
.Bottom().Layer( 6 ) );
|
|
|
|
// Columns; layers 1 - 3
|
|
m_auimgr.AddPane( m_optionsToolBar, EDA_PANE().VToolbar().Name( "OptToolbar" )
|
|
.Left().Layer( 3 ) );
|
|
|
|
m_auimgr.AddPane( m_drawToolBar, EDA_PANE().VToolbar().Name( "ToolsToolbar" )
|
|
.Right().Layer( 3 ) );
|
|
|
|
m_auimgr.AddPane( m_appearancePanel, EDA_PANE().Name( "LayersManager" )
|
|
.Right().Layer( 4 )
|
|
.Caption( _( "Appearance" ) ).PaneBorder( false )
|
|
.MinSize( 180, -1 ).BestSize( 180, -1 ) );
|
|
|
|
m_auimgr.AddPane( m_selectionFilterPanel, EDA_PANE().Name( "SelectionFilter" )
|
|
.Right().Layer( 4 ).Position( 2 )
|
|
.Caption( _( "Selection Filter" ) ).PaneBorder( false )
|
|
.MinSize( 180, -1 ).BestSize( 180, -1 ) );
|
|
|
|
// Center
|
|
m_auimgr.AddPane( GetCanvas(), EDA_PANE().Canvas().Name( "DrawFrame" )
|
|
.Center() );
|
|
|
|
m_auimgr.GetPane( "LayersManager" ).Show( m_show_layer_manager_tools );
|
|
m_auimgr.GetPane( "SelectionFilter" ).Show( m_show_layer_manager_tools );
|
|
|
|
// The selection filter doesn't need to grow in the vertical direction when docked
|
|
m_auimgr.GetPane( "SelectionFilter" ).dock_proportion = 0;
|
|
|
|
m_auimgr.GetArtProvider()->SetColour( wxAUI_DOCKART_ACTIVE_CAPTION_TEXT_COLOUR,
|
|
wxSystemSettings::GetColour( wxSYS_COLOUR_BTNTEXT ) );
|
|
m_auimgr.GetArtProvider()->SetColour( wxAUI_DOCKART_INACTIVE_CAPTION_TEXT_COLOUR,
|
|
wxSystemSettings::GetColour( wxSYS_COLOUR_BTNTEXT ) );
|
|
|
|
FinishAUIInitialization();
|
|
|
|
if( PCBNEW_SETTINGS* settings = dynamic_cast<PCBNEW_SETTINGS*>( config() ) )
|
|
{
|
|
if( settings->m_AuiPanels.right_panel_width > 0 )
|
|
{
|
|
wxAuiPaneInfo& layersManager = m_auimgr.GetPane( "LayersManager" );
|
|
|
|
// wxAUI hack: force width by setting MinSize() and then Fixed()
|
|
// thanks to ZenJu http://trac.wxwidgets.org/ticket/13180
|
|
layersManager.MinSize( settings->m_AuiPanels.right_panel_width, -1 );
|
|
layersManager.Fixed();
|
|
m_auimgr.Update();
|
|
|
|
// now make it resizable again
|
|
layersManager.MinSize( 180, -1 );
|
|
layersManager.Resizable();
|
|
m_auimgr.Update();
|
|
}
|
|
|
|
m_appearancePanel->SetTabIndex( settings->m_AuiPanels.appearance_panel_tab );
|
|
}
|
|
|
|
GetToolManager()->RunAction( ACTIONS::zoomFitScreen, false );
|
|
|
|
// This is used temporarily to fix a client size issue on GTK that causes zoom to fit
|
|
// to calculate the wrong zoom size. See PCB_EDIT_FRAME::onSize().
|
|
Bind( wxEVT_SIZE, &PCB_EDIT_FRAME::onSize, this );
|
|
|
|
resolveCanvasType();
|
|
|
|
setupUnits( config() );
|
|
|
|
// Ensure the Python interpreter is up to date with its environment variables
|
|
PythonSyncEnvironmentVariables();
|
|
PythonSyncProjectName();
|
|
|
|
GetCanvas()->SwitchBackend( m_canvasType );
|
|
ActivateGalCanvas();
|
|
|
|
// Default shutdown reason until a file is loaded
|
|
KIPLATFORM::APP::SetShutdownBlockReason( this, _( "New PCB file is unsaved" ) );
|
|
|
|
// disable Export STEP item if kicad2step does not exist
|
|
wxString strK2S = Pgm().GetExecutablePath();
|
|
|
|
#ifdef __WXMAC__
|
|
if( strK2S.Find( "pcbnew.app" ) != wxNOT_FOUND )
|
|
{
|
|
// On macOS, we have standalone applications inside the main bundle, so we handle that here:
|
|
strK2S += "../../";
|
|
}
|
|
|
|
strK2S += "Contents/MacOS/";
|
|
#endif
|
|
|
|
wxFileName appK2S( strK2S, "kicad2step" );
|
|
|
|
#ifdef _WIN32
|
|
appK2S.SetExt( "exe" );
|
|
#endif
|
|
|
|
// Ensure the window is on top
|
|
Raise();
|
|
|
|
// if( !appK2S.FileExists() )
|
|
// GetMenuBar()->FindItem( ID_GEN_EXPORT_FILE_STEP )->Enable( false );
|
|
|
|
// AUI doesn't refresh properly on wxMac after changes in eb7dc6dd, so force it to
|
|
#ifdef __WXMAC__
|
|
if( Kiface().IsSingle() )
|
|
{
|
|
CallAfter( [&]()
|
|
{
|
|
m_appearancePanel->OnBoardChanged();
|
|
} );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
PCB_EDIT_FRAME::~PCB_EDIT_FRAME()
|
|
{
|
|
// Close modeless dialogs
|
|
wxWindow* open_dlg = wxWindow::FindWindowByName( DIALOG_DRC_WINDOW_NAME );
|
|
|
|
if( open_dlg )
|
|
open_dlg->Close( true );
|
|
|
|
// Shutdown all running tools
|
|
if( m_toolManager )
|
|
m_toolManager->ShutdownAllTools();
|
|
|
|
if( GetBoard() )
|
|
GetBoard()->RemoveListener( m_appearancePanel );
|
|
|
|
delete m_selectionFilterPanel;
|
|
delete m_appearancePanel;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetBoard( BOARD* aBoard )
|
|
{
|
|
if( m_pcb )
|
|
m_pcb->ClearProject();
|
|
|
|
PCB_BASE_EDIT_FRAME::SetBoard( aBoard );
|
|
|
|
aBoard->SetProject( &Prj() );
|
|
aBoard->GetConnectivity()->Build( aBoard );
|
|
|
|
// reload the drawing-sheet
|
|
SetPageSettings( aBoard->GetPageSettings() );
|
|
}
|
|
|
|
|
|
BOARD_ITEM_CONTAINER* PCB_EDIT_FRAME::GetModel() const
|
|
{
|
|
return m_pcb;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetPageSettings( const PAGE_INFO& aPageSettings )
|
|
{
|
|
PCB_BASE_FRAME::SetPageSettings( aPageSettings );
|
|
|
|
// Prepare drawing-sheet template
|
|
DS_PROXY_VIEW_ITEM* drawingSheet = new DS_PROXY_VIEW_ITEM( IU_PER_MILS,
|
|
&m_pcb->GetPageSettings(),
|
|
m_pcb->GetProject(),
|
|
&m_pcb->GetTitleBlock() );
|
|
drawingSheet->SetSheetName( std::string( GetScreenDesc().mb_str() ) );
|
|
|
|
BASE_SCREEN* screen = GetScreen();
|
|
|
|
if( screen != NULL )
|
|
{
|
|
drawingSheet->SetPageNumber(TO_UTF8( screen->GetPageNumber() ) );
|
|
drawingSheet->SetSheetCount( screen->GetPageCount() );
|
|
}
|
|
|
|
if( BOARD* board = GetBoard() )
|
|
drawingSheet->SetFileName( TO_UTF8( board->GetFileName() ) );
|
|
|
|
// PCB_DRAW_PANEL_GAL takes ownership of the drawing-sheet
|
|
GetCanvas()->SetDrawingSheet( drawingSheet );
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::IsContentModified()
|
|
{
|
|
return GetScreen() && GetScreen()->IsModify();
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::isAutoSaveRequired() const
|
|
{
|
|
if( GetScreen() )
|
|
return GetScreen()->IsSave();
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
SELECTION& PCB_EDIT_FRAME::GetCurrentSelection()
|
|
{
|
|
return m_toolManager->GetTool<PCB_SELECTION_TOOL>()->GetSelection();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::setupTools()
|
|
{
|
|
// Create the manager and dispatcher & route draw panel events to the dispatcher
|
|
m_toolManager = new TOOL_MANAGER;
|
|
m_toolManager->SetEnvironment( m_pcb, GetCanvas()->GetView(),
|
|
GetCanvas()->GetViewControls(), config(), this );
|
|
m_actions = new PCB_ACTIONS();
|
|
m_toolDispatcher = new TOOL_DISPATCHER( m_toolManager, m_actions );
|
|
|
|
// Register tools
|
|
m_toolManager->RegisterTool( new COMMON_CONTROL );
|
|
m_toolManager->RegisterTool( new COMMON_TOOLS );
|
|
m_toolManager->RegisterTool( new PCB_SELECTION_TOOL );
|
|
m_toolManager->RegisterTool( new ZOOM_TOOL );
|
|
m_toolManager->RegisterTool( new PCB_PICKER_TOOL );
|
|
m_toolManager->RegisterTool( new ROUTER_TOOL );
|
|
m_toolManager->RegisterTool( new LENGTH_TUNER_TOOL );
|
|
m_toolManager->RegisterTool( new EDIT_TOOL );
|
|
m_toolManager->RegisterTool( new GLOBAL_EDIT_TOOL );
|
|
m_toolManager->RegisterTool( new PAD_TOOL );
|
|
m_toolManager->RegisterTool( new DRAWING_TOOL );
|
|
m_toolManager->RegisterTool( new PCB_POINT_EDITOR );
|
|
m_toolManager->RegisterTool( new PCB_CONTROL );
|
|
m_toolManager->RegisterTool( new BOARD_EDITOR_CONTROL );
|
|
m_toolManager->RegisterTool( new BOARD_INSPECTION_TOOL );
|
|
m_toolManager->RegisterTool( new BOARD_REANNOTATE_TOOL );
|
|
m_toolManager->RegisterTool( new ALIGN_DISTRIBUTE_TOOL );
|
|
m_toolManager->RegisterTool( new MICROWAVE_TOOL );
|
|
m_toolManager->RegisterTool( new POSITION_RELATIVE_TOOL );
|
|
m_toolManager->RegisterTool( new ZONE_FILLER_TOOL );
|
|
m_toolManager->RegisterTool( new AUTOPLACE_TOOL );
|
|
m_toolManager->RegisterTool( new DRC_TOOL );
|
|
m_toolManager->RegisterTool( new PCB_VIEWER_TOOLS );
|
|
m_toolManager->RegisterTool( new CONVERT_TOOL );
|
|
m_toolManager->RegisterTool( new GROUP_TOOL );
|
|
m_toolManager->InitTools();
|
|
|
|
// Run the selection tool, it is supposed to be always active
|
|
m_toolManager->InvokeTool( "pcbnew.InteractiveSelection" );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::setupUIConditions()
|
|
{
|
|
PCB_BASE_EDIT_FRAME::setupUIConditions();
|
|
|
|
ACTION_MANAGER* mgr = m_toolManager->GetActionManager();
|
|
PCB_EDITOR_CONDITIONS cond( this );
|
|
|
|
wxASSERT( mgr );
|
|
|
|
#define ENABLE( x ) ACTION_CONDITIONS().Enable( x )
|
|
#define CHECK( x ) ACTION_CONDITIONS().Check( x )
|
|
|
|
mgr->SetConditions( ACTIONS::save, ENABLE( SELECTION_CONDITIONS::ShowAlways ) );
|
|
mgr->SetConditions( ACTIONS::undo, ENABLE( cond.UndoAvailable() ) );
|
|
mgr->SetConditions( ACTIONS::redo, ENABLE( cond.RedoAvailable() ) );
|
|
|
|
mgr->SetConditions( ACTIONS::toggleGrid, CHECK( cond.GridVisible() ) );
|
|
mgr->SetConditions( ACTIONS::toggleCursorStyle, CHECK( cond.FullscreenCursor() ) );
|
|
mgr->SetConditions( ACTIONS::togglePolarCoords, CHECK( cond.PolarCoordinates() ) );
|
|
mgr->SetConditions( ACTIONS::millimetersUnits, CHECK( cond.Units( EDA_UNITS::MILLIMETRES ) ) );
|
|
mgr->SetConditions( ACTIONS::inchesUnits, CHECK( cond.Units( EDA_UNITS::INCHES ) ) );
|
|
mgr->SetConditions( ACTIONS::milsUnits, CHECK( cond.Units( EDA_UNITS::MILS ) ) );
|
|
|
|
mgr->SetConditions( ACTIONS::cut, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( ACTIONS::copy, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( ACTIONS::paste, ENABLE( SELECTION_CONDITIONS::Idle && cond.NoActiveTool() ) );
|
|
mgr->SetConditions( ACTIONS::pasteSpecial, ENABLE( SELECTION_CONDITIONS::Idle && cond.NoActiveTool() ) );
|
|
mgr->SetConditions( ACTIONS::selectAll, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( ACTIONS::doDelete, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( ACTIONS::duplicate, ENABLE( cond.HasItems() ) );
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::rotateCw, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::rotateCcw, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::group, ENABLE( SELECTION_CONDITIONS::MoreThan( 1 ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::ungroup, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::lock, ENABLE( cond.HasItems() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::unlock, ENABLE( cond.HasItems() ) );
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::padDisplayMode, CHECK( !cond.PadFillDisplay() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::viaDisplayMode, CHECK( !cond.ViaFillDisplay() ) );
|
|
mgr->SetConditions( PCB_ACTIONS::trackDisplayMode, CHECK( !cond.TrackFillDisplay() ) );
|
|
|
|
|
|
#if defined( KICAD_SCRIPTING_WXPYTHON )
|
|
auto pythonConsoleCond =
|
|
[] ( const SELECTION& )
|
|
{
|
|
if( IsWxPythonLoaded() )
|
|
{
|
|
wxWindow* console = PCB_EDIT_FRAME::findPythonConsole();
|
|
return console && console->IsShown();
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::showPythonConsole, CHECK( pythonConsoleCond ) );
|
|
#endif
|
|
|
|
auto enableZoneControlConition =
|
|
[this] ( const SELECTION& )
|
|
{
|
|
return GetBoard()->GetVisibleElements().Contains( LAYER_ZONES )
|
|
&& GetDisplayOptions().m_ZoneOpacity > 0.0;
|
|
};
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::zoneDisplayEnable,
|
|
ENABLE( enableZoneControlConition ).Check( cond.ZoneDisplayMode( ZONE_DISPLAY_MODE::SHOW_FILLED ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::zoneDisplayDisable,
|
|
ENABLE( enableZoneControlConition ).Check( cond.ZoneDisplayMode( ZONE_DISPLAY_MODE::SHOW_ZONE_OUTLINE ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::zoneDisplayOutlines,
|
|
ENABLE( enableZoneControlConition ).Check( cond.ZoneDisplayMode( ZONE_DISPLAY_MODE::SHOW_FILLED_OUTLINE ) ) );
|
|
|
|
auto enableBoardSetupCondition =
|
|
[this] ( const SELECTION& )
|
|
{
|
|
if( DRC_TOOL* tool = m_toolManager->GetTool<DRC_TOOL>() )
|
|
return !tool->IsDRCDialogShown();
|
|
|
|
return true;
|
|
};
|
|
|
|
auto boardFlippedCond =
|
|
[this]( const SELECTION& )
|
|
{
|
|
return GetCanvas()->GetView()->IsMirroredX();
|
|
};
|
|
|
|
auto layerManagerCond =
|
|
[this] ( const SELECTION& )
|
|
{
|
|
return LayerManagerShown();
|
|
};
|
|
|
|
auto highContrastCond =
|
|
[this] ( const SELECTION& )
|
|
{
|
|
return GetDisplayOptions().m_ContrastModeDisplay != HIGH_CONTRAST_MODE::NORMAL;
|
|
};
|
|
|
|
auto globalRatsnestCond =
|
|
[this] (const SELECTION& )
|
|
{
|
|
return GetDisplayOptions().m_ShowGlobalRatsnest;
|
|
};
|
|
|
|
auto curvedRatsnestCond =
|
|
[this] (const SELECTION& )
|
|
{
|
|
return GetDisplayOptions().m_DisplayRatsnestLinesCurved;
|
|
};
|
|
|
|
mgr->SetConditions( ACTIONS::highContrastMode, CHECK( highContrastCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::flipBoard, CHECK( boardFlippedCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::showLayersManager, CHECK( layerManagerCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::showRatsnest, CHECK( globalRatsnestCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::ratsnestLineMode, CHECK( curvedRatsnestCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::boardSetup , ENABLE( enableBoardSetupCondition ) );
|
|
|
|
|
|
auto isHighlightMode =
|
|
[this]( const SELECTION& )
|
|
{
|
|
ROUTER_TOOL* tool = m_toolManager->GetTool<ROUTER_TOOL>();
|
|
return tool->GetRouterMode() == PNS::RM_MarkObstacles;
|
|
};
|
|
|
|
auto isShoveMode =
|
|
[this]( const SELECTION& )
|
|
{
|
|
ROUTER_TOOL* tool = m_toolManager->GetTool<ROUTER_TOOL>();
|
|
return tool->GetRouterMode() == PNS::RM_Shove;
|
|
};
|
|
|
|
auto isWalkaroundMode =
|
|
[this]( const SELECTION& )
|
|
{
|
|
ROUTER_TOOL* tool = m_toolManager->GetTool<ROUTER_TOOL>();
|
|
return tool->GetRouterMode() == PNS::RM_Walkaround;
|
|
};
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::routerHighlightMode, CHECK( isHighlightMode ) );
|
|
mgr->SetConditions( PCB_ACTIONS::routerShoveMode, CHECK( isShoveMode ) );
|
|
mgr->SetConditions( PCB_ACTIONS::routerWalkaroundMode, CHECK( isWalkaroundMode ) );
|
|
|
|
auto haveNetCond =
|
|
[] ( const SELECTION& aSel )
|
|
{
|
|
for( EDA_ITEM* item : aSel )
|
|
{
|
|
if( BOARD_CONNECTED_ITEM* bci = dynamic_cast<BOARD_CONNECTED_ITEM*>( item ) )
|
|
{
|
|
if( bci->GetNetCode() > 0 )
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::showNet, ENABLE( haveNetCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::hideNet, ENABLE( haveNetCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::highlightNet, ENABLE( haveNetCond ) );
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::selectNet,
|
|
ENABLE( SELECTION_CONDITIONS::OnlyTypes( GENERAL_COLLECTOR::Tracks ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::deselectNet,
|
|
ENABLE( SELECTION_CONDITIONS::OnlyTypes( GENERAL_COLLECTOR::Tracks ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::selectSameSheet,
|
|
ENABLE( SELECTION_CONDITIONS::OnlyType( PCB_FOOTPRINT_T ) ) );
|
|
|
|
|
|
SELECTION_CONDITION singleZoneCond = SELECTION_CONDITIONS::Count( 1 ) &&
|
|
SELECTION_CONDITIONS::OnlyTypes( GENERAL_COLLECTOR::Zones );
|
|
|
|
SELECTION_CONDITION zoneMergeCond = SELECTION_CONDITIONS::MoreThan( 1 ) &&
|
|
PCB_SELECTION_CONDITIONS::SameNet( true ) &&
|
|
PCB_SELECTION_CONDITIONS::SameLayer();
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::zoneDuplicate, ENABLE( singleZoneCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::drawZoneCutout, ENABLE( singleZoneCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::drawSimilarZone, ENABLE( singleZoneCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::zoneMerge, ENABLE( zoneMergeCond ) );
|
|
mgr->SetConditions( PCB_ACTIONS::zoneFill, ENABLE( SELECTION_CONDITIONS::MoreThan( 0 ) ) );
|
|
mgr->SetConditions( PCB_ACTIONS::zoneUnfill, ENABLE( SELECTION_CONDITIONS::MoreThan( 0 ) ) );
|
|
|
|
mgr->SetConditions( PCB_ACTIONS::toggleLine45degMode, CHECK( cond.Line45degMode() ) );
|
|
|
|
|
|
#define CURRENT_TOOL( action ) mgr->SetConditions( action, CHECK( cond.CurrentTool( action ) ) )
|
|
|
|
// These tools can be used at any time to inspect the board
|
|
CURRENT_TOOL( ACTIONS::zoomTool );
|
|
CURRENT_TOOL( ACTIONS::measureTool );
|
|
CURRENT_TOOL( ACTIONS::selectionTool );
|
|
CURRENT_TOOL( PCB_ACTIONS::highlightNetTool );
|
|
CURRENT_TOOL( PCB_ACTIONS::localRatsnestTool );
|
|
|
|
|
|
auto isDrcRunning =
|
|
[this] ( const SELECTION& )
|
|
{
|
|
DRC_TOOL* tool = m_toolManager->GetTool<DRC_TOOL>();
|
|
return !tool->IsDRCRunning();
|
|
};
|
|
|
|
#define CURRENT_EDIT_TOOL( action ) mgr->SetConditions( action, ACTION_CONDITIONS().Check( cond.CurrentTool( action ) ).Enable( isDrcRunning ) )
|
|
|
|
// These tools edit the board, so they must be disabled during some operations
|
|
CURRENT_EDIT_TOOL( ACTIONS::deleteTool );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::placeFootprint );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::routeSingleTrack);
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::routeDiffPair);
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawVia );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawZone );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawRuleArea );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawLine );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawRectangle );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawCircle );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawArc );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawPolygon );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::placeText );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawAlignedDimension );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawOrthogonalDimension );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawCenterDimension );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drawLeader );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::placeTarget );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::drillOrigin );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::gridSetOrigin );
|
|
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::microwaveCreateLine );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::microwaveCreateGap );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::microwaveCreateStub );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::microwaveCreateStubArc );
|
|
CURRENT_EDIT_TOOL( PCB_ACTIONS::microwaveCreateFunctionShape );
|
|
|
|
#undef CURRENT_TOOL
|
|
#undef CURRENT_EDIT_TOOL
|
|
#undef ENABLE
|
|
#undef CHECK
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::OnQuit( wxCommandEvent& event )
|
|
{
|
|
if( event.GetId() == wxID_EXIT )
|
|
Kiway().OnKiCadExit();
|
|
|
|
if( event.GetId() == wxID_CLOSE || Kiface().IsSingle() )
|
|
Close( false );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::RecordDRCExclusions()
|
|
{
|
|
BOARD_DESIGN_SETTINGS& bds = GetBoard()->GetDesignSettings();
|
|
bds.m_DrcExclusions.clear();
|
|
|
|
for( PCB_MARKER* marker : GetBoard()->Markers() )
|
|
{
|
|
if( marker->IsExcluded() )
|
|
bds.m_DrcExclusions.insert( marker->Serialize() );
|
|
}
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ResolveDRCExclusions()
|
|
{
|
|
BOARD_COMMIT commit( this );
|
|
|
|
for( PCB_MARKER* marker : GetBoard()->ResolveDRCExclusions() )
|
|
commit.Add( marker );
|
|
|
|
commit.Push( wxEmptyString, false, false );
|
|
|
|
for( PCB_MARKER* marker : GetBoard()->Markers() )
|
|
{
|
|
if( marker->IsExcluded() )
|
|
{
|
|
GetCanvas()->GetView()->Remove( marker );
|
|
GetCanvas()->GetView()->Add( marker );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::canCloseWindow( wxCloseEvent& aEvent )
|
|
{
|
|
// Shutdown blocks must be determined and vetoed as early as possible
|
|
if( KIPLATFORM::APP::SupportsShutdownBlockReason() && aEvent.GetId() == wxEVT_QUERY_END_SESSION
|
|
&& IsContentModified() )
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if( IsContentModified() )
|
|
{
|
|
wxFileName fileName = GetBoard()->GetFileName();
|
|
wxString msg = _( "Save changes to \"%s\" before closing?" );
|
|
|
|
if( !HandleUnsavedChanges( this, wxString::Format( msg, fileName.GetFullName() ),
|
|
[&]() -> bool
|
|
{
|
|
return Files_io_from_id( ID_SAVE_BOARD );
|
|
} ) )
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
// Close modeless dialogs. They're trouble when they get destroyed after the frame and/or
|
|
// board.
|
|
wxWindow* open_dlg = wxWindow::FindWindowByName( DIALOG_DRC_WINDOW_NAME );
|
|
|
|
if( open_dlg )
|
|
open_dlg->Close( true );
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::doCloseWindow()
|
|
{
|
|
// On Windows 7 / 32 bits, on OpenGL mode only, Pcbnew crashes
|
|
// when closing this frame if a footprint was selected, and the footprint editor called
|
|
// to edit this footprint, and when closing pcbnew if this footprint is still selected
|
|
// See https://bugs.launchpad.net/kicad/+bug/1655858
|
|
// I think this is certainly a OpenGL event fired after frame deletion, so this workaround
|
|
// avoid the crash (JPC)
|
|
GetCanvas()->SetEvtHandlerEnabled( false );
|
|
|
|
GetCanvas()->StopDrawing();
|
|
|
|
// Delete the auto save file if it exists.
|
|
wxFileName fn = GetBoard()->GetFileName();
|
|
|
|
// Auto save file name is the normal file name prefixed with 'GetAutoSaveFilePrefix()'.
|
|
fn.SetName( GetAutoSaveFilePrefix() + fn.GetName() );
|
|
|
|
// When the auto save feature does not have write access to the board file path, it falls
|
|
// back to a platform specific user temporary file path.
|
|
if( !fn.IsOk() || !fn.IsDirWritable() )
|
|
fn.SetPath( wxFileName::GetTempDir() );
|
|
|
|
wxLogTrace( traceAutoSave, "Deleting auto save file <" + fn.GetFullPath() + ">" );
|
|
|
|
// Remove the auto save file on a normal close of Pcbnew.
|
|
if( fn.FileExists() && !wxRemoveFile( fn.GetFullPath() ) )
|
|
{
|
|
wxString msg = wxString::Format( _( "The auto save file \"%s\" could not be removed!" ),
|
|
fn.GetFullPath() );
|
|
wxMessageBox( msg, Pgm().App().GetAppName(), wxOK | wxICON_ERROR, this );
|
|
}
|
|
|
|
// Make sure local settings are persisted
|
|
SaveProjectSettings();
|
|
|
|
// Do not show the layer manager during closing to avoid flicker
|
|
// on some platforms (Windows) that generate useless redraw of items in
|
|
// the Layer Manger
|
|
if( m_show_layer_manager_tools )
|
|
m_auimgr.GetPane( "LayersManager" ).Show( false );
|
|
|
|
// Unlink the old project if needed
|
|
GetBoard()->ClearProject();
|
|
|
|
// Delete board structs and undo/redo lists, to avoid crash on exit
|
|
// when deleting some structs (mainly in undo/redo lists) too late
|
|
Clear_Pcb( false, true );
|
|
|
|
// do not show the window because ScreenPcb will be deleted and we do not
|
|
// want any paint event
|
|
Show( false );
|
|
|
|
PCB_BASE_EDIT_FRAME::doCloseWindow();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ActivateGalCanvas()
|
|
{
|
|
PCB_BASE_EDIT_FRAME::ActivateGalCanvas();
|
|
GetCanvas()->UpdateColors();
|
|
GetCanvas()->Refresh();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ShowBoardSetupDialog( const wxString& aInitialPage )
|
|
{
|
|
// Make sure everything's up-to-date
|
|
GetBoard()->BuildListOfNets();
|
|
|
|
DIALOG_BOARD_SETUP dlg( this );
|
|
|
|
if( !aInitialPage.IsEmpty() )
|
|
dlg.SetInitialPage( aInitialPage, wxEmptyString );
|
|
|
|
if( dlg.ShowQuasiModal() == wxID_OK )
|
|
{
|
|
Prj().GetProjectFile().NetSettings().ResolveNetClassAssignments( true );
|
|
|
|
GetBoard()->SynchronizeNetsAndNetClasses();
|
|
SaveProjectSettings();
|
|
|
|
Kiway().CommonSettingsChanged( false, true );
|
|
|
|
const PCB_DISPLAY_OPTIONS& opts = GetDisplayOptions();
|
|
|
|
if( opts.m_ShowTrackClearanceMode || opts.m_DisplayPadClearance )
|
|
{
|
|
// Update clearance outlines
|
|
GetCanvas()->GetView()->UpdateAllItemsConditionally( KIGFX::REPAINT,
|
|
[&]( KIGFX::VIEW_ITEM* aItem ) -> bool
|
|
{
|
|
TRACK* track = dynamic_cast<TRACK*>( aItem );
|
|
PAD* pad = dynamic_cast<PAD*>( aItem );
|
|
|
|
// TRACK is the base class of VIA and ARC so we don't need to
|
|
// check them independently
|
|
|
|
return ( track && opts.m_ShowTrackClearanceMode )
|
|
|| ( pad && opts.m_DisplayPadClearance );
|
|
} );
|
|
}
|
|
|
|
GetCanvas()->Refresh();
|
|
|
|
UpdateUserInterface();
|
|
ReCreateAuxiliaryToolbar();
|
|
m_toolManager->ResetTools( TOOL_BASE::MODEL_RELOAD );
|
|
|
|
//this event causes the routing tool to reload its design rules information
|
|
TOOL_EVENT toolEvent( TC_COMMAND, TA_MODEL_CHANGE, AS_ACTIVE );
|
|
toolEvent.SetHasPosition( false );
|
|
m_toolManager->ProcessEvent( toolEvent );
|
|
}
|
|
|
|
GetCanvas()->SetFocus();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::LoadSettings( APP_SETTINGS_BASE* aCfg )
|
|
{
|
|
PCB_BASE_FRAME::LoadSettings( aCfg );
|
|
|
|
PCBNEW_SETTINGS* cfg = dynamic_cast<PCBNEW_SETTINGS*>( aCfg );
|
|
wxASSERT( cfg );
|
|
|
|
if( cfg )
|
|
{
|
|
m_rotationAngle = cfg->m_RotationAngle;
|
|
m_show_layer_manager_tools = cfg->m_AuiPanels.show_layer_manager;
|
|
m_showPageLimits = cfg->m_ShowPageLimits;
|
|
}
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SaveSettings( APP_SETTINGS_BASE* aCfg )
|
|
{
|
|
PCB_BASE_FRAME::SaveSettings( aCfg );
|
|
|
|
auto cfg = dynamic_cast<PCBNEW_SETTINGS*>( aCfg );
|
|
wxASSERT( cfg );
|
|
|
|
if( cfg )
|
|
{
|
|
cfg->m_RotationAngle = m_rotationAngle;
|
|
cfg->m_AuiPanels.show_layer_manager = m_show_layer_manager_tools;
|
|
cfg->m_AuiPanels.right_panel_width = m_appearancePanel->GetSize().x;
|
|
cfg->m_AuiPanels.appearance_panel_tab = m_appearancePanel->GetTabIndex();
|
|
cfg->m_ShowPageLimits = m_showPageLimits;
|
|
}
|
|
|
|
GetSettingsManager()->SaveColorSettings( GetColorSettings(), "board" );
|
|
}
|
|
|
|
|
|
COLOR4D PCB_EDIT_FRAME::GetGridColor()
|
|
{
|
|
return GetColorSettings()->GetColor( LAYER_GRID );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetGridColor( COLOR4D aColor )
|
|
{
|
|
|
|
GetColorSettings()->SetColor( LAYER_GRID, aColor );
|
|
GetCanvas()->GetGAL()->SetGridColor( aColor );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetActiveLayer( PCB_LAYER_ID aLayer )
|
|
{
|
|
PCB_LAYER_ID oldLayer = GetActiveLayer();
|
|
|
|
if( oldLayer == aLayer )
|
|
return;
|
|
|
|
PCB_BASE_FRAME::SetActiveLayer( aLayer );
|
|
|
|
m_appearancePanel->OnLayerChanged();
|
|
|
|
m_toolManager->RunAction( PCB_ACTIONS::layerChanged ); // notify other tools
|
|
GetCanvas()->SetFocus(); // allow capture of hotkeys
|
|
GetCanvas()->SetHighContrastLayer( aLayer );
|
|
|
|
GetCanvas()->GetView()->UpdateAllItemsConditionally( KIGFX::REPAINT,
|
|
[&]( KIGFX::VIEW_ITEM* aItem ) -> bool
|
|
{
|
|
if( VIA* via = dynamic_cast<VIA*>( aItem ) )
|
|
{
|
|
// Vias on a restricted layer set must be redrawn when the active layer
|
|
// is changed
|
|
return ( via->GetViaType() == VIATYPE::BLIND_BURIED ||
|
|
via->GetViaType() == VIATYPE::MICROVIA );
|
|
}
|
|
else if( PAD* pad = dynamic_cast<PAD*>( aItem ) )
|
|
{
|
|
// Clearances could be layer-dependent so redraw them when the active layer
|
|
// is changed
|
|
if( GetDisplayOptions().m_DisplayPadClearance )
|
|
{
|
|
// Round-corner rects are expensive to draw, but are mostly found on
|
|
// SMD pads which only need redrawing on an active-to-not-active
|
|
// switch.
|
|
if( pad->GetAttribute() == PAD_ATTRIB_SMD )
|
|
{
|
|
if( ( oldLayer == F_Cu || aLayer == F_Cu ) && pad->IsOnLayer( F_Cu ) )
|
|
return true;
|
|
|
|
if( ( oldLayer == B_Cu || aLayer == B_Cu ) && pad->IsOnLayer( B_Cu ) )
|
|
return true;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
else if( TRACK* track = dynamic_cast<TRACK*>( aItem ) )
|
|
{
|
|
// Clearances could be layer-dependent so redraw them when the active layer
|
|
// is changed
|
|
if( GetDisplayOptions().m_ShowTrackClearanceMode )
|
|
{
|
|
// Tracks aren't particularly expensive to draw, but it's an easy check.
|
|
return track->IsOnLayer( oldLayer ) || track->IsOnLayer( aLayer );
|
|
}
|
|
}
|
|
|
|
return false;
|
|
} );
|
|
|
|
GetCanvas()->Refresh();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::onBoardLoaded()
|
|
{
|
|
DRC_TOOL* drcTool = m_toolManager->GetTool<DRC_TOOL>();
|
|
|
|
try
|
|
{
|
|
drcTool->GetDRCEngine()->InitEngine( GetDesignRulesPath() );
|
|
}
|
|
catch( PARSE_ERROR& )
|
|
{
|
|
// Not sure this is the best place to tell the user their rules are buggy, so
|
|
// we'll stay quiet for now. Feel free to revisit this decision....
|
|
}
|
|
|
|
UpdateTitle();
|
|
|
|
wxFileName fn = GetBoard()->GetFileName();
|
|
|
|
// Display a warning that the file is read only
|
|
if( fn.FileExists() && !fn.IsFileWritable() )
|
|
{
|
|
m_infoBar->RemoveAllButtons();
|
|
m_infoBar->AddCloseButton();
|
|
m_infoBar->ShowMessage( _( "Board file is read only." ), wxICON_WARNING );
|
|
}
|
|
|
|
ReCreateLayerBox();
|
|
|
|
// Sync layer and item visibility
|
|
GetCanvas()->SyncLayersVisibility( m_pcb );
|
|
|
|
SetElementVisibility( LAYER_RATSNEST, GetDisplayOptions().m_ShowGlobalRatsnest );
|
|
|
|
m_appearancePanel->OnBoardChanged();
|
|
|
|
// Apply saved display state to the appearance panel after it has been set up
|
|
PROJECT_LOCAL_SETTINGS& localSettings = Prj().GetLocalSettings();
|
|
|
|
m_appearancePanel->ApplyLayerPreset( localSettings.m_ActiveLayerPreset );
|
|
|
|
if( GetBoard()->GetDesignSettings().IsLayerEnabled( localSettings.m_ActiveLayer ) )
|
|
SetActiveLayer( localSettings.m_ActiveLayer );
|
|
|
|
// Updates any auto dimensions and the auxiliary toolbar tracks/via sizes
|
|
unitsChangeRefresh();
|
|
|
|
// Display the loaded board:
|
|
Zoom_Automatique( false );
|
|
|
|
// Invalidate painting as loading the DRC engine will cause clearances to become valid
|
|
GetCanvas()->GetView()->UpdateAllItems( KIGFX::ALL );
|
|
|
|
Refresh();
|
|
|
|
SetMsgPanel( GetBoard() );
|
|
SetStatusText( wxEmptyString );
|
|
|
|
KIPLATFORM::APP::SetShutdownBlockReason( this, _( "PCB file changes are unsaved" ) );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::OnDisplayOptionsChanged()
|
|
{
|
|
m_appearancePanel->UpdateDisplayOptions();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::OnUpdateLayerAlpha( wxUpdateUIEvent & )
|
|
{
|
|
m_appearancePanel->OnLayerAlphaChanged();
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::IsElementVisible( GAL_LAYER_ID aElement ) const
|
|
{
|
|
return GetBoard()->IsElementVisible( aElement );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetElementVisibility( GAL_LAYER_ID aElement, bool aNewState )
|
|
{
|
|
// Force the RATSNEST visible
|
|
if( aElement == LAYER_RATSNEST )
|
|
GetCanvas()->GetView()->SetLayerVisible( aElement, true );
|
|
else
|
|
GetCanvas()->GetView()->SetLayerVisible( aElement , aNewState );
|
|
|
|
GetBoard()->SetElementVisibility( aElement, aNewState );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ShowChangedLanguage()
|
|
{
|
|
// call my base class
|
|
PCB_BASE_EDIT_FRAME::ShowChangedLanguage();
|
|
|
|
wxAuiPaneInfo& pane_info = m_auimgr.GetPane( m_appearancePanel );
|
|
pane_info.Caption( _( "Appearance" ) );
|
|
m_auimgr.Update();
|
|
|
|
m_appearancePanel->OnBoardChanged();
|
|
}
|
|
|
|
|
|
wxString PCB_EDIT_FRAME::GetLastPath( LAST_PATH_TYPE aType )
|
|
{
|
|
PROJECT_FILE& project = Prj().GetProjectFile();
|
|
|
|
if( project.m_PcbLastPath[ aType ].IsEmpty() )
|
|
return wxEmptyString;
|
|
|
|
wxFileName absoluteFileName = project.m_PcbLastPath[ aType ];
|
|
wxFileName pcbFileName = GetBoard()->GetFileName();
|
|
|
|
absoluteFileName.MakeAbsolute( pcbFileName.GetPath() );
|
|
return absoluteFileName.GetFullPath();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::SetLastPath( LAST_PATH_TYPE aType, const wxString& aLastPath )
|
|
{
|
|
PROJECT_FILE& project = Prj().GetProjectFile();
|
|
|
|
wxFileName relativeFileName = aLastPath;
|
|
wxFileName pcbFileName = GetBoard()->GetFileName();
|
|
|
|
relativeFileName.MakeRelativeTo( pcbFileName.GetPath() );
|
|
|
|
if( relativeFileName.GetFullPath() != project.m_PcbLastPath[ aType ] )
|
|
{
|
|
project.m_PcbLastPath[ aType ] = relativeFileName.GetFullPath();
|
|
SaveProjectSettings();
|
|
}
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::OnModify( )
|
|
{
|
|
PCB_BASE_FRAME::OnModify();
|
|
|
|
Update3DView( true );
|
|
|
|
if( !GetTitle().StartsWith( "*" ) )
|
|
UpdateTitle();
|
|
|
|
m_ZoneFillsDirty = true;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ExportSVG( wxCommandEvent& event )
|
|
{
|
|
InvokeExportSVG( this, GetBoard() );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::UpdateTitle()
|
|
{
|
|
wxFileName fn = GetBoard()->GetFileName();
|
|
bool readOnly = false;
|
|
bool unsaved = false;
|
|
|
|
if( fn.IsOk() && fn.FileExists() )
|
|
readOnly = !fn.IsFileWritable();
|
|
else
|
|
unsaved = true;
|
|
|
|
SetTitle( wxString::Format( wxT( "%s%s %s%s\u2014 " ) + _( "PCB Editor" ),
|
|
IsContentModified() ? "*" : "",
|
|
fn.GetName(),
|
|
readOnly ? _( "[Read Only]" ) + wxS( " " ) : "",
|
|
unsaved ? _( "[Unsaved]" ) + wxS( " " ) : "" ) );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::UpdateUserInterface()
|
|
{
|
|
// Update the layer manager and other widgets from the board setup
|
|
// (layer and items visibility, colors ...)
|
|
|
|
// Rebuild list of nets (full ratsnest rebuild)
|
|
GetBoard()->BuildConnectivity();
|
|
Compile_Ratsnest( true );
|
|
|
|
// Update info shown by the horizontal toolbars
|
|
ReCreateLayerBox();
|
|
|
|
LSET activeLayers = GetBoard()->GetEnabledLayers();
|
|
|
|
if( !activeLayers.test( GetActiveLayer() ) )
|
|
SetActiveLayer( activeLayers.Seq().front() );
|
|
|
|
m_SelLayerBox->SetLayerSelection( GetActiveLayer() );
|
|
|
|
ENUM_MAP<PCB_LAYER_ID>& layerEnum = ENUM_MAP<PCB_LAYER_ID>::Instance();
|
|
|
|
layerEnum.Choices().Clear();
|
|
layerEnum.Undefined( UNDEFINED_LAYER );
|
|
|
|
for( LSEQ seq = LSET::AllLayersMask().Seq(); seq; ++seq )
|
|
{
|
|
// Canonical name
|
|
layerEnum.Map( *seq, LSET::Name( *seq ) );
|
|
// User name
|
|
layerEnum.Map( *seq, GetBoard()->GetLayerName( *seq ) );
|
|
}
|
|
|
|
// Sync visibility with canvas
|
|
KIGFX::VIEW* view = GetCanvas()->GetView();
|
|
LSET visible = GetBoard()->GetVisibleLayers();
|
|
|
|
for( PCB_LAYER_ID layer : LSET::AllLayersMask().Seq() )
|
|
view->SetLayerVisible( layer, visible.Contains( layer ) );
|
|
|
|
// Stackup and/or color theme may have changed
|
|
m_appearancePanel->OnBoardChanged();
|
|
}
|
|
|
|
|
|
#if defined( KICAD_SCRIPTING_WXPYTHON )
|
|
|
|
void PCB_EDIT_FRAME::ScriptingConsoleEnableDisable()
|
|
{
|
|
wxWindow * pythonPanelFrame = findPythonConsole();
|
|
bool pythonPanelShown = true;
|
|
|
|
if( pythonPanelFrame == NULL )
|
|
pythonPanelFrame = CreatePythonShellWindow( this, pythonConsoleNameId() );
|
|
else
|
|
pythonPanelShown = ! pythonPanelFrame->IsShown();
|
|
|
|
if( pythonPanelFrame )
|
|
pythonPanelFrame->Show( pythonPanelShown );
|
|
else
|
|
wxMessageBox( wxT( "Error: unable to create the Python Console" ) );
|
|
}
|
|
#endif
|
|
|
|
|
|
void PCB_EDIT_FRAME::SwitchCanvas( EDA_DRAW_PANEL_GAL::GAL_TYPE aCanvasType )
|
|
{
|
|
// switches currently used canvas (Cairo / OpenGL).
|
|
PCB_BASE_FRAME::SwitchCanvas( aCanvasType );
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ToPlotter( int aID )
|
|
{
|
|
PCB_PLOT_PARAMS plotSettings = GetPlotSettings();
|
|
|
|
switch( aID )
|
|
{
|
|
case ID_GEN_PLOT_GERBER:
|
|
plotSettings.SetFormat( PLOT_FORMAT::GERBER );
|
|
break;
|
|
case ID_GEN_PLOT_DXF:
|
|
plotSettings.SetFormat( PLOT_FORMAT::DXF );
|
|
break;
|
|
case ID_GEN_PLOT_HPGL:
|
|
plotSettings.SetFormat( PLOT_FORMAT::HPGL );
|
|
break;
|
|
case ID_GEN_PLOT_PDF:
|
|
plotSettings.SetFormat( PLOT_FORMAT::PDF );
|
|
break;
|
|
case ID_GEN_PLOT_PS:
|
|
plotSettings.SetFormat( PLOT_FORMAT::POST );
|
|
break;
|
|
case ID_GEN_PLOT: /* keep the previous setup */ break;
|
|
default:
|
|
wxFAIL_MSG( "ToPlotter(): unexpected plot type" ); break;
|
|
break;
|
|
}
|
|
|
|
SetPlotSettings( plotSettings );
|
|
|
|
// Force rebuild the dialog if currently open because the old dialog can be not up to date
|
|
// if the board (or units) has changed
|
|
wxWindow* dlg = wxWindow::FindWindowByName( DLG_WINDOW_NAME );
|
|
|
|
if( dlg )
|
|
dlg->Destroy();
|
|
|
|
dlg = new DIALOG_PLOT( this );
|
|
dlg->Show( true );
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::TestStandalone()
|
|
{
|
|
if( Kiface().IsSingle() )
|
|
return false;
|
|
|
|
// Update PCB requires a netlist. Therefore the schematic editor must be running
|
|
// If this is not the case, open the schematic editor
|
|
KIWAY_PLAYER* frame = Kiway().Player( FRAME_SCH, true );
|
|
|
|
if( !frame->IsShown() )
|
|
{
|
|
wxFileName fn( Prj().GetProjectPath(), Prj().GetProjectName(),
|
|
KiCadSchematicFileExtension );
|
|
|
|
// Maybe the file hasn't been converted to the new s-expression file format so
|
|
// see if the legacy schematic file is still in play.
|
|
if( !fn.FileExists() )
|
|
{
|
|
fn.SetExt( LegacySchematicFileExtension );
|
|
|
|
if( !fn.FileExists() )
|
|
{
|
|
DisplayError( this, _( "The schematic for this board cannot be found." ) );
|
|
return false;
|
|
}
|
|
}
|
|
|
|
frame->OpenProjectFiles( std::vector<wxString>( 1, fn.GetFullPath() ) );
|
|
|
|
// we show the schematic editor frame, because do not show is seen as
|
|
// a not yet opened schematic by Kicad manager, which is not the case
|
|
frame->Show( true );
|
|
|
|
// bring ourselves back to the front
|
|
Raise();
|
|
}
|
|
|
|
return true; //Success!
|
|
}
|
|
|
|
|
|
//
|
|
// Sends a Netlist packet to eeSchema.
|
|
// The reply is in aNetlist so it is destroyed by this
|
|
//
|
|
bool PCB_EDIT_FRAME::ReannotateSchematic( std::string& aNetlist )
|
|
{
|
|
Kiway().ExpressMail( FRAME_SCH, MAIL_REANNOTATE, aNetlist, this );
|
|
return true;
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::FetchNetlistFromSchematic( NETLIST& aNetlist, const wxString& aAnnotateMessage )
|
|
{
|
|
if( !TestStandalone() )
|
|
{
|
|
DisplayErrorMessage( this, _( "Cannot update the PCB because PCB editor is opened in "
|
|
"stand-alone mode. In order to create or update PCBs from "
|
|
"schematics, you must launch the KiCad project manager and "
|
|
"create a project." ) );
|
|
return false; // Not in standalone mode
|
|
}
|
|
|
|
Raise(); // Show
|
|
|
|
std::string payload( aAnnotateMessage );
|
|
|
|
Kiway().ExpressMail( FRAME_SCH, MAIL_SCH_GET_NETLIST, payload, this );
|
|
|
|
if( payload == aAnnotateMessage )
|
|
{
|
|
Raise();
|
|
DisplayErrorMessage( this, aAnnotateMessage );
|
|
return false;
|
|
}
|
|
|
|
try
|
|
{
|
|
auto lineReader = new STRING_LINE_READER( payload, _( "Eeschema netlist" ) );
|
|
KICAD_NETLIST_READER netlistReader( lineReader, &aNetlist );
|
|
netlistReader.LoadNetlist();
|
|
}
|
|
catch( const IO_ERROR& )
|
|
{
|
|
Raise();
|
|
assert( false ); // should never happen
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::RunEeschema()
|
|
{
|
|
wxString msg;
|
|
wxFileName schematic( Prj().GetProjectPath(), Prj().GetProjectName(), KiCadSchematicFileExtension );
|
|
|
|
if( !schematic.FileExists() )
|
|
{
|
|
wxFileName legacySchematic( Prj().GetProjectPath(), Prj().GetProjectName(), LegacySchematicFileExtension );
|
|
|
|
if( legacySchematic.FileExists() )
|
|
{
|
|
schematic = legacySchematic;
|
|
}
|
|
else
|
|
{
|
|
msg.Printf( _( "Schematic file \"%s\" not found." ), schematic.GetFullPath() );
|
|
wxMessageBox( msg, _( "KiCad Error" ), wxOK | wxICON_ERROR, this );
|
|
return;
|
|
}
|
|
}
|
|
|
|
if( Kiface().IsSingle() )
|
|
{
|
|
wxString filename = wxT( "\"" ) + schematic.GetFullPath( wxPATH_NATIVE ) + wxT( "\"" );
|
|
ExecuteFile( this, EESCHEMA_EXE, filename );
|
|
}
|
|
else
|
|
{
|
|
KIWAY_PLAYER* frame = Kiway().Player( FRAME_SCH, false );
|
|
|
|
// Please: note: DIALOG_EDIT_LIBENTRY_FIELDS_IN_LIB::initBuffers() calls
|
|
// Kiway.Player( FRAME_SCH, true )
|
|
// therefore, the schematic editor is sometimes running, but the schematic project
|
|
// is not loaded, if the library editor was called, and the dialog field editor was used.
|
|
// On linux, it happens the first time the schematic editor is launched, if
|
|
// library editor was running, and the dialog field editor was open
|
|
// On Windows, it happens always after the library editor was called,
|
|
// and the dialog field editor was used
|
|
if( !frame )
|
|
{
|
|
try
|
|
{
|
|
frame = Kiway().Player( FRAME_SCH, true );
|
|
}
|
|
catch( const IO_ERROR& err )
|
|
{
|
|
wxMessageBox( _( "Eeschema failed to load:\n" ) + err.What(),
|
|
_( "KiCad Error" ), wxOK | wxICON_ERROR, this );
|
|
return;
|
|
}
|
|
}
|
|
|
|
if( !frame->IsShown() ) // the frame exists, (created by the dialog field editor)
|
|
// but no project loaded.
|
|
{
|
|
frame->OpenProjectFiles( std::vector<wxString>( 1, schematic.GetFullPath() ) );
|
|
frame->Show( true );
|
|
}
|
|
|
|
// On Windows, Raise() does not bring the window on screen, when iconized or not shown
|
|
// On linux, Raise() brings the window on screen, but this code works fine
|
|
if( frame->IsIconized() )
|
|
{
|
|
frame->Iconize( false );
|
|
// If an iconized frame was created by Pcbnew, Iconize( false ) is not enough
|
|
// to show the frame at its normal size: Maximize should be called.
|
|
frame->Maximize( false );
|
|
}
|
|
|
|
frame->Raise();
|
|
}
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::PythonPluginsReload()
|
|
{
|
|
// Reload Python plugins if they are newer than the already loaded, and load new plugins
|
|
#if defined(KICAD_SCRIPTING)
|
|
#if defined(KICAD_SCRIPTING_ACTION_MENU)
|
|
// Remove all action plugins so that we don't keep references to old versions
|
|
ACTION_PLUGINS::UnloadAll();
|
|
#endif
|
|
|
|
// Reload plugin list: reload Python plugins if they are newer than the already loaded,
|
|
// and load new plugins
|
|
PythonPluginsReloadBase();
|
|
|
|
#if defined(KICAD_SCRIPTING_ACTION_MENU)
|
|
// Action plugins can be modified, therefore the plugins menu must be updated:
|
|
ReCreateMenuBar();
|
|
// Recreate top toolbar to add action plugin buttons
|
|
ReCreateHToolbar();
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::PythonPluginsShowFolder()
|
|
{
|
|
#if defined(KICAD_SCRIPTING)
|
|
#ifdef __WXMAC__
|
|
wxString msg;
|
|
|
|
// Quote in case there are spaces in the path.
|
|
msg.Printf( "open \"%s\"", PyPluginsPath( true ) );
|
|
|
|
system( msg.c_str() );
|
|
#else
|
|
wxString pypath( PyPluginsPath( true ) );
|
|
|
|
// Quote in case there are spaces in the path.
|
|
AddDelimiterString( pypath );
|
|
|
|
wxLaunchDefaultApplication( pypath );
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::PythonSyncEnvironmentVariables()
|
|
{
|
|
#if defined( KICAD_SCRIPTING )
|
|
const ENV_VAR_MAP& vars = Pgm().GetLocalEnvVariables();
|
|
|
|
// Set the environment variables for python scripts
|
|
// note: the strint will be encoded UTF8 for python env
|
|
for( auto& var : vars )
|
|
pcbnewUpdatePythonEnvVar( var.first, var.second.GetValue() );
|
|
|
|
// Because the env vars can de modifed by the python scripts (rewritten in UTF8),
|
|
// regenerate them (in unicode) for our normal environment
|
|
for( auto& var : vars )
|
|
wxSetEnv( var.first, var.second.GetValue() );
|
|
#endif
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::PythonSyncProjectName()
|
|
{
|
|
#if defined( KICAD_SCRIPTING )
|
|
wxString evValue;
|
|
wxGetEnv( PROJECT_VAR_NAME, &evValue );
|
|
pcbnewUpdatePythonEnvVar( wxString( PROJECT_VAR_NAME ).ToStdString(), evValue );
|
|
|
|
// Because PROJECT_VAR_NAME can be modifed by the python scripts (rewritten in UTF8),
|
|
// regenerate it (in unicode) for our normal environment
|
|
wxSetEnv( PROJECT_VAR_NAME, evValue );
|
|
#endif
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ShowFootprintPropertiesDialog( FOOTPRINT* aFootprint )
|
|
{
|
|
if( aFootprint == NULL )
|
|
return;
|
|
|
|
DIALOG_FOOTPRINT_PROPERTIES dlg( this, aFootprint );
|
|
|
|
// We use quasi modal to allow displaying help dialogs.
|
|
dlg.ShowQuasiModal();
|
|
DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_RETVALUE retvalue = dlg.GetReturnValue();
|
|
/*
|
|
* retvalue =
|
|
* FP_PROPS_UPDATE_FP to show Update Footprints dialog
|
|
* FP_PROPS_CHANGE_FP to show Chanage Footprints dialog
|
|
* FP_PROPS_OK for normal edit
|
|
* FP_PROPS_CANCEL if aborted
|
|
* FP_PROPS_EDIT_BOARD_FP to load board footprint into Footprint Editor
|
|
* FP_PROPS_EDIT_LIBRARY_FP to load library footprint into Footprint Editor
|
|
*/
|
|
|
|
if( retvalue == DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_OK )
|
|
{
|
|
// If something edited, push a refresh request
|
|
GetCanvas()->Refresh();
|
|
}
|
|
else if( retvalue == DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_EDIT_BOARD_FP )
|
|
{
|
|
auto editor = (FOOTPRINT_EDIT_FRAME*) Kiway().Player( FRAME_FOOTPRINT_EDITOR, true );
|
|
|
|
editor->LoadFootprintFromBoard( aFootprint );
|
|
|
|
editor->Show( true );
|
|
editor->Raise(); // Iconize( false );
|
|
}
|
|
|
|
else if( retvalue == DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_EDIT_LIBRARY_FP )
|
|
{
|
|
auto editor = (FOOTPRINT_EDIT_FRAME*) Kiway().Player( FRAME_FOOTPRINT_EDITOR, true );
|
|
|
|
editor->LoadFootprintFromLibrary( aFootprint->GetFPID() );
|
|
|
|
editor->Show( true );
|
|
editor->Raise(); // Iconize( false );
|
|
}
|
|
|
|
else if( retvalue == DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_UPDATE_FP )
|
|
{
|
|
ShowExchangeFootprintsDialog( aFootprint, true, true );
|
|
}
|
|
|
|
else if( retvalue == DIALOG_FOOTPRINT_PROPERTIES::FP_PROPS_CHANGE_FP )
|
|
{
|
|
ShowExchangeFootprintsDialog( aFootprint, false, true );
|
|
}
|
|
}
|
|
|
|
|
|
int PCB_EDIT_FRAME::ShowExchangeFootprintsDialog( FOOTPRINT* aFootprint, bool aUpdateMode,
|
|
bool aSelectedMode )
|
|
{
|
|
DIALOG_EXCHANGE_FOOTPRINTS dialog( this, aFootprint, aUpdateMode, aSelectedMode );
|
|
|
|
return dialog.ShowQuasiModal();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::CommonSettingsChanged( bool aEnvVarsChanged, bool aTextVarsChanged )
|
|
{
|
|
PCB_BASE_EDIT_FRAME::CommonSettingsChanged( aEnvVarsChanged, aTextVarsChanged );
|
|
|
|
GetAppearancePanel()->OnColorThemeChanged();
|
|
|
|
// Netclass definitions could have changed, either by us or by Eeschema
|
|
DRC_TOOL* drcTool = m_toolManager->GetTool<DRC_TOOL>();
|
|
WX_INFOBAR* infobar = GetInfoBar();
|
|
|
|
try
|
|
{
|
|
drcTool->GetDRCEngine()->InitEngine( GetDesignRulesPath() );
|
|
infobar->Hide();
|
|
}
|
|
catch( PARSE_ERROR& )
|
|
{
|
|
wxHyperlinkCtrl* button = new wxHyperlinkCtrl( infobar, wxID_ANY, _( "Edit design rules" ),
|
|
wxEmptyString );
|
|
|
|
button->Bind( wxEVT_COMMAND_HYPERLINK, std::function<void( wxHyperlinkEvent& aEvent )>(
|
|
[&]( wxHyperlinkEvent& aEvent )
|
|
{
|
|
ShowBoardSetupDialog( _( "Custom Rules" ) );
|
|
} ) );
|
|
|
|
infobar->RemoveAllButtons();
|
|
infobar->AddButton( button );
|
|
infobar->AddCloseButton();
|
|
infobar->ShowMessage( _( "Could not compile custom design rules." ), wxICON_ERROR );
|
|
}
|
|
|
|
// Update the environment variables in the Python interpreter
|
|
if( aEnvVarsChanged )
|
|
PythonSyncEnvironmentVariables();
|
|
|
|
Layout();
|
|
SendSizeEvent();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ThemeChanged()
|
|
{
|
|
PCB_BASE_EDIT_FRAME::ThemeChanged();
|
|
|
|
PythonPluginsReload();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::ProjectChanged()
|
|
{
|
|
PythonSyncProjectName();
|
|
}
|
|
|
|
|
|
bool ExportBoardToHyperlynx( BOARD* aBoard, const wxFileName& aPath );
|
|
|
|
|
|
void PCB_EDIT_FRAME::OnExportHyperlynx( wxCommandEvent& event )
|
|
{
|
|
wxString wildcard = wxT( "*.hyp" );
|
|
wxFileName fn = GetBoard()->GetFileName();
|
|
|
|
fn.SetExt( wxT("hyp") );
|
|
|
|
wxFileDialog dlg( this, _( "Export Hyperlynx Layout" ), fn.GetPath(), fn.GetFullName(),
|
|
wildcard, wxFD_SAVE | wxFD_OVERWRITE_PROMPT );
|
|
|
|
if( dlg.ShowModal() != wxID_OK )
|
|
return;
|
|
|
|
fn = dlg.GetPath();
|
|
|
|
// always enforce filename extension, user may not have entered it.
|
|
fn.SetExt( wxT( "hyp" ) );
|
|
|
|
ExportBoardToHyperlynx( GetBoard(), fn );
|
|
}
|
|
|
|
|
|
wxString PCB_EDIT_FRAME::GetCurrentFileName() const
|
|
{
|
|
return GetBoard()->GetFileName();
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::LayerManagerShown()
|
|
{
|
|
return m_auimgr.GetPane( "LayersManager" ).IsShown();
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::onSize( wxSizeEvent& aEvent )
|
|
{
|
|
if( IsShown() )
|
|
{
|
|
// We only need this until the frame is done resizing and the final client size is
|
|
// established.
|
|
Unbind( wxEVT_SIZE, &PCB_EDIT_FRAME::onSize, this );
|
|
GetToolManager()->RunAction( ACTIONS::zoomFitScreen, true );
|
|
}
|
|
|
|
// Skip() is called in the base class.
|
|
EDA_DRAW_FRAME::OnSize( aEvent );
|
|
}
|