KiCad PCB EDA Suite
Loading...
Searching...
No Matches
pcb_base_edit_frame.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 (C) 2014 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Maciej Suminski <[email protected]>
7 *
8 * This program is free software; you can redistribute it and/or
9 * modify it under the terms of the GNU General Public License
10 * as published by the Free Software Foundation; either version 2
11 * of the License, or (at your option) any later version.
12 *
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
17 *
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, you may find one here:
20 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
21 * or you may search the http://www.gnu.org website for the version 2 license,
22 * or you may write to the Free Software Foundation, Inc.,
23 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
24 */
25
26#include <kiface_base.h>
27#include <kiplatform/ui.h>
29#include <pcb_base_edit_frame.h>
31#include <tool/tool_manager.h>
32#include <tools/pcb_actions.h>
35#include <pgm_base.h>
36#include <board.h>
39#include <pcb_dimension.h>
41#include <footprint.h>
42#include <footprint_info_impl.h>
43#include <layer_pairs.h>
44#include <project.h>
52
53#include <widgets/kistatusbar.h>
55#include <id.h>
56
57
59 FRAME_T aFrameType, const wxString& aTitle,
60 const wxPoint& aPos, const wxSize& aSize, long aStyle,
61 const wxString& aFrameName ) :
62 PCB_BASE_FRAME( aKiway, aParent, aFrameType, aTitle, aPos, aSize, aStyle, aFrameName ),
63 m_undoRedoBlocked( false ),
64 m_selectionFilterPanel( nullptr ),
65 m_appearancePanel( nullptr ),
66 m_vertexEditorPane( nullptr ),
67 m_tabbedPanel( nullptr )
68{
69 m_SelLayerBox = nullptr;
71
72 // Do not register the idle event handler if we are running in headless mode.
73 if( !wxApp::GetGUIInstance() )
74 return;
75
76 Bind( wxEVT_IDLE,
77 [this]( wxIdleEvent& aEvent )
78 {
79 // Handle cursor adjustments. While we can get motion and key events through
80 // wxWidgets, we can't get modifier-key-up events.
81 if( m_toolManager )
82 {
84
85 if( selTool )
86 selTool->OnIdle( aEvent );
87 }
88
90 {
93 }
94 } );
95
96 Pgm().GetBackgroundJobMonitor().RegisterStatusBar( static_cast<KISTATUSBAR*>( GetStatusBar() ) );
97}
98
99
106
107
109{
111 wxFileName projectName( Prj().GetProjectFullName() );
112
113 if( mgr->IsProjectOpen() && wxFileName::IsDirWritable( projectName.GetPath() )
114 && projectName.Exists() )
115 {
116 GFootprintList.WriteCacheToFile( Prj().GetProjectPath() + wxT( "fp-info-cache" ) );
117 }
118
119 // Close the project if we are standalone, so it gets cleaned up properly
120 if( mgr->IsProjectOpen() && Kiface().IsSingle() )
121 mgr->UnloadProject( &Prj(), false );
122}
123
124
125bool PCB_BASE_EDIT_FRAME::TryBefore( wxEvent& aEvent )
126{
127 static bool s_presetSwitcherShown = false;
128 static bool s_viewportSwitcherShown = false;
129
130 // wxWidgets generates no key events for the tab key when the ctrl key is held down. One
131 // way around this is to look at all events and inspect the keyboard state of the tab key.
132 // However, this runs into issues on some linux VMs where querying the keyboard state is
133 // very slow. Fortunately we only use ctrl-tab on Mac, so we implement this lovely hack:
134#ifdef __WXMAC__
135 if( wxGetKeyState( WXK_TAB ) )
136#else
137 if( ( aEvent.GetEventType() == wxEVT_CHAR || aEvent.GetEventType() == wxEVT_CHAR_HOOK )
138 && static_cast<wxKeyEvent&>( aEvent ).GetKeyCode() == WXK_TAB )
139#endif
140 {
141 if( !s_presetSwitcherShown && wxGetKeyState( PRESET_SWITCH_KEY ) )
142 {
143 if( m_appearancePanel && this->IsActive() )
144 {
145 const wxArrayString& mru = m_appearancePanel->GetLayerPresetsMRU();
146
147 if( mru.size() > 0 )
148 {
149 EDA_VIEW_SWITCHER switcher( this, mru, PRESET_SWITCH_KEY );
150
151 s_presetSwitcherShown = true;
152 const int switcherDialogRet = switcher.ShowModal();
153 s_presetSwitcherShown = false;
154
155 if( switcherDialogRet == wxID_OK )
156 {
157 int idx = switcher.GetSelection();
158
159 if( idx >= 0 && idx < (int) mru.size() )
160 m_appearancePanel->ApplyLayerPreset( mru[idx] );
161 }
162
163 return true;
164 }
165 }
166 }
167 else if( !s_viewportSwitcherShown && wxGetKeyState( VIEWPORT_SWITCH_KEY ) )
168 {
169 if( m_appearancePanel && this->IsActive() )
170 {
171 const wxArrayString& mru = m_appearancePanel->GetViewportsMRU();
172
173 if( mru.size() > 0 )
174 {
175 EDA_VIEW_SWITCHER switcher( this, mru, VIEWPORT_SWITCH_KEY );
176
177 s_viewportSwitcherShown = true;
178 const int switcherDialogRet = switcher.ShowModal();
179 s_viewportSwitcherShown = false;
180
181 if( switcherDialogRet == wxID_OK )
182 {
183 int idx = switcher.GetSelection();
184
185 if( idx >= 0 && idx < (int) mru.size() )
186 m_appearancePanel->ApplyViewport( mru[idx] );
187 }
188
189 return true;
190 }
191 }
192 }
193 }
194
195 return PCB_BASE_FRAME::TryBefore( aEvent );
196}
197
198
200{
201 // Return a default angle (90 degrees) used for rotate operations.
202 return ANGLE_90;
203}
204
205
212
213
215{
216 bool is_new_board = ( aBoard != m_pcb );
217
218 if( is_new_board )
219 {
220 if( m_toolManager )
222
223 wxCommandEvent e( EDA_EVT_BOARD_CHANGING );
224 ProcessEventLocally( e );
225
226 GetCanvas()->GetView()->Clear();
228 }
229
230 PCB_BASE_FRAME::SetBoard( aBoard, aReporter );
231
233
234 if( is_new_board )
235 {
237 bds.m_DRCEngine = std::make_shared<DRC_ENGINE>( aBoard, &bds );
238 }
239
240 // update the tool manager with the new board and its view.
241 if( m_toolManager )
242 {
243 GetCanvas()->DisplayBoard( aBoard, aReporter );
244
246 m_toolManager->SetEnvironment( aBoard, GetCanvas()->GetView(),
247 GetCanvas()->GetViewControls(), config(), this );
248
249 if( is_new_board )
251 }
252}
253
254
256{
258
259 if( BOARD* board = GetBoard() )
260 {
261 board->UpdateUserUnits( board, GetCanvas()->GetView() );
263 }
264
267}
268
269
271{
273
274 // Update the grid checkbox in the layer widget
276 m_appearancePanel->SetObjectVisible( LAYER_GRID, aVisible );
277}
278
279
281{
283 m_appearancePanel->SetObjectVisible( aLayer, aVisible );
284}
285
286
288{
290 return ::GetColorSettings( cfg ? cfg->m_ColorTheme : DEFAULT_THEME );
291}
292
293
295{
296 if( !GetBoard() )
297 return wxEmptyString;
298
299 wxFileName fn = GetBoard()->GetFileName();
301 return Prj().AbsolutePath( fn.GetFullName() );
302}
303
304
305void PCB_BASE_EDIT_FRAME::handleActivateEvent( wxActivateEvent& aEvent )
306{
308
309 // The text in the collapsible pane headers need to be updated
311 m_appearancePanel->RefreshCollapsiblePanes();
312}
313
314
316{
317 m_appearancePanel->OnDarkModeToggle();
318
320
321 if( viewer )
322 viewer->OnDarkModeToggle();
323}
324
325
327{
328 if( !m_propertiesPanel )
329 return;
330
331 bool show = !m_propertiesPanel->IsShownOnScreen();
332
333 wxAuiPaneInfo& propertiesPaneInfo = m_auimgr.GetPane( PropertiesPaneName() );
334 propertiesPaneInfo.Show( show );
335
337
338 if( show )
339 {
340 SetAuiPaneSize( m_auimgr, propertiesPaneInfo,
341 settings->m_AuiPanels.properties_panel_width, -1 );
342 }
343 else
344 {
346 m_auimgr.Update();
347 }
348}
349
350
351void PCB_BASE_EDIT_FRAME::GetContextualTextVars( BOARD_ITEM* aSourceItem, const wxString& aCrossRef,
352 wxArrayString* aTokens )
353{
354 BOARD* board = aSourceItem->GetBoard();
355
356 if( !aCrossRef.IsEmpty() )
357 {
358 for( FOOTPRINT* candidate : board->Footprints() )
359 {
360 if( candidate->GetReference() == aCrossRef )
361 {
362 candidate->GetContextualTextVars( aTokens );
363 break;
364 }
365 }
366 }
367 else
368 {
369 board->GetContextualTextVars( aTokens );
370
371 if( FOOTPRINT* footprint = aSourceItem->GetParentFootprint() )
372 footprint->GetContextualTextVars( aTokens );
373 }
374}
375
376
378{
379 // Load the toolbar configuration and base controls
381
382 // Layer selector
383 auto layerSelectorFactory =
384 [this]( ACTION_TOOLBAR* aToolbar )
385 {
386 if( !m_SelLayerBox )
387 {
389 m_SelLayerBox->SetBoardFrame( this );
390 }
391
392 m_SelLayerBox->SetToolTip( _( "+/- to switch" ) );
393 m_SelLayerBox->Resync();
394
395 aToolbar->Add( m_SelLayerBox );
396
397 // UI update handler for the control
398 aToolbar->Bind( wxEVT_UPDATE_UI,
399 [this]( wxUpdateUIEvent& aEvent )
400 {
401 if( m_SelLayerBox->GetCount()
402 && ( m_SelLayerBox->GetLayerSelection() != GetActiveLayer() ) )
403 {
404 m_SelLayerBox->SetLayerSelection( GetActiveLayer() );
405 }
406 },
407 m_SelLayerBox->GetId() );
408
409 // Event handler to respond to the user interacting with the control
410 aToolbar->Bind( wxEVT_COMBOBOX,
411 [this]( wxCommandEvent& aEvent )
412 {
413 SetActiveLayer( ToLAYER_ID( m_SelLayerBox->GetLayerSelection() ) );
414
415 if( GetDisplayOptions().m_ContrastModeDisplay != HIGH_CONTRAST_MODE::NORMAL )
416 GetCanvas()->Refresh();
417 },
418 m_SelLayerBox->GetId() );
419 };
420
422}
423
424
426{
428
429 switch( aId )
430 {
431 case ID_ON_LAYER_SELECT: m_SelLayerBox = nullptr; break;
432 }
433}
434
435
437{
438 PCB_SELECTION_FILTER_EVENT evt( aOptions );
439 wxPostEvent( this, evt );
440}
441
443{
444 return wxS( "VertexEditor" );
445}
446
448{
449 if( !m_vertexEditorPane )
450 {
452
453 wxAuiPaneInfo paneInfo = EDA_PANE().Name( VertexEditorPaneName() )
454 .Float()
455 .Caption( _( "Edit Vertices" ) )
456 .PaneBorder( true )
457 .CloseButton( true )
458 .DestroyOnClose( true )
459 .Resizable( true )
460 .MinSize( FromDIP( wxSize( 260, 200 ) ) )
461 .BestSize( FromDIP( wxSize( 260, 320 ) ) )
462 .FloatingSize( FromDIP( wxSize( 320, 360 ) ) );
463
464 m_auimgr.AddPane( m_vertexEditorPane, paneInfo );
465 m_auimgr.Update();
466 }
467
468 m_vertexEditorPane->SetItem( aItem );
469 m_vertexEditorPane->SetFocus();
470}
471
477
479{
481 m_vertexEditorPane->OnSelectionChanged( aItem );
482}
483
KIFACE_BASE & Kiface()
Global KIFACE_BASE "get" accessor.
@ NORMAL
Inactive layers are shown normally (no high-contrast mode)
static ACTION_TOOLBAR_CONTROL layerSelector
Define the structure of a toolbar with buttons that invoke ACTIONs.
wxString m_ColorTheme
Active color theme name.
void UnregisterStatusBar(KISTATUSBAR *aStatusBar)
Removes status bar from handling.
void RegisterStatusBar(KISTATUSBAR *aStatusBar)
Add a status bar for handling.
Container for design settings for a BOARD object.
std::shared_ptr< DRC_ENGINE > m_DRCEngine
const VECTOR2I & GetGridOrigin() const
A base class for any item which can be embedded within the BOARD container class, and therefore insta...
Definition board_item.h:83
virtual const BOARD * GetBoard() const
Return the BOARD in which this BOARD_ITEM resides, or NULL if none.
FOOTPRINT * GetParentFootprint() const
Information pertinent to a Pcbnew printed circuit board.
Definition board.h:322
void GetContextualTextVars(wxArrayString *aVars) const
Definition board.cpp:474
const FOOTPRINTS & Footprints() const
Definition board.h:363
const wxString & GetFileName() const
Definition board.h:359
BOARD_DESIGN_SETTINGS & GetDesignSettings() const
Definition board.cpp:1082
Color settings are a bit different than most of the settings objects in that there can be more than o...
int ShowModal() override
Create and handle a window for the 3d viewer connected to a Kiway and a pcbboard.
virtual APP_SETTINGS_BASE * config() const
Return the settings object used in SaveSettings(), and is overloaded in KICAD_MANAGER_FRAME.
virtual void ClearToolbarControl(int aId)
SETTINGS_MANAGER * GetSettingsManager() const
void RegisterCustomToolbarControlFactory(const ACTION_TOOLBAR_CONTROL &aControlDesc, const ACTION_TOOLBAR_CONTROL_FACTORY &aControlFactory)
Register a creation factory for toolbar controls that are present in this frame.
virtual void configureToolbars()
wxAuiManager m_auimgr
static const wxString PropertiesPaneName()
virtual void ReCreateAuxiliaryToolbar()
virtual void SetGridVisibility(bool aVisible)
PROPERTIES_PANEL * m_propertiesPanel
virtual void Refresh(bool aEraseBackground=true, const wxRect *aRect=nullptr) override
KIGFX::GAL * GetGAL() const
Return a pointer to the GAL instance used in the panel.
Specialization of the wxAuiPaneInfo class for KiCad panels.
static const TOOL_EVENT SelectedItemsModified
Selected items were moved, this can be very high frequency on the canvas, use with care.
Definition actions.h:352
void SetGridOrigin(const VECTOR2D &aGridOrigin)
Set the origin point for the grid.
void Clear()
Remove all items from the view.
Definition view.cpp:1143
void InitPreview()
Definition view.cpp:1722
A minimalistic software bus for communications between various DLLs/DSOs (DSOs) within the same KiCad...
Definition kiway.h:294
AUI_PANELS m_AuiPanels
COLOR_SETTINGS * GetColorSettings(bool aForceRefresh=false) const override
Helper to retrieve the current color settings.
wxString GetDesignRulesPath()
Return the absolute path to the design rules file for the currently-loaded board.
bool m_darkMode
Panel with Layers and Object Inspector tabs.
void SetGridVisibility(bool aVisible) override
Override this function in the PCB_BASE_EDIT_FRAME to refill the layer widget.
virtual EDA_ANGLE GetRotationAngle() const
Return the angle used for rotate operations.
void unitsChangeRefresh() override
Called when when the units setting has changed to allow for any derived classes to handle refreshing ...
void OpenVertexEditor(BOARD_ITEM *aItem)
bool TryBefore(wxEvent &aEvent) override
static wxString VertexEditorPaneName()
PCB_LAYER_BOX_SELECTOR * m_SelLayerBox
virtual void SetBoard(BOARD *aBoard, PROGRESS_REPORTER *aReporter=nullptr) override
Set the #m_Pcb member in such as way as to ensure deleting any previous BOARD.
void ToggleProperties() override
void configureToolbars() override
void ClearToolbarControl(int aId) override
PCB_BASE_EDIT_FRAME(KIWAY *aKiway, wxWindow *aParent, FRAME_T aFrameType, const wxString &aTitle, const wxPoint &aPos, const wxSize &aSize, long aStyle, const wxString &aFrameName)
void HighlightSelectionFilter(const PCB_SELECTION_FILTER_OPTIONS &aOptions)
APPEARANCE_CONTROLS * m_appearancePanel
wxAuiNotebook * m_tabbedPanel
void SetObjectVisible(GAL_LAYER_ID aLayer, bool aVisible=true)
PANEL_SELECTION_FILTER * m_selectionFilterPanel
void GetContextualTextVars(BOARD_ITEM *aSourceItem, const wxString &aCrossRef, wxArrayString *aTokens)
void handleActivateEvent(wxActivateEvent &aEvent) override
Handle a window activation event.
PCB_VERTEX_EDITOR_PANE * m_vertexEditorPane
void UpdateVertexEditorSelection(BOARD_ITEM *aItem)
void ActivateGalCanvas() override
Set the #m_Pcb member in such as way as to ensure deleting any previous BOARD.
void OnVertexEditorPaneClosed(PCB_VERTEX_EDITOR_PANE *aPane)
const PCB_DISPLAY_OPTIONS & GetDisplayOptions() const
Display options control the way tracks, vias, outlines and other things are shown (for instance solid...
void handleActivateEvent(wxActivateEvent &aEvent) override
Handle a window activation event.
virtual void unitsChangeRefresh() override
Called when when the units setting has changed to allow for any derived classes to handle refreshing ...
PCBNEW_SETTINGS * GetPcbNewSettings() const
virtual PCB_LAYER_ID GetActiveLayer() const
PCB_BASE_FRAME(KIWAY *aKiway, wxWindow *aParent, FRAME_T aFrameType, const wxString &aTitle, const wxPoint &aPos, const wxSize &aSize, long aStyle, const wxString &aFrameName)
PCB_DRAW_PANEL_GAL * GetCanvas() const override
Return a pointer to GAL-based canvas of given EDA draw frame.
virtual void ActivateGalCanvas() override
Use to start up the GAL drawing canvas.
virtual void SetBoard(BOARD *aBoard, PROGRESS_REPORTER *aReporter=nullptr)
Set the #m_Pcb member in such as way as to ensure deleting any previous BOARD.
BOARD * GetBoard() const
EDA_3D_VIEWER_FRAME * Get3DViewerFrame()
virtual void SetActiveLayer(PCB_LAYER_ID aLayer)
void UpdateColors()
Update the color settings in the painter and GAL.
virtual KIGFX::PCB_VIEW * GetView() const override
Return a pointer to the #VIEW instance used in the panel.
void DisplayBoard(BOARD *aBoard, PROGRESS_REPORTER *aReporter=nullptr)
Add all items from the current board to the VIEW, so they can be displayed by GAL.
void SyncLayersVisibility(const BOARD *aBoard)
Update "visibility" property of each layer of a given BOARD.
Class to display a pcb layer list in a wxBitmapComboBox.
The selection tool: currently supports:
void OnIdle(wxIdleEvent &aEvent)
virtual BACKGROUND_JOBS_MONITOR & GetBackgroundJobMonitor() const
Definition pgm_base.h:138
A progress reporter interface for use in multi-threaded environments.
virtual const wxString AbsolutePath(const wxString &aFileName) const
Fix up aFileName if it is relative to the project's directory to be an absolute path and filename.
Definition project.cpp:391
bool IsProjectOpen() const
Helper for checking if we have a project open.
bool UnloadProject(PROJECT *aProject, bool aSave=true)
Save, unload and unregister the given PROJECT.
TOOL_MANAGER * m_toolManager
@ MODEL_RELOAD
Model changes (the sheet for a schematic)
Definition tool_base.h:80
#define _(s)
Declaration of the eda_3d_viewer class.
static constexpr EDA_ANGLE ANGLE_90
Definition eda_angle.h:413
#define VIEWPORT_SWITCH_KEY
#define PRESET_SWITCH_KEY
FOOTPRINT_LIST_IMPL GFootprintList
The global footprint info table.
Definition cvpcb.cpp:138
FRAME_T
The set of EDA_BASE_FRAME derivatives, typically stored in EDA_BASE_FRAME::m_Ident.
Definition frame_type.h:33
static const std::string DesignRulesFileExtension
@ ID_ON_LAYER_SELECT
Definition id.h:115
PROJECT & Prj()
Definition kicad.cpp:637
GAL_LAYER_ID
GAL layers are "virtual" layers, i.e.
Definition layer_ids.h:228
@ LAYER_GRID
Definition layer_ids.h:254
PCB_LAYER_ID ToLAYER_ID(int aLayer)
Definition lset.cpp:737
bool IsDarkTheme()
Determine if the desktop interface is currently using a dark theme or a light theme.
Definition wxgtk/ui.cpp:48
PGM_BASE & Pgm()
The global program "get" accessor.
see class PGM_BASE
#define DEFAULT_THEME
This file contains data structures that are saved in the project file or project local settings file ...
VECTOR2< double > VECTOR2D
Definition vector2d.h:694
Definition of file extensions used in Kicad.
void SetAuiPaneSize(wxAuiManager &aManager, wxAuiPaneInfo &aPane, int aWidth, int aHeight)
Sets the size of an AUI pane, working around http://trac.wxwidgets.org/ticket/13180.