62 if( !aProperty.CmpNoCase( property->
Name() ) )
81 return it->second.m_allProperties;
96 return it->second.m_displayOrder;
105 static const std::vector<wxString>
empty;
111 return it->second.m_groupDisplayOrder;
117 if( aBase == aTarget )
120 auto classDesc =
m_classes.find( aBase );
125 auto& converters = classDesc->second.m_typeCasts;
126 auto converter = converters.find( aTarget );
128 if( converter == converters.end() )
129 return IsOfType( aBase, aTarget ) ? aSource :
nullptr;
131 return (*converter->second)( aSource );
137 const wxString&
name = aProperty->
Name();
145 if( !classDesc.
m_groups.count( aGroup ) )
148 classDesc.
m_groups.insert( aGroup );
160 classDesc.
m_replaced.insert( std::make_pair( aBase, aName ) );
170 wxASSERT_MSG( typeCasts.count( derivedHash ) == 0,
"Such converter already exists" );
171 typeCasts.emplace( derivedHash, aCast );
177 wxASSERT_MSG( aDerived != aBase,
"Class cannot inherit from itself" );
181 derived.
m_bases.push_back( base );
185 "You need to add a TYPE_CAST for classes inheriting from multiple bases" );
191 wxASSERT_MSG( aDerived != aBase,
"Class cannot mask from itself" );
200 const wxString& aName,
203 wxASSERT_MSG( aDerived != aBase,
"Class cannot override from itself" );
212 const wxString& aName,
215 wxASSERT_MSG( aDerived != aBase,
"Class cannot override from itself" );
235 return it->second( aItem );
253 return it->second( aItem );
261 if( aDerived == aBase )
264 auto derived =
m_classes.find( aDerived );
265 wxCHECK( derived !=
m_classes.end(),
false );
268 for(
auto& base : derived->second.m_bases )
270 if(
IsOfType( base.get().m_id, aBase ) )
280 for( std::pair<const TYPE_ID, CLASS_DESC>& classEntry :
m_classes )
281 classEntry.second.rebuild();
308 std::vector<wxString> displayOrder;
309 std::set<wxString> groups;
312 [&]( std::set<wxString>& aSet, std::vector<wxString>& aResult )
314 auto collectGroupsRecursive =
315 [](
auto& aSelf, std::set<wxString>& aSetR, std::vector<wxString>& aResultR,
318 for(
const wxString&
group : aClassR.m_groupDisplayOrder )
320 if( !aSetR.count(
group ) )
322 aSetR.insert(
group );
323 aResultR.emplace_back(
group );
328 aSelf( aSelf, aSetR, aResultR, base );
331 collectGroupsRecursive( collectGroupsRecursive, aSet, aResult, *
this );
336 collectGroups( groups, displayOrder );
346 for(
const std::pair<size_t, wxString>& replacedEntry : m_replaced )
347 aReplaced.emplace( replacedEntry );
349 for(
const std::pair<size_t, wxString>& maskedEntry : m_maskedBaseProperties )
350 aMasked.emplace( maskedEntry );
356 int displayOrderStart = 0;
358 if( !aDisplayOrder.empty() )
360 int firstSoFar = std::min_element( aDisplayOrder.begin(), aDisplayOrder.end(),
361 [](
const std::pair<PROPERTY_BASE*, int>& aFirst,
362 const std::pair<PROPERTY_BASE*, int>& aSecond )
364 return aFirst.second < aSecond.second;
367 displayOrderStart = firstSoFar - m_ownProperties.size();
374 PROPERTY_SET::key_type propertyKey = std::make_pair( property->OwnerHash(),
377 if( aReplaced.count( propertyKey ) )
381 if( aMasked.count( propertyKey ) )
384 aDisplayOrder[property] = displayOrderStart + idx++;
385 aResult.push_back( property );
389 for(
auto it = m_bases.rbegin(); it != m_bases.rend(); ++it )
390 it->get().collectPropsRecur( aResult, aReplaced, aDisplayOrder, aMasked );
396 std::vector<TYPE_ID> ids;
418 for( std::pair<const TYPE_ID, CLASS_DESC>& classEntry :
m_classes )
422 info.type = classEntry.first;
425 for(
PROPERTY_BASE* prop : classEntry.second.m_allProperties )
426 info.properties.push_back( prop );
428 rv.push_back(
info );
451 callListeners( objectClass.
m_id );
454 callListeners( superClass.
m_id );
461 return,
"Can't have more than one managed commit at a time!" );
470 "Something went wrong: m_managedCommit already null!" );
Represent a set of changes (additions, deletions or modifications) of a data model (e....
Class that other classes need to inherit from, in order to be inspectable.
PROPERTY_BASE & SetGroup(const wxString &aGroup)
virtual size_t BaseHash() const =0
Return type-id of the Base class.
virtual bool Writeable(INSPECTABLE *aObject) const
bool Available(INSPECTABLE *aObject) const
Return true if aObject offers this PROPERTY.
const wxString & Name() const
virtual size_t OwnerHash() const =0
Return type-id of the Owner class.
~PROPERTY_COMMIT_HANDLER()
PROPERTY_COMMIT_HANDLER(COMMIT *aCommit)
Provide class metadata.Helper macro to map type hashes to names.
bool m_dirty
Flag indicating that the list of properties needs to be rebuild (RebuildProperties())
CLASSES_INFO GetAllClasses()
std::vector< CLASS_INFO > CLASSES_INFO
const wxString & ResolveType(TYPE_ID aType) const
Return name of a type.
const PROPERTY_LIST & GetProperties(TYPE_ID aType) const
Return all properties for a specific type.
void InheritsAfter(TYPE_ID aDerived, TYPE_ID aBase)
Declare an inheritance relationship between types.
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 ...
void PropertyChanged(INSPECTABLE *aObject, PROPERTY_BASE *aProperty)
Callback to alert the notification system that a property has changed.
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()
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
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 ...
std::vector< TYPE_ID > GetMatchingClasses(PROPERTY_BASE *aProperty)
std::map< TYPE_ID, std::vector< PROPERTY_LISTENER > > m_listeners
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.
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.
virtual size_t BaseHash() const =0
virtual size_t DerivedHash() const =0
static bool empty(const wxTextEntryBase *aCtrl)
static wxString EMPTY_STRING(wxEmptyString)
std::function< void(INSPECTABLE *, PROPERTY_BASE *, COMMIT *)> PROPERTY_LISTENER
std::vector< PROPERTY_BASE * > PROPERTY_LIST
std::map< PROPERTY_BASE *, int > PROPERTY_DISPLAY_ORDER
std::set< std::pair< size_t, wxString > > PROPERTY_SET
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)
PROPERTY_SET m_replaced
Recreates the list of properties.
PROPERTY_SET m_maskedBaseProperties
Overrides for base class property availabilities.
std::vector< wxString > m_groupDisplayOrder
Non-owning list of classes's direct properties in display order.
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::vector< PROPERTY_BASE * > m_allProperties
Compiled display order for all properties.
PROPERTY_DISPLAY_ORDER m_displayOrder
List of property groups provided by this class in display order.
PROPERTY_FUNCTOR_MAP m_writeabilityOverrides
All properties (both unique to the type and inherited)
std::map< TYPE_ID, std::unique_ptr< TYPE_CAST_BASE > > m_typeCasts
Properties from bases that should be masked (hidden) on this subclass.
void rebuild()
Traverses the class inheritance hierarchy bottom-to-top, gathering all properties available to a type...
void collectPropsRecur(PROPERTY_LIST &aResult, PROPERTY_SET &aReplaced, PROPERTY_DISPLAY_ORDER &aDisplayOrder, PROPERTY_SET &aMasked) const
PROPERTY_FUNCTOR_MAP m_availabilityOverrides
Overrides for base class property writeable status.
const TYPE_ID m_id
Types after which this type inherits.
std::set< wxString > m_groups
Replaced properties (TYPE_ID / name)