/* * 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 * Copyright (C) 2013 Wayne Stambaugh * Copyright (C) 2013-2019 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 . */ #include #include #include #include #include #include #include #include <3d_viewer/eda_3d_viewer.h> #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 #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 #if defined(KICAD_SCRIPTING) || defined(KICAD_SCRIPTING_WXPYTHON) #include #include #endif using namespace std::placeholders; ///@{ /// \ingroup config static const wxString PlotLineWidthEntry = "PlotLineWidth_mm"; static const wxString ShowMicrowaveEntry = "ShowMicrowaveTools"; static const wxString ShowLayerManagerEntry = "ShowLayerManagerTools"; static const wxString ShowPageLimitsEntry = "ShowPageLimits"; ///@} 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_CLOSE( PCB_EDIT_FRAME::OnCloseWindow ) EVT_SIZE( PCB_EDIT_FRAME::OnSize ) EVT_TOOL( ID_MENU_READ_BOARD_BACKUP_FILE, PCB_EDIT_FRAME::Files_io ) EVT_TOOL( ID_MENU_RECOVER_BOARD_AUTOSAVE, PCB_EDIT_FRAME::Files_io ) EVT_TOOL( ID_OPEN_MODULE_EDITOR, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_TOOL( ID_OPEN_MODULE_VIEWER, PCB_EDIT_FRAME::Process_Special_Functions ) // Menu Files: EVT_MENU( ID_MAIN_MENUBAR, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_FLIP_VIEW, PCB_EDIT_FRAME::OnFlipPcbView ) EVT_MENU( ID_APPEND_FILE, PCB_EDIT_FRAME::Files_io ) 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_GEN_PLOT, PCB_EDIT_FRAME::ToPlotter ) EVT_MENU( ID_GEN_PLOT_GERBER, PCB_EDIT_FRAME::ToPlotter ) EVT_MENU( ID_GEN_EXPORT_SPECCTRA, PCB_EDIT_FRAME::ExportToSpecctra ) EVT_MENU( ID_GEN_EXPORT_FILE_GENCADFORMAT, PCB_EDIT_FRAME::ExportToGenCAD ) EVT_MENU( ID_GEN_EXPORT_FILE_MODULE_REPORT, PCB_EDIT_FRAME::GenFootprintsReport ) 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_GEN_IMPORT_SPECCTRA_SESSION,PCB_EDIT_FRAME::ImportSpecctraSession ) EVT_MENU( ID_GEN_IMPORT_SPECCTRA_DESIGN, PCB_EDIT_FRAME::ImportSpecctraDesign ) EVT_MENU( ID_MENU_ARCHIVE_MODULES_IN_LIBRARY, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_CREATE_LIBRARY_AND_ARCHIVE_MODULES, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( wxID_EXIT, PCB_EDIT_FRAME::OnQuit ) // menu Config EVT_MENU( ID_PCB_3DSHAPELIB_WIZARD, PCB_EDIT_FRAME::Process_Config ) EVT_MENU( ID_PREFERENCES_HOTKEY_SHOW_CURRENT_LIST, PCB_EDIT_FRAME::Process_Config ) EVT_MENU( wxID_PREFERENCES, PCB_EDIT_FRAME::Process_Config ) EVT_MENU( ID_GRID_SETTINGS, PCB_EDIT_FRAME::OnGridSettings ) // menu Postprocess EVT_MENU( ID_PCB_GEN_POS_MODULES_FILE, PCB_EDIT_FRAME::GenFootprintsPositionFile ) EVT_MENU( ID_PCB_GEN_DRILL_FILE, PCB_EDIT_FRAME::InstallDrillFrame ) EVT_MENU( ID_PCB_GEN_D356_FILE, PCB_EDIT_FRAME::GenD356File ) EVT_MENU( ID_PCB_GEN_CMP_FILE, PCB_EDIT_FRAME::RecreateCmpFileFromBoard ) EVT_MENU( ID_PCB_GEN_BOM_FILE_FROM_BOARD, PCB_EDIT_FRAME::RecreateBOMFileFromBoard ) // menu Miscellaneous EVT_MENU( ID_MENU_LIST_NETS, PCB_EDIT_FRAME::ListNetsAndSelect ) EVT_MENU( ID_PCB_EDIT_TRACKS_AND_VIAS, PCB_EDIT_FRAME::OnEditTracksAndVias ) EVT_MENU( ID_PCB_GLOBAL_DELETE, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_CLEAN, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_UPDATE_FOOTPRINTS, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_EXCHANGE_FOOTPRINTS, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_SWAP_LAYERS, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_MENU( ID_MENU_PCB_EDIT_TEXT_AND_GRAPHICS, PCB_EDIT_FRAME::OnEditTextAndGraphics ) // Menu Get Design Rules Editor EVT_MENU( ID_BOARD_SETUP_DIALOG, PCB_EDIT_FRAME::ShowBoardSetupDialog ) // Horizontal toolbar EVT_TOOL( ID_RUN_LIBRARY, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_TOOL( ID_GEN_PLOT_SVG, PCB_EDIT_FRAME::ExportSVG ) EVT_TOOL( ID_GET_NETLIST, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_TOOL( ID_DRC_CONTROL, PCB_EDIT_FRAME::Process_Special_Functions ) EVT_TOOL( ID_AUX_TOOLBAR_PCB_SELECT_LAYER_PAIR, PCB_EDIT_FRAME::Process_Special_Functions ) 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 ) #endif #if defined( KICAD_SCRIPTING_WXPYTHON ) // has meaning only with KICAD_SCRIPTING_WXPYTHON enabled EVT_TOOL( ID_TOOLBARH_PCB_SCRIPTING_CONSOLE, PCB_EDIT_FRAME::ScriptingConsoleEnableDisable ) EVT_UPDATE_UI( ID_TOOLBARH_PCB_SCRIPTING_CONSOLE, PCB_EDIT_FRAME::OnUpdateScriptingConsoleState ) #endif // Option toolbar EVT_TOOL( ID_TB_OPTIONS_SHOW_EXTRA_VERTICAL_TOOLBAR_MICROWAVE, PCB_EDIT_FRAME::OnSelectOptionToolbar ) EVT_TOOL( ID_UPDATE_PCB_FROM_SCH, PCB_EDIT_FRAME::OnUpdatePCBFromSch ) EVT_TOOL( ID_RUN_EESCHEMA, PCB_EDIT_FRAME::OnRunEeschema ) EVT_TOOL( ID_TB_OPTIONS_SHOW_MANAGE_LAYERS_VERTICAL_TOOLBAR, PCB_EDIT_FRAME::OnSelectOptionToolbar ) EVT_MENU_RANGE( ID_POPUP_PCB_START_RANGE, ID_POPUP_PCB_END_RANGE, PCB_EDIT_FRAME::Process_Special_Functions ) // 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 ) // popup menus EVT_MENU_RANGE( ID_POPUP_GENERAL_START_RANGE, ID_POPUP_GENERAL_END_RANGE, PCB_EDIT_FRAME::Process_Special_Functions ) // User interface update event handlers. EVT_UPDATE_UI( ID_AUX_TOOLBAR_PCB_SELECT_LAYER_PAIR, PCB_EDIT_FRAME::OnUpdateLayerPair ) EVT_UPDATE_UI( ID_TOOLBARH_PCB_SELECT_LAYER, PCB_EDIT_FRAME::OnUpdateLayerSelectBox ) EVT_UPDATE_UI( ID_NO_TOOL_SELECTED, PCB_EDIT_FRAME::OnUpdateVerticalToolbar ) EVT_UPDATE_UI( ID_ZOOM_SELECTION, PCB_EDIT_FRAME::OnUpdateVerticalToolbar ) 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_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 ) EVT_UPDATE_UI_RANGE( ID_PCB_HIGHLIGHT_BUTT, ID_PCB_MEASUREMENT_TOOL, PCB_EDIT_FRAME::OnUpdateVerticalToolbar ) EVT_UPDATE_UI_RANGE( ID_PCB_MUWAVE_START_CMD, ID_PCB_MUWAVE_END_CMD, PCB_EDIT_FRAME::OnUpdateMuWaveToolbar ) EVT_COMMAND( wxID_ANY, LAYER_WIDGET::EVT_LAYER_COLOR_CHANGE, PCB_EDIT_FRAME::OnLayerColorChange ) END_EVENT_TABLE() PCB_EDIT_FRAME::PCB_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent ) : PCB_BASE_EDIT_FRAME( aKiway, aParent, FRAME_PCB, wxT( "Pcbnew" ), wxDefaultPosition, wxDefaultSize, KICAD_DEFAULT_DRAWFRAME_STYLE, PCB_EDIT_FRAME_NAME ) { m_showBorderAndTitleBlock = true; // true to display sheet references m_showAxis = false; // true to display X and Y axis m_showOriginAxis = true; m_showGridAxis = true; m_SelTrackWidthBox = NULL; m_SelViaSizeBox = NULL; m_SelLayerBox = NULL; m_show_microwave_tools = false; m_show_layer_manager_tools = true; m_hotkeysDescrList = g_Board_Editor_Hotkeys_Descr; m_hasAutoSave = true; m_microWaveToolBar = NULL; m_Layers = nullptr; m_FrameSize = ConvertDialogToPixels( wxSize( 500, 350 ) ); // default in case of no prefs // 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 = "Pcbnew"; // Create GAL canvas auto galCanvas = new PCB_DRAW_PANEL_GAL( this, -1, wxPoint( 0, 0 ), m_FrameSize, GetGalDisplayOptions(), EDA_DRAW_PANEL_GAL::GAL_TYPE_CAIRO ); SetGalCanvas( galCanvas ); SetBoard( new BOARD() ); // Create the PCB_LAYER_WIDGET *after* SetBoard(): m_Layers = new PCB_LAYER_WIDGET( this, GetGalCanvas() ); m_drc = new DRC( this ); // these 2 objects point to each other wxIcon icon; icon.CopyFromBitmap( KiBitmap( icon_pcbnew_xpm ) ); SetIcon( icon ); // LoadSettings() *after* creating m_LayersManager, because LoadSettings() // initialize parameters in m_LayersManager LoadSettings( config() ); SetScreen( new PCB_SCREEN( GetPageSettings().GetSizeIU() ) ); GetScreen()->SetMaxUndoItems( m_UndoRedoCountMax ); // PCB drawings start in the upper left corner. GetScreen()->m_Center = false; SetSize( m_FramePos.x, m_FramePos.y, m_FrameSize.x, m_FrameSize.y ); GetScreen()->AddGrid( m_UserGridSize, EDA_UNITS_T::UNSCALED_UNITS, ID_POPUP_GRID_USER ); GetScreen()->SetGrid( ID_POPUP_GRID_LEVEL_1000 + m_LastGridSizeId ); setupTools(); ReCreateMenuBar(); ReCreateHToolbar(); ReCreateAuxiliaryToolbar(); ReCreateVToolbar(); ReCreateOptToolbar(); ReCreateMicrowaveVToolbar(); m_auimgr.SetManagedWindow( this ); m_auimgr.SetArtProvider( new EDA_DOCKART() ); // Horizontal items; 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(4) ); m_auimgr.AddPane( m_messagePanel, EDA_PANE().Messages().Name( "MsgPanel" ).Bottom().Layer(6) ); // Vertical items; layers 1 - 3 m_auimgr.AddPane( m_optionsToolBar, EDA_PANE().VToolbar().Name( "OptToolbar" ).Left().Layer(3) ); m_auimgr.AddPane( m_microWaveToolBar, EDA_PANE().VToolbar().Name( "MicrowaveToolbar" ).Right().Layer(1) ); m_auimgr.AddPane( m_drawToolBar, EDA_PANE().VToolbar().Name( "ToolsToolbar" ).Right().Layer(2) ); m_auimgr.AddPane( m_Layers, EDA_PANE().Palette().Name( "LayersManager" ).Right().Layer(3) .Caption( _( "Layers Manager" ) ).PaneBorder( false ) .MinSize( 80, -1 ).BestSize( m_Layers->GetBestSize() ) ); m_auimgr.AddPane( GetGalCanvas(), EDA_PANE().Canvas().Name( "DrawFrame" ).Center() ); m_auimgr.GetPane( "LayersManager" ).Show( m_show_layer_manager_tools ); m_auimgr.GetPane( "MicrowaveToolbar" ).Show( m_show_microwave_tools ); ReFillLayerWidget(); // this is near end because contents establish size m_Layers->ReFillRender(); // Update colors in Render after the config is read syncLayerWidgetLayer(); m_auimgr.Update(); GetToolManager()->RunAction( ACTIONS::gridPreset, true, m_LastGridSizeId ); GetToolManager()->RunAction( ACTIONS::zoomFitScreen ); m_canvasType = LoadCanvasTypeSetting(); // Nudge user to switch to OpenGL if they are on Cairo if( m_firstRunDialogSetting < 1 ) { if( m_canvasType != EDA_DRAW_PANEL_GAL::GAL_TYPE_OPENGL ) { wxString msg = _( "KiCad can use your graphics card to give you a smoother " "and faster experience. This option is turned off by " "default since it is not compatible with all computers.\n\n" "Would you like to try enabling graphics acceleration?\n\n" "If you'd like to choose later, select Accelerated Graphics " "in the Preferences menu." ); wxMessageDialog dlg( this, msg, _( "Enable Graphics Acceleration" ), wxYES_NO ); dlg.SetYesNoLabels( _( "&Enable Acceleration" ), _( "&No Thanks" ) ); if( dlg.ShowModal() == wxID_YES ) { // Save Cairo as default in case OpenGL crashes saveCanvasTypeSetting( EDA_DRAW_PANEL_GAL::GAL_TYPE_CAIRO ); // Switch to OpenGL, which will save the new setting if successful GetToolManager()->RunAction( ACTIONS::acceleratedGraphics, true ); // Switch back to Cairo if OpenGL is not supported if( GetGalCanvas()->GetBackend() == EDA_DRAW_PANEL_GAL::GAL_TYPE_NONE ) { GetToolManager()->RunAction( ACTIONS::standardGraphics, true ); } } else { // If they were on legacy, switch to Cairo GetToolManager()->RunAction( ACTIONS::standardGraphics, true ); } } m_firstRunDialogSetting = 1; SaveSettings( config() ); } GetGalCanvas()->SwitchBackend( m_canvasType ); galCanvas->GetGAL()->SetGridSize( VECTOR2D( GetScreen()->GetGridSize() ) ); galCanvas->GetView()->SetScale( GetZoomLevelCoeff() / GetScreen()->GetZoom() ); ActivateGalCanvas(); // 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 // if( !appK2S.FileExists() ) // GetMenuBar()->FindItem( ID_GEN_EXPORT_FILE_STEP )->Enable( false ); } PCB_EDIT_FRAME::~PCB_EDIT_FRAME() { delete m_drc; } void PCB_EDIT_FRAME::SetBoard( BOARD* aBoard ) { PCB_BASE_EDIT_FRAME::SetBoard( aBoard ); aBoard->GetConnectivity()->Build( aBoard ); // reload the worksheet 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 ); PCB_DRAW_PANEL_GAL* drawPanel = static_cast( GetGalCanvas() ); // Prepare worksheet template KIGFX::WS_PROXY_VIEW_ITEM* worksheet; worksheet = new KIGFX::WS_PROXY_VIEW_ITEM( IU_PER_MILS ,&m_Pcb->GetPageSettings(), &m_Pcb->GetTitleBlock() ); worksheet->SetSheetName( std::string( GetScreenDesc().mb_str() ) ); BASE_SCREEN* screen = GetScreen(); if( screen != NULL ) { worksheet->SetSheetNumber( screen->m_ScreenNumber ); worksheet->SetSheetCount( screen->m_NumberOfScreens ); } if( auto board = GetBoard() ) worksheet->SetFileName( TO_UTF8( board->GetFileName() ) ); // PCB_DRAW_PANEL_GAL takes ownership of the worksheet drawPanel->SetWorksheet( worksheet ); } bool PCB_EDIT_FRAME::isAutoSaveRequired() const { if( GetScreen() ) return GetScreen()->IsSave(); return false; } 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, GetGalCanvas()->GetView(), GetGalCanvas()->GetViewControls(), this ); m_actions = new PCB_ACTIONS(); m_toolDispatcher = new TOOL_DISPATCHER( m_toolManager, m_actions ); // Register tools m_toolManager->RegisterTool( new COMMON_TOOLS ); m_toolManager->RegisterTool( new SELECTION_TOOL ); m_toolManager->RegisterTool( new ZOOM_TOOL ); m_toolManager->RegisterTool( new PCBNEW_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 PAD_TOOL ); m_toolManager->RegisterTool( new DRAWING_TOOL ); m_toolManager->RegisterTool( new POINT_EDITOR ); m_toolManager->RegisterTool( new PCBNEW_CONTROL ); m_toolManager->RegisterTool( new PCB_EDITOR_CONTROL ); 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->InitTools(); // Run the selection tool, it is supposed to be always active m_toolManager->InvokeTool( "pcbnew.InteractiveSelection" ); } void PCB_EDIT_FRAME::ReFillLayerWidget() { m_Layers->ReFill(); wxAuiPaneInfo& lyrs = m_auimgr.GetPane( m_Layers ); wxSize bestz = m_Layers->GetBestSize(); lyrs.MinSize( bestz ); lyrs.BestSize( bestz ); lyrs.FloatingSize( bestz ); if( lyrs.IsDocked() ) m_auimgr.Update(); else m_Layers->SetSize( bestz ); } void PCB_EDIT_FRAME::OnQuit( wxCommandEvent& event ) { Close( false ); } void PCB_EDIT_FRAME::OnCloseWindow( wxCloseEvent& Event ) { if( GetScreen()->IsModify() && !GetBoard()->IsEmpty() ) { wxString msg = _( "Save changes to\n\"%s\"\nbefore closing?" ); if( !HandleUnsavedChanges( this, wxString::Format( msg, GetBoard()->GetFileName() ), [&]()->bool { return Files_io_from_id( ID_SAVE_BOARD ); } ) ) { Event.Veto(); return; } } // 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) GetGalCanvas()->SetEvtHandlerEnabled( false ); GetGalCanvas()->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 ); } // 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 ); // 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 ); // do not show the window because ScreenPcb will be deleted and we do not // want any paint event Show( false ); Destroy(); } void PCB_EDIT_FRAME::ActivateGalCanvas() { PCB_BASE_EDIT_FRAME::ActivateGalCanvas(); COLORS_DESIGN_SETTINGS& cds = Settings().Colors(); GetGalCanvas()->GetGAL()->SetGridColor( cds.GetLayerColor( LAYER_GRID ) ); auto view = GetGalCanvas()->GetView(); view->GetPainter()->GetSettings()->ImportLegacyColors( &cds ); GetGalCanvas()->Refresh(); } void PCB_EDIT_FRAME::ShowBoardSetupDialog( wxCommandEvent& event ) { DoShowBoardSetupDialog(); } void PCB_EDIT_FRAME::DoShowBoardSetupDialog( const wxString& aInitialPage, const wxString& aInitialParentPage ) { DIALOG_BOARD_SETUP dlg( this ); if( !aInitialPage.IsEmpty() ) dlg.SetInitialPage( aInitialPage, aInitialParentPage ); if( dlg.ShowModal() == wxID_OK ) { SaveProjectSettings( false ); UpdateUserInterface(); ReCreateAuxiliaryToolbar(); for( auto module : GetBoard()->Modules() ) GetGalCanvas()->GetView()->Update( module ); GetGalCanvas()->Refresh(); //this event causes the routing tool to reload its design rules information TOOL_EVENT toolEvent( TC_COMMAND, TA_MODEL_CHANGE, AS_ACTIVE ); m_toolManager->ProcessEvent( toolEvent ); OnModify(); } } void PCB_EDIT_FRAME::LoadSettings( wxConfigBase* aCfg ) { PCB_BASE_FRAME::LoadSettings( aCfg ); wxConfigLoadSetups( aCfg, GetConfigurationSettings() ); m_configSettings.Load( aCfg ); double dtmp; aCfg->Read( PlotLineWidthEntry, &dtmp, 0.1 ); // stored in mm dtmp = std::max( 0.01, std::min( dtmp, 5.0 ) ); g_DrawDefaultLineThickness = Millimeter2iu( dtmp ); aCfg->Read( ShowMicrowaveEntry, &m_show_microwave_tools ); aCfg->Read( ShowLayerManagerEntry, &m_show_layer_manager_tools ); aCfg->Read( ShowPageLimitsEntry, &m_showPageLimits ); } void PCB_EDIT_FRAME::SaveSettings( wxConfigBase* aCfg ) { m_configSettings.Save( aCfg ); PCB_BASE_FRAME::SaveSettings( aCfg ); wxConfigSaveSetups( aCfg, GetConfigurationSettings() ); // This value is stored in mm ) aCfg->Write( PlotLineWidthEntry, MM_PER_IU * g_DrawDefaultLineThickness ); aCfg->Write( ShowMicrowaveEntry, (long) m_show_microwave_tools ); aCfg->Write( ShowLayerManagerEntry, (long)m_show_layer_manager_tools ); aCfg->Write( ShowPageLimitsEntry, m_showPageLimits ); } bool PCB_EDIT_FRAME::IsGridVisible() const { return IsElementVisible( LAYER_GRID ); } void PCB_EDIT_FRAME::SetGridVisibility(bool aVisible) { SetElementVisibility( LAYER_GRID, aVisible ); } COLOR4D PCB_EDIT_FRAME::GetGridColor() { return Settings().Colors().GetItemColor( LAYER_GRID ); } void PCB_EDIT_FRAME::SetGridColor( COLOR4D aColor ) { Settings().Colors().SetItemColor( LAYER_GRID, aColor ); GetGalCanvas()->GetGAL()->SetGridColor( aColor ); } bool PCB_EDIT_FRAME::IsMicroViaAcceptable() { int copperlayercnt = GetBoard()->GetCopperLayerCount( ); PCB_LAYER_ID currLayer = GetActiveLayer(); if( !GetDesignSettings().m_MicroViasAllowed ) return false; // Obvious.. if( copperlayercnt < 4 ) return false; // Only on multilayer boards.. if( ( currLayer == B_Cu ) || ( currLayer == F_Cu ) || ( currLayer == copperlayercnt - 2 ) || ( currLayer == In1_Cu ) ) return true; return false; } void PCB_EDIT_FRAME::SetActiveLayer( PCB_LAYER_ID aLayer ) { PCB_BASE_FRAME::SetActiveLayer( aLayer ); syncLayerWidgetLayer(); m_toolManager->RunAction( PCB_ACTIONS::layerChanged ); // notify other tools GetGalCanvas()->SetFocus(); // allow capture of hotkeys GetGalCanvas()->SetHighContrastLayer( aLayer ); GetGalCanvas()->Refresh(); } void PCB_EDIT_FRAME::onBoardLoaded() { UpdateTitle(); // Re-create layers manager based on layer info in board ReFillLayerWidget(); ReCreateLayerBox(); // Sync layer and item visibility syncLayerVisibilities(); syncLayerWidgetLayer(); syncRenderStates(); // Update the tracks / vias available sizes list: ReCreateAuxiliaryToolbar(); // Update the RATSNEST items, which were not loaded at the time // BOARD::SetVisibleElements() was called from within any PLUGIN. // See case LAYER_RATSNEST: in BOARD::SetElementVisibility() GetBoard()->SetVisibleElements( GetBoard()->GetVisibleElements() ); // Display the loaded board: Zoom_Automatique( false ); Refresh(); SetMsgPanel( GetBoard() ); SetStatusText( wxEmptyString ); } void PCB_EDIT_FRAME::syncLayerWidgetLayer() { m_Layers->SelectLayer( GetActiveLayer() ); m_Layers->OnLayerSelected(); } void PCB_EDIT_FRAME::syncRenderStates() { m_Layers->ReFillRender(); } void PCB_EDIT_FRAME::syncLayerVisibilities() { m_Layers->SyncLayerVisibilities(); static_cast( GetGalCanvas() )->SyncLayersVisibility( m_Pcb ); } void PCB_EDIT_FRAME::OnUpdateLayerAlpha( wxUpdateUIEvent & ) { m_Layers->SyncLayerAlphaIndicators(); } 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 ) { GetGalCanvas()->GetView()->SetLayerVisible( aElement , aNewState ); GetBoard()->SetElementVisibility( aElement, aNewState ); m_Layers->SetRenderState( aElement, aNewState ); } void PCB_EDIT_FRAME::SetVisibleAlls() { GetBoard()->SetVisibleAlls(); for( GAL_LAYER_ID ii = GAL_LAYER_ID_START; ii < GAL_LAYER_ID_BITMASK_END; ++ii ) m_Layers->SetRenderState( ii, true ); } void PCB_EDIT_FRAME::ShowChangedLanguage() { // call my base class PCB_BASE_EDIT_FRAME::ShowChangedLanguage(); // update the layer manager m_Layers->Freeze(); wxAuiPaneInfo& pane_info = m_auimgr.GetPane( m_Layers ); pane_info.Caption( _( "Visibles" ) ); m_auimgr.Update(); m_Layers->SetLayersManagerTabsText(); ReFillLayerWidget(); // m_Layers->ReFillRender(); // syncRenderStates() does this // upate the layer widget to match board visibility states, both layers and render columns. syncLayerVisibilities(); syncLayerWidgetLayer(); syncRenderStates(); m_Layers->Thaw(); // pcbnew-specific toolbars ReCreateMicrowaveVToolbar(); } wxString PCB_EDIT_FRAME::GetLastNetListRead() { wxFileName absoluteFileName = m_lastNetListRead; wxFileName pcbFileName = GetBoard()->GetFileName(); if( !absoluteFileName.MakeAbsolute( pcbFileName.GetPath() ) || !absoluteFileName.FileExists() ) { absoluteFileName.Clear(); m_lastNetListRead = wxEmptyString; } return absoluteFileName.GetFullPath(); } void PCB_EDIT_FRAME::SetLastNetListRead( const wxString& aLastNetListRead ) { wxFileName relativeFileName = aLastNetListRead; wxFileName pcbFileName = GetBoard()->GetFileName(); if( relativeFileName.MakeRelativeTo( pcbFileName.GetPath() ) && relativeFileName.GetFullPath() != aLastNetListRead ) { m_lastNetListRead = relativeFileName.GetFullPath(); } } void PCB_EDIT_FRAME::OnModify( ) { PCB_BASE_FRAME::OnModify(); Update3DView( false ); m_ZoneFillsDirty = true; } void PCB_EDIT_FRAME::ExportSVG( wxCommandEvent& event ) { InvokeExportSVG( this, GetBoard() ); } void PCB_EDIT_FRAME::UpdateTitle() { wxFileName fileName = GetBoard()->GetFileName(); wxString fileinfo; if( fileName.IsOk() && fileName.FileExists() ) fileinfo = fileName.IsFileWritable() ? wxString( wxEmptyString ) : _( " [Read Only]" ); else fileinfo = _( " [Unsaved]" ); SetTitle( wxString::Format( _( "Pcbnew" ) + wxT( " \u2014 %s%s" ), fileName.GetFullPath(), fileinfo ) ); } 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) Compile_Ratsnest( true ); GetBoard()->BuildConnectivity(); // Update info shown by the horizontal toolbars ReCreateLayerBox(); // Update the layer manager m_Layers->Freeze(); ReFillLayerWidget(); // m_Layers->ReFillRender(); // syncRenderStates() does this // upate the layer widget to match board visibility states, both layers and render columns. syncLayerVisibilities(); syncLayerWidgetLayer(); syncRenderStates(); m_Layers->Thaw(); } #if defined( KICAD_SCRIPTING_WXPYTHON ) void PCB_EDIT_FRAME::ScriptingConsoleEnableDisable( wxCommandEvent& aEvent ) { 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::OnLayerColorChange( wxCommandEvent& aEvent ) { ReCreateLayerBox(); } void PCB_EDIT_FRAME::SwitchCanvas( EDA_DRAW_PANEL_GAL::GAL_TYPE aCanvasType ) { // switches currently used canvas (Cairo / OpenGL). PCB_BASE_FRAME::SwitchCanvas( aCanvasType ); GetGalCanvas()->GetGAL()->SetGridSize( VECTOR2D( GetScreen()->GetGridSize() ) ); // The base class method *does not reinit* the layers manager. We must upate the // layer widget to match board visibility states, both layers and render columns. syncLayerVisibilities(); syncLayerWidgetLayer(); syncRenderStates(); } void PCB_EDIT_FRAME::ToPlotter( wxCommandEvent& event ) { PCB_PLOT_PARAMS plotSettings = GetPlotSettings(); switch( event.GetId() ) { 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_SVG: wxFAIL_MSG( "Must be handled by ExportSVG()" ); break; case ID_GEN_PLOT: default: // called with no specified plot type: keep the previous setup 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::SetCurrentNetClass( const wxString& aNetClassName ) { bool change = GetDesignSettings().SetCurrentNetClass( aNetClassName ); if( change ) ReCreateAuxiliaryToolbar(); return change; } void PCB_EDIT_FRAME::OnUpdatePCBFromSch( wxCommandEvent& event ) { NETLIST netlist; if( FetchNetlistFromSchematic( netlist, ANNOTATION_DIALOG ) ) UpdatePCBFromNetlist( netlist ); } bool PCB_EDIT_FRAME::FetchNetlistFromSchematic( NETLIST& aNetlist, FETCH_NETLIST_MODE aMode ) { if( Kiface().IsSingle() ) { DisplayError( this, _( "Cannot update the PCB because Pcbnew 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; } // 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 schfn( Prj().GetProjectPath(), Prj().GetProjectName(), SchematicFileExtension ); frame->OpenProjectFiles( std::vector( 1, schfn.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(); } std::string payload; if( aMode == NO_ANNOTATION ) payload = "no-annotate"; else if( aMode == QUIET_ANNOTATION ) payload = "quiet-annotate"; Kiway().ExpressMail( FRAME_SCH, MAIL_SCH_GET_NETLIST, payload, this ); try { auto lineReader = new STRING_LINE_READER( payload, _( "Eeschema netlist" ) ); KICAD_NETLIST_READER netlistReader( lineReader, &aNetlist ); netlistReader.LoadNetlist(); } catch( const IO_ERROR& ) { assert( false ); // should never happen return false; } return true; } void PCB_EDIT_FRAME::UpdatePCBFromNetlist( NETLIST& aNetlist ) { DIALOG_UPDATE_PCB updateDialog( this, &aNetlist ); updateDialog.ShowModal(); auto selectionTool = static_cast( m_toolManager->FindTool( "pcbnew.InteractiveSelection" ) ); if( !selectionTool->GetSelection().Empty() ) GetToolManager()->InvokeTool( "pcbnew.InteractiveEdit" ); } void PCB_EDIT_FRAME::DoUpdatePCBFromNetlist( NETLIST& aNetlist, bool aUseTimestamps ) { BOARD_NETLIST_UPDATER updater( this, GetBoard() ); updater.SetLookupByTimestamp( aUseTimestamps ); updater.SetDeleteUnusedComponents( false ); updater.SetReplaceFootprints( true ); updater.SetDeleteSinglePadNets( false ); updater.UpdateNetlist( aNetlist ); } void PCB_EDIT_FRAME::OnRunEeschema( wxCommandEvent& event ) { wxString msg; wxFileName schfn( Prj().GetProjectPath(), Prj().GetProjectName(), SchematicFileExtension ); if( !schfn.FileExists() ) { msg.Printf( _( "Schematic file \"%s\" not found." ), schfn.GetFullPath() ); wxMessageBox( msg, _( "KiCad Error" ), wxOK | wxICON_ERROR, this ); return; } if( Kiface().IsSingle() ) { wxString filename = wxT( "\"" ) + schfn.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( 1, schfn.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::OnFlipPcbView( wxCommandEvent& evt ) { auto view = GetGalCanvas()->GetView(); view->SetMirror( evt.IsChecked(), false ); view->RecacheAllItems(); Refresh(); } void PCB_EDIT_FRAME::PythonPluginsReload() { // Reload Python plugins if they are newer than // the already loaded, and load new plugins #if defined(KICAD_SCRIPTING) //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: RebuildActionPluginMenus(); // Recreate top toolbar to add action plugin buttons ReCreateHToolbar(); #endif #endif } void PCB_EDIT_FRAME::InstallFootprintPropertiesDialog( MODULE* Module ) { if( Module == NULL ) return; DIALOG_FOOTPRINT_BOARD_EDITOR* dlg = new DIALOG_FOOTPRINT_BOARD_EDITOR( this, Module ); int retvalue = dlg->ShowModal(); /* retvalue = * FP_PRM_EDITOR_RETVALUE::PRM_EDITOR_WANT_UPDATE_FP if update footprint * FP_PRM_EDITOR_RETVALUE::PRM_EDITOR_WANT_EXCHANGE_FP if change footprint * FP_PRM_EDITOR_RETVALUE::PRM_EDITOR_WANT_MODEDIT for a goto editor command * FP_PRM_EDITOR_RETVALUE::PRM_EDITOR_EDIT_OK for normal edit */ dlg->Close(); dlg->Destroy(); if( retvalue == DIALOG_FOOTPRINT_BOARD_EDITOR::PRM_EDITOR_EDIT_OK ) { // If something edited, push a refresh request GetGalCanvas()->Refresh(); } else if( retvalue == DIALOG_FOOTPRINT_BOARD_EDITOR::PRM_EDITOR_EDIT_BOARD_FOOTPRINT ) { FOOTPRINT_EDIT_FRAME* editor = (FOOTPRINT_EDIT_FRAME*) Kiway().Player( FRAME_PCB_MODULE_EDITOR, true ); editor->Load_Module_From_BOARD( Module ); editor->Show( true ); editor->Raise(); // Iconize( false ); } else if( retvalue == DIALOG_FOOTPRINT_BOARD_EDITOR::PRM_EDITOR_EDIT_LIBRARY_FOOTPRINT ) { FOOTPRINT_EDIT_FRAME* editor = (FOOTPRINT_EDIT_FRAME*) Kiway().Player( FRAME_PCB_MODULE_EDITOR, true ); editor->LoadModuleFromLibrary( Module->GetFPID() ); editor->Show( true ); editor->Raise(); // Iconize( false ); } else if( retvalue == DIALOG_FOOTPRINT_BOARD_EDITOR::PRM_EDITOR_WANT_UPDATE_FP ) { InstallExchangeModuleFrame( Module, true, true ); } else if( retvalue == DIALOG_FOOTPRINT_BOARD_EDITOR::PRM_EDITOR_WANT_EXCHANGE_FP ) { InstallExchangeModuleFrame( Module, false, true ); } } int PCB_EDIT_FRAME::InstallExchangeModuleFrame( MODULE* aModule, bool updateMode, bool selectedMode ) { DIALOG_EXCHANGE_FOOTPRINTS dialog( this, aModule, updateMode, selectedMode ); return dialog.ShowQuasiModal(); } void PCB_EDIT_FRAME::CommonSettingsChanged() { PCB_BASE_EDIT_FRAME::CommonSettingsChanged(); ReCreateMicrowaveVToolbar(); Layout(); SendSizeEvent(); } void PCB_EDIT_FRAME::LockModule( MODULE* aModule, bool aLocked ) { const wxString ModulesMaskSelection = wxT( "*" ); if( aModule ) { aModule->SetLocked( aLocked ); SetMsgPanel( aModule ); OnModify(); } else { for( auto mod : GetBoard()->Modules() ) { if( WildCompareString( ModulesMaskSelection, mod->GetReference() ) ) { mod->SetLocked( aLocked ); OnModify(); } } } } 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 ); }