AUI Framework
master
Cross-platform base for C++ UI apps
|
Basic easy-to-use property implementation containing T. More...
#include <AUI/Common/AProperty.h>
Public Types# | |
using | Underlying = T |
Public Member Functions# | |
template<aui::convertible_to< T > U> | |
AProperty (U &&value) noexcept(noexcept(T(std::forward< U >(value)))) | |
AObjectBase * | boundObject () |
const AObjectBase * | boundObject () const |
AProperty (const AProperty &value) | |
AProperty (AProperty &&value) noexcept | |
AProperty & | operator= (const AProperty &value) |
AProperty & | operator= (AProperty &&value) noexcept |
template<aui::convertible_to< T > U> | |
AProperty & | operator= (U &&value) noexcept |
template<ASignalInvokable SignalInvokable> | |
void | operator^ (SignalInvokable &&t) |
void | notify () |
Notify observers that a change was occurred (no preconditions). | |
const T & | value () const noexcept |
operator const T & () const noexcept | |
const T * | operator-> () const noexcept |
const T & | operator* () const noexcept |
aui::PropertyModifier< AProperty > | writeScope () noexcept |
template<aui::invocable< const T & > Projection> | |
auto | readProjected (Projection &&projection) const noexcept |
Makes a readonly projection of this property. | |
template<aui::invocable< const T & > ProjectionRead, aui::invocable< const std::invoke_result_t< ProjectionRead, T > & > ProjectionWrite> | |
auto | biProjected (ProjectionRead &&projectionRead, ProjectionWrite &&projectionWrite) noexcept |
Makes a bidirectional projection of this property. | |
template<aui::detail::property::ProjectionBidirectional< T > Projection> | |
auto | biProjected (Projection &&projectionBidirectional) noexcept |
Makes a bidirectional projection of this property (by a single aui::lambda_overloaded). | |
![]() | |
AObjectBase (AObjectBase &&rhs) noexcept | |
AObjectBase (const AObjectBase &rhs) noexcept | |
AObjectBase & | operator= (const AObjectBase &rhs) noexcept |
AObjectBase & | operator= (AObjectBase &&rhs) noexcept |
Signals and public fields# | |
T | raw {} |
emits< T > | changed |
Additional Inherited Members# | |
![]() | |
static ASpinlockMutex | SIGNAL_SLOT_GLOBAL_SYNC |
![]() | |
void | clearAllIngoingConnections () noexcept |
virtual void | handleSlotException (std::exception_ptr exception) |
Called then an exception has thrown during slot processing of the signal emitted by this object. | |
AProperty<T>
is a container holding an instance of T
. You can assign a value to it with operator=
and read value with value()
method or implicit conversion operator T()
.See property system for usage examples.
To declare a property inside your data model, use AProperty template:
AProperty<T>
is a container holding an instance of T
. You can assign a value to it with operator=
and read value with value()
method or implicit conversion operator T()
.
AProperty behaves like a class/struct data member:
Non-const operators have side effects; const operators don't, so you can perform seamlessly:
In most cases, property is implicitly convertible to its underlying type (const only):
If it doesn't, simply put an asterisk:
All property types offer .changed
field which is a signal reporting value changes. Let's make little observer object for demonstration:
Example usage:
At the moment, the program prints nothing. When we change the property:
Code produces the following output:
As you can see, observer received the update. But, for example, if we would like to display the value via label, the label wouldn't display the current value until the next update. We want the label to display current value without requiring an update. To do this, connect to the property directly, without explicitly asking for changed
:
Code above produces the following output:
As you can see, observer receives the value without making updates to the value. The call of LogObserver::log
is made by AObject::connect
itself. In this document, we will call this behaviour as "pre-fire".
Subsequent changes to field would send updates as well:
Assignment operation above makes an additional line to output:
Whole program output when connecting to property directly:
Copying AProperty
is considered as a valid operation as it's a data holder. However, it's worth to note that AProperty
copies it's underlying data field only, the signal-slot relations are not borrowed.
This part is similar to previous examples, nothing new. Let's introduce a copy:
Now, let's change origin->name
and check that observer received an update, but value in the copy
remains:
In this example, observer is aware of changes "Chloe"
-> "Marinette"
. The copy is not aware because it is a copy. If we try to change the copy
's name:
The observer is not aware about changes in copy
. In fact. copy->name
has zero connections.
The situation with copy assigning auto copy = _new<User>(); *copy = *original;
is similar to copy construction auto copy = _new<User>(*original);
, except that we are copying to some pre-existing data structure that potentially have signal-slot relations already. So, not only connections should be kept as is but a notification for copy destination's observers is needed.
As with copy construction, copy operation of AProperty
does not affect signal-slot relations. Moreover, it notifies the observers.
This part is similar to previous examples, nothing new. Let's perform copy-assignment:
See, not only the connection remains, but it also receives notification about the change.
Similary to copy, AProperty is both move assignable and constructible except that underlying value is moved instead of copying. Also, the observers of the source object receive notification that the value was emptied. The signal-slot relations are left unchanged.
This part is similar to previous examples, nothing new. Let's introduce a move:
Now, let's change origin->name
and check that observer received an update, but value in the moved
remains:
In this example, observer is aware of changes "Chloe"
-> ""
-> "Marinette"
. The moved
is not aware. If we try to change the moved
's name:
The observer is not aware about changes in moved
. In fact. moved->name
has zero connections. Move assignment work in a similar way to copy assignment:
This part is similar to previous examples, nothing new. Let's perform move-assignment:
See, not only the connection remains, but it also receives notification about the change.
Refer to aui::PropertyModifier.
|
inline |
In common, you won't need to use this function. AProperty is reevaluated automatically as soon as one updates the value within property.
If your scenario goes beyond writeScope that explicitly defines modification scope within RAII scope, you can modify the underlying value by accessing AProperty::raw
and then call notify to notify the observers that value is changed.
|
inlinenoexcept |