KiCad PCB EDA Suite
Loading...
Searching...
No Matches
property_mgr.h
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 *
7 * @author Tomasz Wlostowski <[email protected]>
8 * @author Maciej Suminski <[email protected]>
9 *
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License
12 * as published by the Free Software Foundation; either version 3
13 * of the License, or (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program. If not, see <https://www.gnu.org/licenses/>.
22 */
23
24#ifndef PROPERTY_MGR_H
25#define PROPERTY_MGR_H
26
27#include <core/wx_stl_compat.h> // Needed for stl hash extensions
28
29#include <wx/string.h>
30
31#include <algorithm>
32#include <functional>
33#include <list>
34#include <map>
35#include <unordered_map>
36#include <set>
37#include <vector>
38#include <memory>
39#include <eda_units.h>
40
41class PROPERTY_BASE;
42class TYPE_CAST_BASE;
44class INSPECTABLE;
45class COMMIT;
46
48using TYPE_ID = size_t;
49
50using PROPERTY_LISTENER = std::function<void( INSPECTABLE*, PROPERTY_BASE*, COMMIT* )>;
51
54using PROPERTY_LISTENER_HANDLE = std::size_t;
55
57{
58public:
60
62};
63
79{
80public:
82 {
83 static PROPERTY_MANAGER pm;
84 return pm;
85 }
86
95 void RegisterType( TYPE_ID aType, const wxString& aName );
96
104 PROPERTY_BASE* GetProperty( TYPE_ID aType, const wxString& aProperty ) const;
105
112 const std::vector<PROPERTY_BASE*>& GetProperties( TYPE_ID aType ) const;
113
114 const std::map<PROPERTY_BASE*, int>& GetDisplayOrder( TYPE_ID aType ) const;
115
116 const std::vector<wxString>& GetGroupDisplayOrder( TYPE_ID aType ) const;
117
129 const void* TypeCast( const void* aSource, TYPE_ID aBase, TYPE_ID aTarget ) const;
130
131 void* TypeCast( void* aSource, TYPE_ID aBase, TYPE_ID aTarget ) const
132 {
133 return const_cast<void*>( TypeCast( (const void*) aSource, aBase, aTarget ) );
134 }
135
146 PROPERTY_BASE& AddProperty( PROPERTY_BASE* aProperty, const wxString& aGroup = wxEmptyString );
147
159 PROPERTY_BASE& ReplaceProperty( size_t aBase, const wxString& aName, PROPERTY_BASE* aNew,
160 const wxString& aGroup = wxEmptyString );
161
167 void AddTypeCast( TYPE_CAST_BASE* aCast );
168
175 void InheritsAfter( TYPE_ID aDerived, TYPE_ID aBase );
176
185 void Mask( TYPE_ID aDerived, TYPE_ID aBase, const wxString& aName );
186
195 void OverrideAvailability( TYPE_ID aDerived, TYPE_ID aBase, const wxString& aName,
196 std::function<bool( INSPECTABLE* )> aFunc );
197
206 void OverrideWriteability( TYPE_ID aDerived, TYPE_ID aBase, const wxString& aName,
207 std::function<bool( INSPECTABLE* )> aFunc );
208
215 bool IsAvailableFor( TYPE_ID aItemClass, PROPERTY_BASE* aProp, INSPECTABLE* aItem );
216
223 bool IsWriteableFor( TYPE_ID aItemClass, PROPERTY_BASE* aProp, INSPECTABLE* aItem );
224
228 bool IsOfType( TYPE_ID aDerived, TYPE_ID aBase ) const;
229
234 void Rebuild();
235
237 {
238 wxString name;
240 std::vector<PROPERTY_BASE*> properties;
241 };
242
243 typedef std::vector<CLASS_INFO> CLASSES_INFO;
244
246
252 void PropertyChanged( INSPECTABLE* aObject, PROPERTY_BASE* aProperty );
253
260 [[nodiscard]] class PROPERTY_LISTENER_SUBSCRIPTION
261 RegisterListener( TYPE_ID aType, PROPERTY_LISTENER aListenerFunc );
262
273
274private:
276 m_dirty( false ),
277 m_managedCommit( nullptr )
278 {
279 }
280
281 // Explicit noexcept on this user-declared destructor short-circuits the
282 // implicit noexcept computation the compiler would otherwise perform by
283 // walking every member's destructor. That walk reaches the
284 // `std::map<wxString, std::unique_ptr<PROPERTY_BASE>>` member inside
285 // CLASS_DESC and, since this header only forward-declares PROPERTY_BASE
286 // / TYPE_CAST_BASE, fails with an "incomplete type" diagnostic at every
287 // TU that includes property_mgr.h. The destructor body in
288 // property_mgr.cpp sees the complete types via the property.h include
289 // there.
291
293
296 {
297 // Constructor body lives in property_mgr.cpp so it doesn't force
298 // every TU including this header to instantiate the destructors of
299 // `std::map<X, std::unique_ptr<PROPERTY_BASE>>` and
300 // `std::map<X, std::unique_ptr<TYPE_CAST_BASE>>` (which require
301 // complete PROPERTY_BASE / TYPE_CAST_BASE — only forward-declared
302 // here). Same applies to the destructor and the defaulted move
303 // operations.
304 CLASS_DESC( TYPE_ID aId );
305 ~CLASS_DESC() noexcept;
306 CLASS_DESC( CLASS_DESC&& ) noexcept;
307
310
312 std::vector<std::reference_wrapper<CLASS_DESC>> m_bases;
313
316
319
321 std::set<std::pair<size_t, wxString>> m_maskedBaseProperties;
322
324 std::map<std::pair<size_t, wxString>, std::function<bool( INSPECTABLE* )>> m_availabilityOverrides;
325
327 std::map<std::pair<size_t, wxString>, std::function<bool( INSPECTABLE* )>> m_writeabilityOverrides;
328
331
334
336 std::vector<wxString> m_groupDisplayOrder;
337
340
342 std::set<wxString> m_groups;
343
345 std::set<std::pair<size_t, wxString>> m_replaced;
346
348 void rebuild();
349
352 void collectPropsRecur( std::vector<PROPERTY_BASE*>& aResult,
353 std::set<std::pair<size_t, wxString>>& aReplaced,
354 std::map<PROPERTY_BASE*, int>& aDisplayOrder,
355 std::set<std::pair<size_t, wxString>>& aMasked ) const;
356 };
357
359 CLASS_DESC& getClass( TYPE_ID aTypeId );
360
361 std::unordered_map<TYPE_ID, wxString> m_classNames;
362
364 std::unordered_map<TYPE_ID, CLASS_DESC> m_classes;
365
368
369 std::map<TYPE_ID,
371
373
375};
376
377
379#define REGISTER_TYPE(x) PROPERTY_MANAGER::Instance().RegisterType(TYPE_HASH(x), TYPE_NAME(x))
380
381
388{
389public:
391
393 m_type( aType ),
394 m_handle( aHandle )
395 {
396 }
397
398 // Destructor + reset() are out-of-line so the chain reset() ->
399 // PROPERTY_MANAGER::UnregisterListener doesn't force consumers of this
400 // header to fully instantiate PROPERTY_MANAGER (which would require
401 // complete PROPERTY_BASE / TYPE_CAST_BASE for the unique_ptr members
402 // inside CLASS_DESC). Holders of this subscription as a class member
403 // (e.g. SCHEMATIC) therefore only see the declarations here, not the
404 // implicit PROPERTY_MANAGER destructor instantiation.
406
409
411 m_type( aOther.m_type ),
412 m_handle( aOther.m_handle )
413 {
414 aOther.m_handle = 0;
415 }
416
418
419 void reset();
420
421private:
424};
425
426
427#endif /* PROPERTY_MGR_H */
Represent a set of changes (additions, deletions or modifications) of a data model (e....
Definition commit.h:68
Class that other classes need to inherit from, in order to be inspectable.
Definition inspectable.h:38
A class to perform either relative or absolute display origin transforms for a single axis of a point...
PROPERTY_COMMIT_HANDLER(COMMIT *aCommit)
Move-only RAII wrapper around a single listener registration.
PROPERTY_LISTENER_HANDLE m_handle
PROPERTY_LISTENER_SUBSCRIPTION(PROPERTY_LISTENER_SUBSCRIPTION &&aOther) noexcept
PROPERTY_LISTENER_SUBSCRIPTION(const PROPERTY_LISTENER_SUBSCRIPTION &)=delete
PROPERTY_LISTENER_SUBSCRIPTION(TYPE_ID aType, PROPERTY_LISTENER_HANDLE aHandle)
PROPERTY_LISTENER_SUBSCRIPTION & operator=(const PROPERTY_LISTENER_SUBSCRIPTION &)=delete
const std::vector< PROPERTY_BASE * > & GetProperties(TYPE_ID aType) const
Return all properties for a specific type.
bool m_dirty
Flag indicating that the list of properties needs to be rebuild (RebuildProperties())
CLASSES_INFO GetAllClasses()
std::vector< CLASS_INFO > CLASSES_INFO
void InheritsAfter(TYPE_ID aDerived, TYPE_ID aBase)
Declare an inheritance relationship between types.
bool IsWriteableFor(TYPE_ID aItemClass, PROPERTY_BASE *aProp, INSPECTABLE *aItem)
Checks overriden availability and original availability of a property, returns false if the property ...
void PropertyChanged(INSPECTABLE *aObject, PROPERTY_BASE *aProperty)
Callback to alert the notification system that a property has changed.
COMMIT * m_managedCommit
void Mask(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName)
Sets a base class property as masked in a derived class.
static PROPERTY_MANAGER & Instance()
friend class PROPERTY_COMMIT_HANDLER
Structure holding type meta-data.
PROPERTY_BASE & AddProperty(PROPERTY_BASE *aProperty, const wxString &aGroup=wxEmptyString)
Register a property.
void Rebuild()
Rebuild the list of all registered properties.
PROPERTY_BASE * GetProperty(TYPE_ID aType, const wxString &aProperty) const
Return a property for a specific type.
std::unordered_map< TYPE_ID, CLASS_DESC > m_classes
~PROPERTY_MANAGER() noexcept
const void * TypeCast(const void *aSource, TYPE_ID aBase, TYPE_ID aTarget) const
Cast a type to another type.
CLASS_DESC & getClass(TYPE_ID aTypeId)
void RegisterType(TYPE_ID aType, const wxString &aName)
Associate a name with a 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 ...
void * TypeCast(void *aSource, TYPE_ID aBase, TYPE_ID aTarget) const
class PROPERTY_LISTENER_SUBSCRIPTION RegisterListener(TYPE_ID aType, PROPERTY_LISTENER aListenerFunc)
Register a listener for the given type and return a move-only subscription that auto-unregisters in i...
const std::map< PROPERTY_BASE *, int > & GetDisplayOrder(TYPE_ID aType) const
PROPERTY_LISTENER_HANDLE m_nextListenerHandle
void UnregisterListener(TYPE_ID aType, PROPERTY_LISTENER_HANDLE aHandle)
Remove a single listener registration by its handle.
void OverrideAvailability(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName, std::function< bool(INSPECTABLE *)> aFunc)
Sets an override availability functor for a base class property of a given derived class.
bool IsOfType(TYPE_ID aDerived, TYPE_ID aBase) const
Return true if aDerived is inherited from aBase.
PROPERTY_BASE & ReplaceProperty(size_t aBase, const wxString &aName, PROPERTY_BASE *aNew, const wxString &aGroup=wxEmptyString)
Replace an existing property for a specific type.
std::map< TYPE_ID, std::vector< std::pair< PROPERTY_LISTENER_HANDLE, PROPERTY_LISTENER > > > m_listeners
void OverrideWriteability(TYPE_ID aDerived, TYPE_ID aBase, const wxString &aName, std::function< bool(INSPECTABLE *)> aFunc)
Sets an override writeability functor for a base class property of a given derived class.
std::unordered_map< TYPE_ID, wxString > m_classNames
Map of all available types.
const std::vector< wxString > & GetGroupDisplayOrder(TYPE_ID aType) const
void AddTypeCast(TYPE_CAST_BASE *aCast)
Register a type converter.
STL class.
STL namespace.
std::function< void(INSPECTABLE *, PROPERTY_BASE *, COMMIT *)> PROPERTY_LISTENER
Opaque handle returned by RegisterListener so a specific registration can be removed without disturbi...
std::size_t PROPERTY_LISTENER_HANDLE
size_t TYPE_ID
Unique type identifier.
Returns metadata for a specific type.
std::vector< std::reference_wrapper< CLASS_DESC > > m_bases
Properties unique to this type (i.e. not inherited)
std::set< std::pair< size_t, wxString > > m_replaced
Recreates the list of properties.
std::vector< wxString > m_groupDisplayOrder
Non-owning list of classes's direct properties in display order.
std::map< std::pair< size_t, wxString >, std::function< bool(INSPECTABLE *)> > m_availabilityOverrides
Overrides for base class property writeable status.
std::map< std::pair< size_t, wxString >, std::function< bool(INSPECTABLE *)> > m_writeabilityOverrides
All properties (both unique to the type and inherited)
std::map< wxString, std::unique_ptr< PROPERTY_BASE > > m_ownProperties
Type converters available for this type.
std::vector< PROPERTY_BASE * > m_ownDisplayOrder
The property groups provided by this class.
std::set< std::pair< size_t, wxString > > m_maskedBaseProperties
Overrides for base class property availabilities.
std::map< PROPERTY_BASE *, int > m_displayOrder
List of property groups provided by this class in display order.
std::vector< PROPERTY_BASE * > m_allProperties
Compiled display order for all properties.
std::map< TYPE_ID, std::unique_ptr< TYPE_CAST_BASE > > m_typeCasts
Properties from bases that should be masked (hidden) on this subclass.
void collectPropsRecur(std::vector< PROPERTY_BASE * > &aResult, std::set< std::pair< size_t, wxString > > &aReplaced, std::map< PROPERTY_BASE *, int > &aDisplayOrder, std::set< std::pair< size_t, wxString > > &aMasked) const
void rebuild()
Traverses the class inheritance hierarchy bottom-to-top, gathering all properties available to a type...
const TYPE_ID m_id
Types after which this type inherits.
std::set< wxString > m_groups
Replaced properties (TYPE_ID / name)
std::vector< PROPERTY_BASE * > properties