AUI Framework
master
Cross-platform base for C++ UI apps
|
Readonly property that holds a value computed by an expression. More...
#include <AUI/Common/APropertyPrecomputed.h>
Public Types# | |
using | Underlying = T |
using | Factory = std::function<T()> |
Public Member Functions# | |
template<aui::factory< T > Factory> | |
APropertyPrecomputed (Factory &&expression) | |
APropertyPrecomputed (const APropertyPrecomputed &)=delete | |
APropertyPrecomputed (APropertyPrecomputed &&) noexcept=delete | |
template<ASignalInvokable SignalInvokable> | |
void | operator^ (SignalInvokable &&t) |
void | invalidate () override |
Marks this precomputed property to be reevaluated. | |
AObjectBase * | boundObject () |
const T & | value () const |
operator const T & () const | |
const T & | operator* () const |
template<aui::invocable< const Underlying & > Projection> | |
auto | readProjected (Projection &&projection) noexcept |
Makes a readonly projection of this property. | |
const T * | operator-> () const noexcept |
![]() | |
AObjectBase (AObjectBase &&rhs) noexcept | |
AObjectBase (const AObjectBase &rhs) noexcept | |
AObjectBase & | operator= (const AObjectBase &rhs) noexcept |
AObjectBase & | operator= (AObjectBase &&rhs) noexcept |
Signals and public fields# | |
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. | |
APropertyPrecomputed<T>
is a readonly property similar to AProperty<T>
. It holds an instance of T
as well. Its value is determined by the C++ function specified in its constructor, typically a C++ lambda expression.See property system for usage info.
Despite properties offer projection methods, you might want to track and process values of several properties.
APropertyPrecomputed<T>
is a readonly property similar to AProperty<T>
. It holds an instance of T
as well. Its value is determined by a reactive expression specified in APropertyPrecomputed<T>
's constructor, typically a C++ lambda.
It's convenient to access values from another properties inside the expression. The properties accessed during invocation of the expression are tracked behind the scenes so they become dependencies of APropertyPrecomputed
automatically. If one of the tracked properties fires changed
signal, APropertyPrecomputed
invalidates its T
. APropertyPrecomputed
follows lazy semantics so the expression is re-evaluated and the new result is applied to APropertyPrecomputed
as soon as the latter is accessed for the next time.
In other words, it allows to specify relationships between different object properties and reactively update APropertyPrecomputed
value whenever its dependencies change. APropertyPrecomputed<T>
is somewhat similar to Qt Bindable Properties.
APropertyPrecomputed
is a readonly property, hence you can't update its value with assignment. You can get its value with value()
method or implicit conversion operator T()
as with other properties.
Declare a property with custom expression determining it's value as follows:
Let's make little observer object for demonstration:
Usage:
The example above prints "Emma Watson". If we try to update one of dependencies of APropertyPrecomputed
(i.e., name
or surname
), APropertyPrecomputed
responds immediately:
The example above prints "Emma Stone".
Any C++ callable evaluating to T
can be used as an expression for APropertyPrecomputed<T>
. However, to formulate correct expression, some rules must be satisfied.
Dependency tracking only works on other properties. It is the developer's responsibility to ensure all values referenced in the expression are properties, or, at least, non-property values that wouldn't change or whose changes are not interesting. You definitely can use branching inside the expression, but you must be confident about what are you doing. Generally speaking, use as trivial expressions as possible.
In this expression, we have a fast path return if name
is empty.
As soon as we set name
to ""
, we don't access surname
. If we try to trigger the fast path return:
surname
can't trigger re-evaluation anyhow. Re-evaluation can be triggered by name
only. So, at the moment, we are interested in name
changes only.
APropertyPrecomputed
might evaluate its expression several times during its lifetime. The developer must make sure that all objects referenced in the expression live longer than APropertyPrecomputed
.
The expression should not read from the property it's a binding for, including other referenced APropertyPrecomputes. Otherwise, there's an infinite evaluation loop, and AEvaluationLoopException is thrown.
If copy construction of APropertyPrecomputed
were possible, consider the following code:
copy
has copied factory function of user
, which refers to fields of user
, not to copy
's fields. Copy construction of a class or struct discards default values of all fields - this is the way APropertyPrecomputed
's factory function is set to APropertyPrecomputed.
|
inlineoverridevirtual |
In common, you won't need to use this function. APropertyPrecomputed is reevaluated automatically as soon as one updates a property expression depends on.
Implements aui::react::DependencyObserver.