KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
dialog_board_setup.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
5 *
6 * This program is free software: you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation, either version 3 of the License, or (at your
9 * option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program. If not, see <http://www.gnu.org/licenses/>.
18 */
19#include <pcb_edit_frame.h>
20#include <panel_setup_layers.h>
25#include <../board_stackup_manager/panel_board_stackup.h>
26#include <../board_stackup_manager/panel_board_finish.h>
27#include <confirm.h>
29#include <kiface_base.h>
30#include <drc/drc_item.h>
32#include <pcb_io/pcb_io.h>
33#include <pcb_io/pcb_io_mgr.h>
42#include <project.h>
48
49#include "dialog_board_setup.h"
50
51#include <footprint.h>
52
53
54#define RESOLVE_PAGE( T, pageIndex ) static_cast<T*>( m_treebook->ResolvePage( pageIndex ) )
55
57 PAGED_DIALOG( aFrame, _( "Board Setup" ), false, false,
58 _( "Import Settings from Another Board..." ), wxSize( 980, 600 ) ),
59 m_frame( aFrame ),
60 m_layers( nullptr ),
61 m_boardFinish( nullptr ),
62 m_physicalStackup( nullptr ),
63 m_currentPage( 0 ),
64 m_layersPage( 0 ),
65 m_physicalStackupPage( 0 ),
66 m_boardFinishPage( 0 ),
67 m_textAndGraphicsPage( 0 ),
68 m_formattingPage( 0 ),
69 m_maskAndPagePage( 0 ),
70 m_constraintsPage( 0 ),
71 m_tracksAndViasPage( 0 ),
72 m_teardropsPage( 0 ),
73 m_tuningPatternsPage( 0 ),
74 m_netclassesPage( 0 ),
75 m_customRulesPage( 0 ),
76 m_severitiesPage( 0 )
77
78{
79 SetEvtHandlerEnabled( false );
80
81 /*
82 * WARNING: If you change page names you MUST update calls to ShowBoardSetupDialog().
83 */
84
85 m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Board Stackup" ) );
86
87 /*
88 * WARNING: Code currently relies on the layers setup coming before the physical stackup panel,
89 * and thus transferring data to the board first. See comment in
90 * PANEL_SETUP_BOARD_STACKUP::TransferDataFromWindow and rework this logic if it is determined
91 * that the order of these pages should be changed.
92 */
93 m_layersPage = m_treebook->GetPageCount();
95 [this]( wxWindow* aParent ) -> wxWindow*
96 {
97 return new PANEL_SETUP_LAYERS( aParent, m_frame );
98 }, _( "Board Editor Layers" ) );
99
100 m_physicalStackupPage = m_treebook->GetPageCount();
102 [this]( wxWindow* aParent ) -> wxWindow*
103 {
107 }, _( "Physical Stackup" ) );
108
109 m_boardFinishPage = m_treebook->GetPageCount();
111 [this]( wxWindow* aParent ) -> wxWindow*
112 {
113 return new PANEL_SETUP_BOARD_FINISH( aParent, m_frame );
114 }, _( "Board Finish" ) );
115
116 m_maskAndPagePage = m_treebook->GetPageCount();
118 [this]( wxWindow* aParent ) -> wxWindow*
119 {
120 return new PANEL_SETUP_MASK_AND_PASTE( aParent, m_frame );
121 }, _( "Solder Mask/Paste" ) );
122
123 m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Text & Graphics" ) );
124
125 m_textAndGraphicsPage = m_treebook->GetPageCount();
127 [this]( wxWindow* aParent ) -> wxWindow*
128 {
129 return new PANEL_SETUP_TEXT_AND_GRAPHICS( aParent, m_frame );
130 }, _( "Defaults" ) );
131
132 m_formattingPage = m_treebook->GetPageCount();
134 [this]( wxWindow* aParent ) -> wxWindow*
135 {
136 return new PANEL_SETUP_FORMATTING( aParent, m_frame );
137 }, _( "Formatting" ) );
138
140 [this]( wxWindow* aParent ) -> wxWindow*
141 {
142 return new PANEL_TEXT_VARIABLES( aParent, &Prj() );
143 }, _( "Text Variables" ) );
144
145 m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Design Rules" ) );
146
147 m_constraintsPage = m_treebook->GetPageCount();
149 [this]( wxWindow* aParent ) -> wxWindow*
150 {
151 return new PANEL_SETUP_CONSTRAINTS( aParent, m_frame );
152 }, _( "Constraints" ) );
153
154 m_tracksAndViasPage = m_treebook->GetPageCount();
156 [this]( wxWindow* aParent ) -> wxWindow*
157 {
158 return new PANEL_SETUP_TRACKS_AND_VIAS( aParent, m_frame );
159 }, _( "Pre-defined Sizes" ) );
160
161 m_teardropsPage = m_treebook->GetPageCount();
163 [this]( wxWindow* aParent ) -> wxWindow*
164 {
165 return new PANEL_SETUP_TEARDROPS( aParent, m_frame );
166 }, _( "Teardrops" ) );
167
168 m_tuningPatternsPage = m_treebook->GetPageCount();
170 [this]( wxWindow* aParent ) -> wxWindow*
171 {
173
174 return new PANEL_SETUP_TUNING_PATTERNS( aParent, m_frame,
178 }, _( "Length-tuning Patterns" ) );
179
180 m_netclassesPage = m_treebook->GetPageCount();
182 [this]( wxWindow* aParent ) -> wxWindow*
183 {
184 BOARD* board = m_frame->GetBoard();
185 return new PANEL_SETUP_NETCLASSES( aParent, m_frame,
188 false );
189 }, _( "Net Classes" ) );
190
191 m_componentClassesPage = m_treebook->GetPageCount();
193 [this]( wxWindow* aParent ) -> wxWindow*
194 {
195 // Construct the panel
198 this );
199 },
200 _( "Component Classes" ) );
201
202 m_customRulesPage = m_treebook->GetPageCount();
204 [this]( wxWindow* aParent ) -> wxWindow*
205 {
206 return new PANEL_SETUP_RULES( aParent, m_frame );
207 }, _( "Custom Rules" ) );
208
209 m_severitiesPage = m_treebook->GetPageCount();
211 [this]( wxWindow* aParent ) -> wxWindow*
212 {
213 BOARD* board = m_frame->GetBoard();
216 }, _( "Violation Severity" ) );
217
218 m_treebook->AddPage( new wxPanel( GetTreebook() ), _( "Board Data" ) );
219 m_embeddedFilesPage = m_treebook->GetPageCount();
221 [this]( wxWindow* aParent ) -> wxWindow*
222 {
223 return new PANEL_EMBEDDED_FILES( aParent, m_frame->GetBoard() );
224 }, _( "Embedded Files" ) );
225
226 for( size_t i = 0; i < m_treebook->GetPageCount(); ++i )
227 m_treebook->ExpandNode( i );
228
229 SetEvtHandlerEnabled( true );
230
232
233 if( Prj().IsReadOnly() )
234 {
235 m_infoBar->ShowMessage( _( "Project is missing or read-only. Some settings will not "
236 "be editable." ), wxICON_WARNING );
237 }
238
239 wxBookCtrlEvent evt( wxEVT_TREEBOOK_PAGE_CHANGED, wxID_ANY, 0 );
240
241 wxQueueEvent( m_treebook, evt.Clone() );
242}
243
244
246{
247}
248
249
250void DIALOG_BOARD_SETUP::onPageChanged( wxBookCtrlEvent& aEvent )
251{
253
254 size_t page = aEvent.GetSelection();
255
256 if( m_physicalStackupPage > 0 ) // Don't run this during initialization
257 {
259 {
262 }
263
264 // Ensure layer page always gets updated even if we aren't moving towards it
267
268 if( page == m_physicalStackupPage )
270 else if( Prj().IsReadOnly() )
271 KIUI::Disable( m_treebook->GetPage( page ) );
272 }
273
274 m_currentPage = page;
275}
276
277
278void DIALOG_BOARD_SETUP::onAuxiliaryAction( wxCommandEvent& aEvent )
279{
280 DIALOG_IMPORT_SETTINGS importDlg( this, m_frame );
281
282 if( importDlg.ShowModal() == wxID_CANCEL )
283 return;
284
285 wxFileName boardFn( importDlg.GetFilePath() );
286 wxFileName projectFn( boardFn );
287
288 projectFn.SetExt( FILEEXT::ProjectFileExtension );
289
290 if( !m_frame->GetSettingsManager()->LoadProject( projectFn.GetFullPath(), false ) )
291 {
292 wxString msg = wxString::Format( _( "Error importing settings from board:\n"
293 "Associated project file %s could not be loaded" ),
294 projectFn.GetFullPath() );
295 DisplayErrorMessage( this, msg );
296
297 return;
298 }
299
303
304 // Flag so user can stop work if it will result in deleted inner copper layers
305 // and still clean up this function properly.
306 bool okToProceed = true;
307
308 PROJECT* otherPrj = m_frame->GetSettingsManager()->GetProject( projectFn.GetFullPath() );
309
311 BOARD* otherBoard = nullptr;
312
313 try
314 {
315 WX_PROGRESS_REPORTER progressReporter( this, _( "Loading PCB" ), 1 );
316
317 pi->SetProgressReporter( &progressReporter );
318
319 otherBoard = pi->LoadBoard( boardFn.GetFullPath(), nullptr );
320
321 if( importDlg.m_LayersOpt->GetValue() )
322 {
323 BOARD* loadedBoard = m_frame->GetBoard();
324
325 // Check if "Import Settings" board has more layers than the current board.
326 okToProceed = m_layers->CheckCopperLayerCount( loadedBoard, otherBoard );
327 }
328 }
329 catch( const IO_ERROR& ioe )
330 {
331 // You wouldn't think boardFn.GetFullPath() would throw, but we get a stack buffer
332 // underflow from ASAN. While it's probably an ASAN error, a second try/catch doesn't
333 // cost us much.
334 try
335 {
336 if( ioe.Problem() != wxT( "CANCEL" ) )
337 {
338 wxString msg = wxString::Format( _( "Error loading board file:\n%s" ),
339 boardFn.GetFullPath() );
340 DisplayErrorMessage( this, msg, ioe.What() );
341 }
342
343 if( otherPrj != &m_frame->Prj() )
344 m_frame->GetSettingsManager()->UnloadProject( otherPrj, false );
345 }
346 catch(...)
347 {
348 // That was already our best-efforts
349 }
350
351 return;
352 }
353
354 if( okToProceed )
355 {
356 otherBoard->SetProject( otherPrj );
357
358 // If layers options are imported, import also the stackup
359 // layers options and stackup are linked, so they cannot be imported
360 // separately, and stackup can be imported only after layers options
361 if( importDlg.m_LayersOpt->GetValue() )
362 {
364 m_layers->ImportSettingsFrom( otherBoard );
365 m_boardFinish->ImportSettingsFrom( otherBoard );
366 }
367
368 if( importDlg.m_TextAndGraphicsOpt->GetValue() )
369 {
371 m_textAndGraphicsPage )->ImportSettingsFrom( otherBoard );
372 }
373
374 if( importDlg.m_FormattingOpt->GetValue() )
375 {
377 m_formattingPage )->ImportSettingsFrom( otherBoard );
378 }
379
380 if( importDlg.m_ConstraintsOpt->GetValue() )
381 {
383 m_constraintsPage )->ImportSettingsFrom( otherBoard );
384 }
385
386 if( importDlg.m_NetclassesOpt->GetValue() )
387 {
388 PROJECT_FILE& otherProjectFile = otherPrj->GetProjectFile();
389
391 m_netclassesPage )->ImportSettingsFrom( otherProjectFile.m_NetSettings );
392 }
393
394 if( importDlg.m_ComponentClassesOpt->GetValue() )
395 {
396 PROJECT_FILE& otherProjectFile = otherPrj->GetProjectFile();
397
399 ->ImportSettingsFrom( otherProjectFile.m_ComponentClassSettings );
400 }
401
402 if( importDlg.m_TracksAndViasOpt->GetValue() )
403 {
405 m_tracksAndViasPage )->ImportSettingsFrom( otherBoard );
406 }
407
408 if( importDlg.m_TeardropsOpt->GetValue() )
409 {
411 m_teardropsPage )->ImportSettingsFrom( otherBoard );
412 }
413
414 if( importDlg.m_TuningPatternsOpt->GetValue() )
415 {
417 m_tuningPatternsPage )->ImportSettingsFrom( otherBoard );
418 }
419
420 if( importDlg.m_MaskAndPasteOpt->GetValue() )
421 {
423 m_maskAndPagePage )->ImportSettingsFrom( otherBoard );
424 }
425
426 if( importDlg.m_CustomRulesOpt->GetValue() )
427 {
429 m_customRulesPage )->ImportSettingsFrom( otherBoard );
430 }
431
432 if( importDlg.m_SeveritiesOpt->GetValue() )
433 {
434 BOARD_DESIGN_SETTINGS& otherSettings = otherBoard->GetDesignSettings();
435
437 m_severitiesPage )->ImportSettingsFrom( otherSettings.m_DRCSeverities );
438 }
439
440 if( otherPrj != &m_frame->Prj() )
441 otherBoard->ClearProject();
442 }
443
444 // Clean up and free memory before leaving
445 if( otherPrj != &m_frame->Prj() )
446 m_frame->GetSettingsManager()->UnloadProject( otherPrj, false );
447
448 delete otherBoard;
449}
Container for design settings for a BOARD object.
std::map< int, SEVERITY > m_DRCSeverities
PNS::MEANDER_SETTINGS m_DiffPairMeanderSettings
PNS::MEANDER_SETTINGS m_SingleTrackMeanderSettings
PNS::MEANDER_SETTINGS m_SkewMeanderSettings
Information pertinent to a Pcbnew printed circuit board.
Definition: board.h:297
std::set< wxString > GetNetClassAssignmentCandidates() const
Return the set of netname candidates for netclass assignment.
Definition: board.cpp:2121
void SetProject(PROJECT *aProject, bool aReferenceOnly=false)
Link a board to a given project.
Definition: board.cpp:195
void ClearProject()
Definition: board.cpp:233
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition: board.cpp:946
DIALOG_BOARD_SETUP(PCB_EDIT_FRAME *aFrame)
PCB_EDIT_FRAME * m_frame
void onAuxiliaryAction(wxCommandEvent &aEvent) override
PANEL_SETUP_BOARD_STACKUP * m_physicalStackup
PANEL_SETUP_BOARD_FINISH * m_boardFinish
PANEL_SETUP_LAYERS * m_layers
void onPageChanged(wxBookCtrlEvent &aEvent) override
void finishDialogSettings()
In all dialogs, we must call the same functions to fix minimal dlg size, the default position and per...
int ShowModal() override
static std::vector< std::reference_wrapper< RC_ITEM > > GetItemsWithSeverities(bool aIncludeDeprecated=false)
Definition: drc_item.h:133
SETTINGS_MANAGER * GetSettingsManager() const
Hold an error message and may be used when throwing exceptions containing meaningful error messages.
Definition: ki_exception.h:77
virtual const wxString What() const
A composite of Problem() and Where()
Definition: exceptions.cpp:30
virtual const wxString Problem() const
what was the problem?
Definition: exceptions.cpp:46
PROJECT & Prj() const
Return a reference to the PROJECT associated with this KIWAY.
WX_TREEBOOK * GetTreebook()
Definition: paged_dialog.h:39
WX_INFOBAR * m_infoBar
Definition: paged_dialog.h:72
WX_TREEBOOK * m_treebook
Definition: paged_dialog.h:68
virtual void onPageChanged(wxBookCtrlEvent &aEvent)
void ImportSettingsFrom(BOARD *aBoard)
void ImportSettingsFrom(BOARD *aBoard)
void OnLayersOptionsChanged(const LSET &aNewLayerSet)
Must be called if the copper layers count has changed or solder mask, solder paste or silkscreen laye...
bool CheckCopperLayerCount(BOARD *aWorkingBoard, BOARD *aImportedBoard)
Check and warn if inner copper layers will be deleted.
void SyncCopperLayers(int aNumCopperLayers)
Called when switching to this tab to make sure that any changes to the copper layer count made on the...
void ImportSettingsFrom(BOARD *aBoard)
BOARD * GetBoard() const
The main frame for Pcbnew.
static PCB_IO * PluginFind(PCB_FILE_T aFileType)
Return a #PLUGIN which the caller can use to import, export, save, or load design documents.
Definition: pcb_io_mgr.cpp:69
@ KICAD_SEXP
S-expression Pcbnew file format.
Definition: pcb_io_mgr.h:58
The backing store for a PROJECT, in JSON format.
Definition: project_file.h:73
std::shared_ptr< NET_SETTINGS > & NetSettings()
Definition: project_file.h:104
std::shared_ptr< NET_SETTINGS > m_NetSettings
Net settings for this project (owned here)
Definition: project_file.h:189
std::shared_ptr< COMPONENT_CLASS_SETTINGS > & ComponentClassSettings()
Definition: project_file.h:109
std::shared_ptr< COMPONENT_CLASS_SETTINGS > m_ComponentClassSettings
Component class settings for the project (owned here)
Definition: project_file.h:194
Container for project specific data.
Definition: project.h:64
virtual PROJECT_FILE & GetProjectFile() const
Definition: project.h:203
bool LoadProject(const wxString &aFullPath, bool aSetActive=true)
Load a project or sets up a new project with a specified path.
PROJECT * GetProject(const wxString &aFullPath) const
Retrieve a loaded project by name.
bool UnloadProject(PROJECT *aProject, bool aSave=true)
Save, unload and unregister the given PROJECT.
void ShowMessage(const wxString &aMessage, int aFlags=wxICON_INFORMATION) override
Show the info bar with the provided message and icon.
Definition: wx_infobar.cpp:154
Multi-thread safe progress reporter dialog, intended for use of tasks that parallel reporting back of...
bool AddLazySubPage(std::function< wxWindow *(wxWindow *aParent)> aLazyCtor, const wxString &text, bool bSelect=false, int imageId=NO_IMAGE)
Definition: wx_treebook.cpp:96
void DisplayErrorMessage(wxWindow *aParent, const wxString &aText, const wxString &aExtraInfo)
Display an error message with aMessage.
Definition: confirm.cpp:194
This file is part of the common library.
#define RESOLVE_PAGE(T, pageIndex)
#define _(s)
static const std::string ProjectFileExtension
std::unique_ptr< T > IO_RELEASER
Helper to hold and release an IO_BASE object when exceptions are thrown.
Definition: io_mgr.h:33
KICOMMON_API void Disable(wxWindow *aWindow)
Makes a window read-only.
Definition: ui_common.cpp:326
Definition of file extensions used in Kicad.