|
|
/*
* This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2017-2023 KiCad Developers, see AUTHORS.txt for contributors. * * This program is free software: you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the * Free Software Foundation, either version 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 <pcb_edit_frame.h>
#include <panel_setup_layers.h>
#include <panel_setup_text_and_graphics.h>
#include <panel_setup_constraints.h>
#include <dialogs/panel_setup_netclasses.h>
#include <panel_setup_tracks_and_vias.h>
#include <panel_setup_mask_and_paste.h>
#include <../board_stackup_manager/panel_board_stackup.h>
#include <../board_stackup_manager/panel_board_finish.h>
#include <confirm.h>
#include <board_design_settings.h>
#include <kiface_base.h>
#include <drc/drc_item.h>
#include <dialog_import_settings.h>
#include <io_mgr.h>
#include <dialogs/panel_setup_severities.h>
#include <panel_text_variables.h>
#include <project.h>
#include <project/project_file.h>
#include <settings/settings_manager.h>
#include <widgets/resettable_panel.h>
#include <widgets/wx_progress_reporters.h>
#include <wildcards_and_files_ext.h>
#include "dialog_board_setup.h"
#include "panel_setup_rules.h"
std::mutex DIALOG_BOARD_SETUP::g_Mutex;
DIALOG_BOARD_SETUP::DIALOG_BOARD_SETUP( PCB_EDIT_FRAME* aFrame ) : PAGED_DIALOG( aFrame, _( "Board Setup" ), false, _( "Import Settings from Another Board..." ) ), m_frame( aFrame ){ SetEvtHandlerEnabled( false );
m_layers = new PANEL_SETUP_LAYERS( this, m_frame ); m_physicalStackup = new PANEL_SETUP_BOARD_STACKUP( this, m_frame, m_layers ); m_boardFinish = new PANEL_SETUP_BOARD_FINISH( this, m_frame );
m_currentPage = -1;
/*
* WARNING: If you change page names you MUST update calls to ShowBoardSetupDialog(). */
m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Board Stackup" ) );
/*
* WARNING: Code currently relies on the layers setup coming before the physical stackup panel, * and thus transferring data to the board first. See comment in * PANEL_SETUP_BOARD_STACKUP::TransferDataFromWindow and rework this logic if it is determined * that the order of these pages should be changed. */ m_treebook->AddSubPage( m_layers, _( "Board Editor Layers" ) );
m_physicalStackupPage = m_treebook->GetPageCount(); m_treebook->AddSubPage( m_physicalStackup, _( "Physical Stackup" ) ); m_treebook->AddSubPage( m_boardFinish, _( "Board Finish" ) );
m_maskAndPagePage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_MASK_AND_PASTE( aParent, m_frame ); }, _( "Solder Mask/Paste" ) );
m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Text & Graphics" ) );
m_textAndGraphicsPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_TEXT_AND_GRAPHICS( aParent, m_frame ); }, _( "Defaults" ) );
m_formattingPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_FORMATTING( aParent, m_frame ); }, _( "Formatting" ) );
m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_TEXT_VARIABLES( aParent, &Prj() ); }, _( "Text Variables" ) );
m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Design Rules" ) );
m_constraintsPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_CONSTRAINTS( aParent, m_frame ); }, _( "Constraints" ) );
m_tracksAndViasPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_TRACKS_AND_VIAS( aParent, m_frame ); }, _( "Pre-defined Sizes" ) );
m_netclassesPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { BOARD* board = m_frame->GetBoard(); return new PANEL_SETUP_NETCLASSES( aParent, m_frame, m_frame->Prj().GetProjectFile().NetSettings(), board->GetNetClassAssignmentCandidates(), false ); }, _( "Net Classes" ) );
m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { return new PANEL_SETUP_RULES( aParent, m_frame ); }, _( "Custom Rules" ) );
m_severitiesPage = m_treebook->GetPageCount(); m_treebook->AddLazySubPage( [this]( wxWindow* aParent ) -> wxWindow* { BOARD* board = m_frame->GetBoard(); return new PANEL_SETUP_SEVERITIES( aParent, DRC_ITEM::GetItemsWithSeverities(), board->GetDesignSettings().m_DRCSeverities ); }, _( "Violation Severity" ) );
for( size_t i = 0; i < m_treebook->GetPageCount(); ++i ) m_treebook->ExpandNode( i );
// This is unfortunate, but it's the cost of lazy-loading the panels
m_treebook->SetMinSize( wxSize( 980, 600 ) ); m_treebook->SetInitialSize( wxSize( 980, 600 ) );
SetEvtHandlerEnabled( true );
finishDialogSettings();
if( Prj().IsReadOnly() ) { m_infoBar->ShowMessage( _( "Project is missing or read-only. Some settings will not " "be editable." ), wxICON_WARNING ); }
wxBookCtrlEvent evt( wxEVT_TREEBOOK_PAGE_CHANGED, wxID_ANY, 0 );
wxQueueEvent( m_treebook, evt.Clone() );}
DIALOG_BOARD_SETUP::~DIALOG_BOARD_SETUP(){}
void DIALOG_BOARD_SETUP::onPageChanged( wxBookCtrlEvent& aEvent ){ PAGED_DIALOG::onPageChanged( aEvent );
size_t page = aEvent.GetSelection();
// Ensure layer page always gets updated even if we aren't moving towards it
if( m_currentPage == m_physicalStackupPage ) m_layers->SyncCopperLayers( m_physicalStackup->GetCopperLayerCount() );
if( page == m_physicalStackupPage ) m_physicalStackup->OnLayersOptionsChanged( m_layers->GetUILayerMask() ); else if( Prj().IsReadOnly() ) KIUI::Disable( m_treebook->GetPage( page ) );
m_currentPage = page;}
void DIALOG_BOARD_SETUP::onAuxiliaryAction( wxCommandEvent& aEvent ){ DIALOG_IMPORT_SETTINGS importDlg( this, m_frame );
if( importDlg.ShowModal() == wxID_CANCEL ) return;
wxFileName boardFn( importDlg.GetFilePath() ); wxFileName projectFn( boardFn );
projectFn.SetExt( ProjectFileExtension );
if( !m_frame->GetSettingsManager()->LoadProject( projectFn.GetFullPath(), false ) ) { wxString msg = wxString::Format( _( "Error importing settings from board:\n" "Associated project file %s could not be loaded" ), projectFn.GetFullPath() ); DisplayErrorMessage( this, msg );
return; }
// Flag so user can stop work if it will result in deleted inner copper layers
// and still clean up this function properly.
bool okToProceed = true;
PROJECT* otherPrj = m_frame->GetSettingsManager()->GetProject( projectFn.GetFullPath() );
PLUGIN::RELEASER pi( IO_MGR::PluginFind( IO_MGR::KICAD_SEXP ) );
BOARD* otherBoard = nullptr;
try { WX_PROGRESS_REPORTER progressReporter( this, _( "Loading PCB" ), 1 );
otherBoard = pi->Load( boardFn.GetFullPath(), nullptr, nullptr, nullptr, &progressReporter );
if( importDlg.m_LayersOpt->GetValue() ) { BOARD* loadedBoard = m_frame->GetBoard();
// Check if "Import Settings" board has more layers than the current board.
okToProceed = m_layers->CheckCopperLayerCount( loadedBoard, otherBoard ); } } catch( const IO_ERROR& ioe ) { // You wouldn't think boardFn.GetFullPath() would throw, but we get a stack buffer
// underflow from ASAN. While it's probably an ASAN error, a second try/catch doesn't
// cost us much.
try { if( ioe.Problem() != wxT( "CANCEL" ) ) { wxString msg = wxString::Format( _( "Error loading board file:\n%s" ), boardFn.GetFullPath() ); DisplayErrorMessage( this, msg, ioe.What() ); }
if( otherPrj != &m_frame->Prj() ) m_frame->GetSettingsManager()->UnloadProject( otherPrj, false ); } catch(...) { // That was already our best-efforts
}
return; }
if( okToProceed ) { otherBoard->SetProject( otherPrj );
// If layers options are imported, import also the stackup
// layers options and stackup are linked, so they cannot be imported
// separately, and stackup can be imported only after layers options
if( importDlg.m_LayersOpt->GetValue() ) { m_physicalStackup->ImportSettingsFrom( otherBoard ); m_layers->ImportSettingsFrom( otherBoard ); m_boardFinish->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_TextAndGraphicsOpt->GetValue() ) { static_cast<PANEL_SETUP_TEXT_AND_GRAPHICS*>( m_treebook->ResolvePage( m_textAndGraphicsPage ) ) ->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_FormattingOpt->GetValue() ) { static_cast<PANEL_SETUP_FORMATTING*>( m_treebook->ResolvePage( m_formattingPage ) ) ->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_ConstraintsOpt->GetValue() ) { static_cast<PANEL_SETUP_CONSTRAINTS*>( m_treebook->ResolvePage( m_constraintsPage ) ) ->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_NetclassesOpt->GetValue() ) { static_cast<PANEL_SETUP_NETCLASSES*>( m_treebook->ResolvePage( m_netclassesPage ) ) ->ImportSettingsFrom( otherPrj->GetProjectFile().m_NetSettings ); }
if( importDlg.m_TracksAndViasOpt->GetValue() ) { static_cast<PANEL_SETUP_TRACKS_AND_VIAS*>( m_treebook->ResolvePage( m_tracksAndViasPage ) ) ->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_MaskAndPasteOpt->GetValue() ) { static_cast<PANEL_SETUP_MASK_AND_PASTE*>( m_treebook->ResolvePage( m_maskAndPagePage ) ) ->ImportSettingsFrom( otherBoard ); }
if( importDlg.m_SeveritiesOpt->GetValue() ) { static_cast<PANEL_SETUP_SEVERITIES*>( m_treebook->ResolvePage( m_severitiesPage ) ) ->ImportSettingsFrom( otherBoard->GetDesignSettings().m_DRCSeverities ); }
if( otherPrj != &m_frame->Prj() ) otherBoard->ClearProject(); }
// Clean up and free memory before leaving
if( otherPrj != &m_frame->Prj() ) m_frame->GetSettingsManager()->UnloadProject( otherPrj, false );
delete otherBoard;}
|