KiCad PCB EDA Suite
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages Concepts
properties_panel.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) 2020-2023 CERN
5 * Copyright The KiCad Developers, see AUTHORS.txt for contributors.
6 * @author Maciej Suminski <maciej.suminski@cern.ch>
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 3
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 along
19 * with this program. If not, see <http://www.gnu.org/licenses/>.
20 */
21
22#include "properties_panel.h"
23#include <tool/selection.h>
24#include <eda_base_frame.h>
25#include <eda_item.h>
26#include <import_export.h>
27#include <pgm_base.h>
29
30#include <algorithm>
31#include <set>
32
33#include <wx/settings.h>
34#include <wx/stattext.h>
35#include <wx/propgrid/advprops.h>
36
37
38// This is provided by wx >3.3.0
39#if !wxCHECK_VERSION( 3, 3, 0 )
40extern APIIMPORT wxPGGlobalVarsClass* wxPGGlobalVars;
41#endif
42
44 wxPanel( aParent ),
45 m_SuppressGridChangeEvents( 0 ),
46 m_frame( aFrame ),
47 m_splitter_key_proportion( -1 )
48{
49 wxBoxSizer* mainSizer = new wxBoxSizer( wxVERTICAL );
50
51 // on some platforms wxPGGlobalVars is initialized automatically,
52 // but others need an explicit init
53 if( !wxPGGlobalVars )
54 wxPGInitResourceModule();
55
56 // See https://gitlab.com/kicad/code/kicad/-/issues/12297
57 // and https://github.com/wxWidgets/wxWidgets/issues/11787
58 if( wxPGGlobalVars->m_mapEditorClasses.empty() )
59 {
60 wxPGEditor_TextCtrl = nullptr;
61 wxPGEditor_Choice = nullptr;
62 wxPGEditor_ComboBox = nullptr;
63 wxPGEditor_TextCtrlAndButton = nullptr;
64 wxPGEditor_CheckBox = nullptr;
65 wxPGEditor_ChoiceAndButton = nullptr;
66 wxPGEditor_SpinCtrl = nullptr;
67 wxPGEditor_DatePickerCtrl = nullptr;
68 }
69
70 if( !Pgm().m_PropertyGridInitialized )
71 {
72 delete wxPGGlobalVars->m_defaultRenderer;
73 wxPGGlobalVars->m_defaultRenderer = new PG_CELL_RENDERER();
75 }
76
77 m_caption = new wxStaticText( this, wxID_ANY, _( "No objects selected" ) );
78 mainSizer->Add( m_caption, 0, wxALL | wxEXPAND, 5 );
79
80 m_grid = new wxPropertyGrid( this );
81 m_grid->SetUnspecifiedValueAppearance( wxPGCell( wxT( "<...>" ) ) );
82 m_grid->SetExtraStyle( wxPG_EX_HELP_AS_TOOLTIPS );
83
84#if wxCHECK_VERSION( 3, 3, 0 )
85 m_grid->SetValidationFailureBehavior( wxPGVFBFlags::MarkCell );
86#else
87 m_grid->SetValidationFailureBehavior( wxPG_VFB_MARK_CELL );
88#endif
89
90#if wxCHECK_VERSION( 3, 3, 0 )
91 m_grid->AddActionTrigger( wxPGKeyboardAction::NextProperty, WXK_RETURN );
92 m_grid->AddActionTrigger( wxPGKeyboardAction::NextProperty, WXK_NUMPAD_ENTER );
93 m_grid->AddActionTrigger( wxPGKeyboardAction::NextProperty, WXK_DOWN );
94 m_grid->AddActionTrigger( wxPGKeyboardAction::PrevProperty, WXK_UP );
95 m_grid->AddActionTrigger( wxPGKeyboardAction::Edit, WXK_SPACE );
96#else
97 m_grid->AddActionTrigger( wxPG_ACTION_NEXT_PROPERTY, WXK_RETURN );
98 m_grid->AddActionTrigger( wxPG_ACTION_NEXT_PROPERTY, WXK_NUMPAD_ENTER );
99 m_grid->AddActionTrigger( wxPG_ACTION_NEXT_PROPERTY, WXK_DOWN );
100 m_grid->AddActionTrigger( wxPG_ACTION_PREV_PROPERTY, WXK_UP );
101 m_grid->AddActionTrigger( wxPG_ACTION_EDIT, WXK_SPACE );
102#endif
103
104 m_grid->DedicateKey( WXK_RETURN );
105 m_grid->DedicateKey( WXK_NUMPAD_ENTER );
106 m_grid->DedicateKey( WXK_DOWN );
107 m_grid->DedicateKey( WXK_UP );
108 mainSizer->Add( m_grid, 1, wxEXPAND, 5 );
109
110 m_grid->SetCellDisabledTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_GRAYTEXT ) );
111
112#ifdef __WXGTK__
113 // Needed for dark mode, on wx 3.0 at least.
114 m_grid->SetCaptionTextColour( wxSystemSettings::GetColour( wxSYS_COLOUR_CAPTIONTEXT ) );
115#endif
116
117 SetFont( KIUI::GetDockedPaneFont( this ) );
118
119 SetSizer( mainSizer );
120 Layout();
121
122 m_grid->CenterSplitter();
123
124 Connect( wxEVT_CHAR_HOOK, wxKeyEventHandler( PROPERTIES_PANEL::onCharHook ), nullptr, this );
125 Connect( wxEVT_PG_CHANGED, wxPropertyGridEventHandler( PROPERTIES_PANEL::valueChanged ),
126 nullptr, this );
127 Connect( wxEVT_PG_CHANGING, wxPropertyGridEventHandler( PROPERTIES_PANEL::valueChanging ),
128 nullptr, this );
129 Connect( wxEVT_SHOW, wxShowEventHandler( PROPERTIES_PANEL::onShow ), nullptr, this );
130
131 Bind( wxEVT_PG_COL_END_DRAG,
132 [&]( wxPropertyGridEvent& )
133 {
135 static_cast<float>( m_grid->GetSplitterPosition() ) / m_grid->GetSize().x;
136 } );
137
138 Bind( wxEVT_SIZE,
139 [&]( wxSizeEvent& aEvent )
140 {
141 CallAfter( [this]()
142 {
144 } );
145 aEvent.Skip();
146 } );
147
148 m_frame->Bind( EDA_LANG_CHANGED, &PROPERTIES_PANEL::OnLanguageChanged, this );
149}
150
151
153{
154 m_frame->Unbind( EDA_LANG_CHANGED, &PROPERTIES_PANEL::OnLanguageChanged, this );
155}
156
157
158void PROPERTIES_PANEL::OnLanguageChanged( wxCommandEvent& aEvent )
159{
160 if( m_grid->IsEditorFocused() )
161 m_grid->CommitChangesFromEditor();
162
163 m_grid->Clear();
164 m_displayed.clear();
165
166 UpdateData();
167
168 aEvent.Skip();
169}
170
171
173{
174public:
176 m_panel( aPanel )
177 {
179 }
180
182 {
184 }
185
186private:
188};
189
190
192{
193 SUPPRESS_GRID_CHANGED_EVENTS raii( this );
194
195 auto reset =
196 [&]()
197 {
198 if( m_grid->IsEditorFocused() )
199 m_grid->CommitChangesFromEditor();
200
201 m_grid->Clear();
202 m_displayed.clear();
203 };
204
205 if( aSelection.Empty() )
206 {
207 m_caption->SetLabel( _( "No objects selected" ) );
208 reset();
209 return;
210 }
211 else if( aSelection.Size() == 1 )
212 {
213 m_caption->SetLabel( aSelection.Front()->GetFriendlyName() );
214 }
215 else
216 {
217 m_caption->SetLabel( wxString::Format( _( "%d objects selected" ), aSelection.Size() ) );
218 }
219
220 // Get all the selected types
221 std::set<TYPE_ID> types;
222
223 for( EDA_ITEM* item : aSelection )
224 types.insert( TYPE_HASH( *item ) );
225
226 wxCHECK( !types.empty(), /* void */ ); // already guarded above, but Coverity doesn't know that
227
229 std::set<PROPERTY_BASE*> commonProps;
230 const PROPERTY_LIST& allProperties = propMgr.GetProperties( *types.begin() );
231
232 copy( allProperties.begin(), allProperties.end(),
233 inserter( commonProps, commonProps.begin() ) );
234
235 PROPERTY_DISPLAY_ORDER displayOrder = propMgr.GetDisplayOrder( *types.begin() );
236
237 std::vector<wxString> groupDisplayOrder = propMgr.GetGroupDisplayOrder( *types.begin() );
238 std::set<wxString> groups( groupDisplayOrder.begin(), groupDisplayOrder.end() );
239
240 std::set<PROPERTY_BASE*> availableProps;
241
242 // Get all possible properties
243 for( const TYPE_ID& type : types )
244 {
245 const PROPERTY_LIST& itemProps = propMgr.GetProperties( type );
246
247 const PROPERTY_DISPLAY_ORDER& itemDisplayOrder = propMgr.GetDisplayOrder( type );
248
249 copy( itemDisplayOrder.begin(), itemDisplayOrder.end(),
250 inserter( displayOrder, displayOrder.begin() ) );
251
252 const std::vector<wxString>& itemGroups = propMgr.GetGroupDisplayOrder( type );
253
254 for( const wxString& group : itemGroups )
255 {
256 if( !groups.count( group ) )
257 {
258 groupDisplayOrder.emplace_back( group );
259 groups.insert( group );
260 }
261 }
262
263 for( auto it = commonProps.begin(); it != commonProps.end(); /* ++it in the loop */ )
264 {
265 if( !binary_search( itemProps.begin(), itemProps.end(), *it ) )
266 it = commonProps.erase( it );
267 else
268 ++it;
269 }
270 }
271
272 EDA_ITEM* firstItem = aSelection.Front();
273
274 bool isLibraryEditor = m_frame->IsType( FRAME_FOOTPRINT_EDITOR )
276
277 bool isDesignEditor = m_frame->IsType( FRAME_PCB_EDITOR )
278 || m_frame->IsType( FRAME_SCH );
279
280 // Find a set of properties that is common to all selected items
281 for( PROPERTY_BASE* property : commonProps )
282 {
283 if( property->IsHiddenFromPropertiesManager() )
284 continue;
285
286 if( isLibraryEditor && property->IsHiddenFromLibraryEditors() )
287 continue;
288
289 if( isDesignEditor && property->IsHiddenFromDesignEditors() )
290 continue;
291
292 if( propMgr.IsAvailableFor( TYPE_HASH( *firstItem ), property, firstItem ) )
293 availableProps.insert( property );
294 }
295
296 bool writeable = true;
297 std::set<PROPERTY_BASE*> existingProps;
298
299 for( wxPropertyGridIterator it = m_grid->GetIterator(); !it.AtEnd(); it.Next() )
300 {
301 wxPGProperty* pgProp = it.GetProperty();
302 PROPERTY_BASE* property = propMgr.GetProperty( TYPE_HASH( *firstItem ), pgProp->GetName() );
303
304 // Switching item types? Property may no longer be valid
305 if( !property )
306 continue;
307
308 wxVariant commonVal;
309
310 extractValueAndWritability( aSelection, property, commonVal, writeable );
311 pgProp->SetValue( commonVal );
312 pgProp->Enable( writeable );
313
314 existingProps.insert( property );
315 }
316
317 if( !existingProps.empty() && existingProps == availableProps )
318 return;
319
320 // Some difference exists: start from scratch
321 reset();
322
323 std::map<wxPGProperty*, int> pgPropOrders;
324 std::map<wxString, std::vector<wxPGProperty*>> pgPropGroups;
325
326 for( PROPERTY_BASE* property : availableProps )
327 {
328 wxPGProperty* pgProp = createPGProperty( property );
329 wxVariant commonVal;
330
331 if( !extractValueAndWritability( aSelection, property, commonVal, writeable ) )
332 continue;
333
334 if( pgProp )
335 {
336 pgProp->SetValue( commonVal );
337 pgProp->Enable( writeable );
338 m_displayed.push_back( property );
339
340 wxASSERT( displayOrder.count( property ) );
341 pgPropOrders[pgProp] = displayOrder[property];
342 pgPropGroups[property->Group()].emplace_back( pgProp );
343 }
344 }
345
346 const wxString unspecifiedGroupCaption = _( "Basic Properties" );
347
348 for( const wxString& groupName : groupDisplayOrder )
349 {
350 if( !pgPropGroups.count( groupName ) )
351 continue;
352
353 std::vector<wxPGProperty*>& properties = pgPropGroups[groupName];
354 wxString groupCaption = wxGetTranslation( groupName );
355
356 auto groupItem = new wxPropertyCategory( groupName.IsEmpty() ? unspecifiedGroupCaption
357 : groupCaption );
358
359 m_grid->Append( groupItem );
360
361 std::sort( properties.begin(), properties.end(),
362 [&]( wxPGProperty*& aFirst, wxPGProperty*& aSecond )
363 {
364 return pgPropOrders[aFirst] < pgPropOrders[aSecond];
365 } );
366
367 for( wxPGProperty* property : properties )
368 m_grid->Append( property );
369 }
370
372}
373
374
375bool PROPERTIES_PANEL::getItemValue( EDA_ITEM* aItem, PROPERTY_BASE* aProperty, wxVariant& aValue )
376{
377 const wxAny& any = aItem->Get( aProperty );
378 bool converted = false;
379
380 if( aProperty->HasChoices() )
381 {
382 // handle enums as ints, since there are no default conversion functions for wxAny
383 int tmp;
384 converted = any.GetAs<int>( &tmp );
385
386 if( converted )
387 aValue = wxVariant( tmp );
388 }
389
390 if( !converted ) // all other types
391 converted = any.GetAs( &aValue );
392
393 if( !converted )
394 {
395 wxString propName = aProperty->Name();
396 propName.Replace( ' ', '_' );
397 wxFAIL_MSG( wxString::Format( wxS( "Could not convert wxAny to wxVariant for %s::%s" ),
398 aItem->GetClass(),
399 propName ) );
400 }
401
402 return converted;
403}
404
405
407 PROPERTY_BASE* aProperty,
408 wxVariant& aValue, bool& aWritable )
409{
411 bool different = false;
412 wxVariant commonVal;
413
414 aWritable = true;
415
416 for( EDA_ITEM* item : aSelection )
417 {
418 if( !propMgr.IsAvailableFor( TYPE_HASH( *item ), aProperty, item ) )
419 return false;
420
421 if( aProperty->IsHiddenFromPropertiesManager() )
422 return false;
423
424 // If read-only for any of the selection, read-only for the whole selection.
425 if( !propMgr.IsWriteableFor( TYPE_HASH( *item ), aProperty, item ) )
426 aWritable = false;
427
428 wxVariant value;
429
430 if( getItemValue( item, aProperty, value ) )
431 {
432 // Null value indicates different property values between items
433 if( !different && !aValue.IsNull() && value != aValue )
434 {
435 different = true;
436 aValue.MakeNull();
437 }
438 else if( !different )
439 {
440 aValue = value;
441 }
442 }
443 else
444 {
445 // getItemValue returned false -- not available for this item
446 return false;
447 }
448 }
449
450 return true;
451}
452
453
454void PROPERTIES_PANEL::onShow( wxShowEvent& aEvent )
455{
456 if( aEvent.IsShown() )
457 UpdateData();
458
459 aEvent.Skip();
460}
461
462
463void PROPERTIES_PANEL::onCharHook( wxKeyEvent& aEvent )
464{
465 // m_grid->IsAnyModified() doesn't work for the first modification
466 if( aEvent.GetKeyCode() == WXK_TAB && !aEvent.ShiftDown() )
467 {
468 wxVariant oldValue = m_grid->GetSelectedProperty()->GetValue();
469
470 m_grid->CommitChangesFromEditor();
471
472 // If there was no change, treat it as a navigation key
473 if( oldValue == m_grid->GetSelectedProperty()->GetValue() )
474 aEvent.Skip();
475
476 return;
477 }
478
479 if( aEvent.GetKeyCode() == WXK_SPACE )
480 {
481 if( wxPGProperty* prop = m_grid->GetSelectedProperty() )
482 {
483 if( prop->GetValueType() == wxT( "bool" ) )
484 {
485 m_grid->SetPropertyValue( prop, !prop->GetValue().GetBool() );
486 return;
487 }
488 }
489 }
490
491 if( aEvent.GetKeyCode() == WXK_RETURN || aEvent.GetKeyCode() == WXK_NUMPAD_ENTER )
492 {
493 m_grid->CommitChangesFromEditor();
494 /* don't skip this one; if we're not the last property we'll also go to the next row */
495 }
496
497 aEvent.Skip();
498}
499
500
502{
504 m_grid->CenterSplitter();
505 else
506 m_grid->SetSplitterPosition( m_splitter_key_proportion * m_grid->GetSize().x );
507}
508
509
511{
512 m_splitter_key_proportion = aProportion;
514}
The base frame for deriving all KiCad main window classes.
bool IsType(FRAME_T aType) const
A base class for most all the KiCad significant classes used in schematics and boards.
Definition: eda_item.h:96
virtual wxString GetFriendlyName() const
Definition: eda_item.cpp:374
virtual wxString GetClass() const =0
Return the class name.
wxAny Get(PROPERTY_BASE *aProperty) const
Definition: inspectable.h:100
bool m_PropertyGridInitialized
Definition: pgm_base.h:384
Enhanced renderer to work around some limitations in wxWidgets 3.0 capabilities.
float m_splitter_key_proportion
Proportion of the grid column splitter that is used for the key column (0.0 - 1.0)
PROPERTIES_PANEL(wxWindow *aParent, EDA_BASE_FRAME *aFrame)
virtual void valueChanging(wxPropertyGridEvent &aEvent)
bool extractValueAndWritability(const SELECTION &aSelection, PROPERTY_BASE *aProperty, wxVariant &aValue, bool &aWritable)
Processes a selection and determines whether the given property should be available or not and what t...
wxPropertyGrid * m_grid
wxStaticText * m_caption
void SetSplitterProportion(float aProportion)
std::vector< PROPERTY_BASE * > m_displayed
virtual void OnLanguageChanged(wxCommandEvent &aEvent)
virtual void UpdateData()=0
virtual void valueChanged(wxPropertyGridEvent &aEvent)
bool getItemValue(EDA_ITEM *aItem, PROPERTY_BASE *aProperty, wxVariant &aValue)
Utility to fetch a property value and convert to wxVariant Precondition: aItem is known to have prope...
virtual void rebuildProperties(const SELECTION &aSelection)
Generates the property grid for a given selection of items.
void onCharHook(wxKeyEvent &aEvent)
virtual wxPGProperty * createPGProperty(const PROPERTY_BASE *aProperty) const =0
void onShow(wxShowEvent &aEvent)
EDA_BASE_FRAME * m_frame
bool IsHiddenFromPropertiesManager() const
Definition: property.h:299
virtual bool HasChoices() const
Return true if this PROPERTY has a limited set of possible values.
Definition: property.h:241
const wxString & Name() const
Definition: property.h:217
Provide class metadata.Helper macro to map type hashes to names.
Definition: property_mgr.h:85
const PROPERTY_LIST & GetProperties(TYPE_ID aType) const
Return all properties for a specific type.
const PROPERTY_DISPLAY_ORDER & GetDisplayOrder(TYPE_ID aType) const
bool IsWriteableFor(TYPE_ID aItemClass, PROPERTY_BASE *aProp, INSPECTABLE *aItem)
Checks overriden availability and original availability of a property, returns false if the property ...
static PROPERTY_MANAGER & Instance()
Definition: property_mgr.h:87
PROPERTY_BASE * GetProperty(TYPE_ID aType, const wxString &aProperty) const
Return a property for a specific type.
bool IsAvailableFor(TYPE_ID aItemClass, PROPERTY_BASE *aProp, INSPECTABLE *aItem)
Checks overriden availability and original availability of a property, returns false if the property ...
const std::vector< wxString > & GetGroupDisplayOrder(TYPE_ID aType) const
EDA_ITEM * Front() const
Definition: selection.h:172
int Size() const
Returns the number of selected parts.
Definition: selection.h:116
bool Empty() const
Checks if there is anything selected.
Definition: selection.h:110
SUPPRESS_GRID_CHANGED_EVENTS(PROPERTIES_PANEL *aPanel)
#define _(s)
Base window classes and related definitions.
@ FRAME_PCB_EDITOR
Definition: frame_type.h:42
@ FRAME_SCH_SYMBOL_EDITOR
Definition: frame_type.h:35
@ FRAME_SCH
Definition: frame_type.h:34
@ FRAME_FOOTPRINT_EDITOR
Definition: frame_type.h:43
#define APIIMPORT
Definition: import_export.h:43
KICOMMON_API wxFont GetDockedPaneFont(wxWindow *aWindow)
Definition: ui_common.cpp:144
PGM_BASE & Pgm()
The global program "get" accessor.
Definition: pgm_base.cpp:1071
see class PGM_BASE
APIIMPORT wxPGGlobalVarsClass * wxPGGlobalVars
#define TYPE_HASH(x)
Definition: property.h:71
std::vector< PROPERTY_BASE * > PROPERTY_LIST
Definition: property_mgr.h:49
std::map< PROPERTY_BASE *, int > PROPERTY_DISPLAY_ORDER
Definition: property_mgr.h:58
size_t TYPE_ID
Unique type identifier.
Definition: property_mgr.h:47