AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
Property System

Property System is a data binding mechanism based on signal-slot system. More...

Detailed Description#

Warning
This API is experimental. Experimental APIs are likely to contain bugs, might be changed or removed in the future.
AUI property system, a compiler-agnostic alternative to __property or [property]. Based on signal-slot system for platform-independent C++ development. Unlike Qt, AUI's properties don't involve external tools (like moc). They are written in pure C++.
Note
This page is about presenting individual values. For lists, see AForEachUI.

AUI property system is relatively complex, as it involves a lot of features in a single place:

  1. thread safe
  2. many-to-many relationships between objects
  3. optional data modification when passing values between objects (like STL projections)
  4. emitter can be either signal or property
  5. slot can be either lambda, method or property
  6. for the latter case, system must set up backward connection as well (including projection support)
  7. again, for the latter case, there's an option to make property-to-slot connection, where the "slot" is property's assignment operation
  8. 2 syntax variants: procedural (straightforward) and declarative
  9. three property variants: simple field (AProperty), custom getter/setter (APropertyDef) and custom evaluation (APropertyPrecomputed)
  10. some properties can be readonly
  11. propagating strong types' traits on views

Learning curve is relatively flat, so be sure to ask questions and open issues on our GitHub page.

Main difference between basic value lying somewhere inside your class and a property is that the latter explicitly ties getter, setter and a signal reporting value changes. Property acts almost transparently, as if there's no extra wrapper around your data. This allows to work with properties in the same way as with their underlying values. You can read the intermediate value of a property and subscribe to its changes via a single connect call. Also, when connecting property to property, it is possible to make them observe changes of each other bia biConnect call:

struct User {
    AProperty<AString> name;
};
auto user = aui::ptr::manage(new User { .name = "Robert" });
auto tf = _new<ATextField>();
AObject::biConnect(user->name, tf->text());
auto window = _new<AWindow>();
window->setContents(Centered { tf });
window->show();
static void biConnect(PropertySource &&propertySource, PropertyDestination &&propertyDestination)
Connects source property to the destination property and opposite (bidirectionally).
Definition AObject.h:156
Basic easy-to-use property implementation containing T.
Definition AProperty.h:30
static _< T > manage(T *raw)
Delegates memory management of the raw pointer T* raw to the shared pointer, which is returned.
Definition SharedPtrTypes.h:424

Or simpler:

// ...
window->setContents(Centered {
  tf && user->name,
});
// ...

The code above generates a window with a text field:

A single call of biConnect:

  • Prefills text field with the current user->name value (pre fire):
    EXPECT_EQ(tf->text(), "Robert");
  • Connects user->named.changed to tf to notify the text field about changes of user->name:
    user->name = "Angela";           // changing user->name programmatically...
    EXPECT_EQ(tf->text(), "Angela"); // ...should reflect on the text field

  • Connects tf->text().changed to notify the user->name property about changes in text field (i.e., if the user typed another value to the text field):
// user typed "Snezhana", now let's check the value in user->name:
EXPECT_EQ(user->name, "Snezhana");

This is basic example of setting up property-to-property connection.

Declaring Properties#

There are several ways to define a property in AUI:

  • AProperty - basic wrapper property type for data models
  • APropertyDef - property-compliant view type to tie custom getter, setter and signal together
  • APropertyPrecomputed - readonly property whose value is determined by a callable that references other properties

Please check their respective documentation pages for an additional information.

UI data binding with let#

Note
This is a comprehensive, straightforward way of setting up a connection. We are demonstrating it here so you can get deeper understanding on how connections are made and what does declarative way do under the hood. This way may be used in favour of declarative way if the latter not work for you. For declarative way, go to UI declarative data binding.

This approach allows more control over the binding process by using AObject::connect/AObject::biConnect which is a procedural way of setting up connections. As a downside, it requires "let" syntax clause which may seem as overkill for such a simple operation.

Label via let#

Use let expression to connect the model's username property to the label's text() property.

using namespace declarative;
struct User {
    AProperty<AString> name;
};
auto user = aui::ptr::manage(new User { .name = "Roza" });
class MyWindow: public AWindow {
public:
    MyWindow(const _<User>& user) {
        setContents(Centered {
            _new<ALabel>() let {
              // Data goes from left to right:
              // current value (pre fire) or changed event
              // goes to assignment operation of it->text()
              AObject::connect(user->name, it->text());
              //                ->  ->  ->  ->  ->
              // in other words, this connection is essentially the
              // same as
              // AObject::connect(user->name, slot(it)::setText);
              //
              // if you want user->name to be aware or it->text()
              // changes (i.e., if it were an editable view
              // like ATextField) use AObject::biConnect instead
              // (see "Bidirectional connection" sample).
            },
        });
    }
};
_new<MyWindow>(user)->show();
void setContents(const _< AViewContainer > &container)
Moves (like via std::move) all children and layout of the specified container to this container.
Represents a window in the underlying windowing system.
Definition AWindow.h:45
void show()
static decltype(auto) connect(const Signal &signal, Object *object, Function &&function)
Connects signal to the slot of the specified object.
Definition AObject.h:86
#define let
Performs multiple operations on a single object without repeating its name (in place) This function c...
Definition kAUI.h:262

This gives the following result:

Note that label already displays the value stored in User.

Let's change the name:

user->name = "Vasil";

By simply performing assignment on user we changed ALabel display text. Magic, huh?

Label via let projection#

It's fairly easy to define a projection because one-sided connection requires exactly one projection, obviously.

_new<ALabel>() let {
    // Data goes from left to right:
    // current value (pre fire) or changed event
    // goes through projection (&AString::uppercase) first
    // then it goes to assignment operation of it->text()
    // property.
    AObject::connect(user->name.readProjected(&AString::uppercase), it->text());
    //                ->  ->  ->  ->  ->  ->  ->  ->  ->  ->  ->  ->
    // in other words, this connection is essentially the same as
    // AObject::connect(user->name.projected(&AString::uppercase), slot(it)::setText);
    // if view's property gets changed (i.e., by user or by occasional
    // ALabel::setText), these changes DO NOT reflect on model
    // as we requested connect() here instead of biConnect().
},

This gives the following result:

Note that the label already displays the projected value stored in User.

Let's change the name:

user->name = "Vasil";

This way, we've set up data binding with projection.

Bidirectional connection#

In previous examples, we've used AObject::connect to make one directional (one sided) connection. This is perfectly enough for ALabel because it cannot be changed by user.

In some cases, you might want to use property-to-property as it's bidirectional. It's used for populating view from model and obtaining data from view back to the model.

For this example, let's use ATextField instead of ALabel as it's an editable view. In this case, we'd want to use AObject::biConnect because we do want user->name to be aware of changes of the view.

_new<ATextField>() let {
  // Data goes from left to right in the first place
  // (i.e., user->name current value overrides it->text())
  // if view's property gets changed (i.e., by user),
  // these changes reflect on model
  // as we requested biConnect here
  //                -> value + changes ->
  AObject::biConnect(user->name, it->text());
  //                <-  changes only   <-
},

This gives the following result:

Let's change the name programmatically:

user->name = "Vasil";

ATextField will respond:

If the user changes the value from UI, these changes will reflect on user->model as well:

EXPECT_EQ(user->name, "Changed from UI");

This way we've set up bidirectional projection via AObject::biConnect which makes user->name aware of UI changes.

Bidirectional projection#

Bidirectional connection updates values in both directions, hence it requires the projection to work in both sides as well.

It is the case for ADropdownList with enums. ADropdownList works with string list model and indices. It does not know anything about underlying values.

For example, define enum with AUI_ENUM_VALUES and model:

enum class Gender {
    MALE,
    FEMALE,
    OTHER,
};
                Gender::MALE,
                Gender::FEMALE,
                Gender::OTHER)
#define AUI_ENUM_VALUES(enum_t,...)
Defines all enum values for AEnumerate.
Definition AEnumerate.h:208
struct User {
    AProperty<Gender> gender;
    // we've omitted other fields for sake of simplicity
};

Now, let's get a mapping for our Gender enum:

static constexpr auto GENDERS = aui::enumerate::ALL_VALUES<Gender>;
constexpr auto ALL_VALUES
constexpr std::array of all possible enum values is the order they've been passed to AUI_ENUM_VALUES.
Definition AEnumerate.h:176

The compile-time constant above is equivalent to:

/* pseudocode */
GENDERS = std::array { Gender::MALE, Gender::FEMALE, GENDER::OTHER };

We just using aui::enumerate::ALL_VALUES because it was provided conveniently by AUI_ENUM_VALUES for us.

It's not hard to guess that we'll use indices of this array to uniquely identify Gender associated with this index:

/* pseudocode */
GENDERS[0]; // -> MALE
GENDERS[1]; // -> FEMALE
GENDERS[2]; // -> OTHER

To perform opposite operation (i.e., Gender to int), we can use aui::indexOf:

/* pseudocode */
aui::indexOf(GENDERS, Gender::MALE);   // -> 0
aui::indexOf(GENDERS, Gender::FEMALE); // -> 1
aui::indexOf(GENDERS, Gender::OTHER);  // -> 2
AOptional< size_t > indexOf(const Container &c, const typename Container::const_reference value) noexcept
Finds the index of the first occurrence of the value.
Definition containers.h:227

To bring these conversions together, let's use overloaded lambda:

static constexpr auto GENDER_INDEX_PROJECTION = aui::lambda_overloaded {
    [](Gender g) -> int { return aui::indexOf(GENDERS, g).valueOr(0); },
    [](int i) -> Gender { return GENDERS[i]; },
};
Definition callables.h:36
Note
It's convenient to use lambda trailing return type syntax (i.e., ... -> int, ... -> Gender) to make it obvious what do transformations do and how one type is transformed to another.

The function-like object above detects the direction of transformation and performs as follows:

GENDER_INDEX_PROJECTION(0); // -> MALE
GENDER_INDEX_PROJECTION(Gender::MALE); // -> 0

It is all what we need to set up bidirectional transformations. Inside AUI_ENTRY:

auto user = aui::ptr::manage(new User { .gender = Gender::MALE });
class MyWindow: public AWindow {
public:
    MyWindow(const _<User>& user) {
        // generate a string list model for genders from GENDERS array defined earlier
        auto gendersStr = AListModel<AString>::fromVector(
            GENDERS
            | ranges::views::transform(AEnumerate<Gender>::toName)
            | ranges::to_vector);
        // equivalent:
        // gendersStr = { "MALE", "FEMALE", "OTHER" }
        // you can customize the displayed strings by playing with
        // ranges::views::transform argument.
        setContents(Centered {
          _new<ADropdownList>(gendersStr) let {
              // AObject::connect(user->gender, it->selectionId());
              //
              // The code above would break, because Gender and int
              // (selectionId() type) are incompatible.
              //
              // Instead, define bidirectional projection:
               AObject::biConnect(
                   user->gender.biProjected(GENDER_INDEX_PROJECTION),
                   it->selectionId());
              },
        });
    }
};
_new<MyWindow>(user)->show();
static const AString & toName(enum_t value)
Map runtime enum value to name. Throws an exception if no such value.
Definition AEnumerate.h:134
static _< AListModel< StoredType > > fromVector(AVector< V > t)
Definition AListModel.h:268
An std::weak_ptr with AUI extensions.
Definition SharedPtrTypes.h:179
  • If we try to change user->gender programmatically, ADropdownList will respond:
    user->gender = Gender::FEMALE;
    EXPECT_EQ(dropdownList->getSelectedId(), 1); // second option
  • If the user changes the value of ADropdownList, it reflects on the model as well:
    EXPECT_EQ(user->gender, Gender::OTHER);

UI declarative data binding#

As said earlier, let syntax is a little bit clunky and requires extra boilerplate code to set up.

Here's where declarative syntax comes into play. The logic behind the syntax is the same as in AObject::connect/AObject::biConnect (for ease of replacement/understanding).

Declarative syntax uses & and && operators to set up connections. These were chosen intentionally: && resembles chain, so we "chaining view and property up".

Also, > operator (resembles arrow) is used to specify the destination slot.

The example below is essentially the same as Label via let but uses declarative connection set up syntax.

Label via declarative#

Use & and > expression to connect the model's username property to the label's text property.

using namespace declarative;
struct User {
    AProperty<AString> name;
};
auto user = aui::ptr::manage(new User { .name = "Roza" });
class MyWindow: public AWindow {
public:
    MyWindow(const _<User>& user) {
        setContents(Centered {
          _new<ALabel>() & user->name > &ALabel::text
        });
    }
};
auto window = _new<MyWindow>(user);
window->show();
auto text() const
Label's text property.
Definition AAbstractLabel.h:37

Note that the label already displays the value stored in User.

Let's change the name:

user->name = "Vasil";

In this example, we've achieved the same intuitive behaviour of data binding of user->name (like in Label via let example) but using declarative syntax. The logic behind & is almost the same as with let and AObject::connect so projection use cases can be adapted in a similar manner.

ADataBindingDefault for omitting view property#

In previous example we have explicitly specified ALabel's property to connect with.

One of notable features of declarative way (in comparison to procedural let way) is that we can omit the view's property to connect with if such ADataBindingDefault specialization exist for the target view and the property type. Some views have already predefined such specialization for their underlying types. For instance, ALabel has such specialization:

/* PREDEFINED! You don't need to define it! This listing is an example */
template<>
public:
    static auto property(const _<ALabel>& view) { return view->text(); }
};
Represents a simple single-line text display view.
Definition ALabel.h:23
Represents a Unicode character string.
Definition AString.h:38
Defines how View handles properties of FieldType type.
Definition ADataBinding.h:37
static auto property(const _< View > &view)
Returns property definition for FieldType.
Definition ADataBinding.h:49

We can use this predefined specialization to omit the destination property:

_new<ALabel>() & user->name

Behaviour of such connection is equal to Label via declarative:

Note that the label already displays the value stored in User.

Let's change the name:

user->name = "Vasil";

In this example, we've omitted the destination property of the connection while maintaining the same behaviour as in Label via declarative.

ADataBindingDefault strong type propagation#

Think of ADataBindingDefault as we're not only connecting properties to properties, but also creating a "property to view" relationship. This philosophy covers the following scenario.

In AUI, there's aui::ranged_number template which stores valid value range right inside the type:

struct User {
};

These strong types can be used to propagate their traits on views, i.e., ANumberPicker. When using declarative syntax, the property system calls ADataBindingDefault::setup to apply some extra traits of the bound value on the view. Here's an abstract on how ANumberPicker defines specialization of ADataBingingDefault with aui::ranged_number:

/* PREDEFINED! You don't need to define it! This listing is an example */
template <aui::arithmetic UnderlyingType, auto min, auto max>
struct ADataBindingDefault<ANumberPicker, aui::ranged_number<UnderlyingType, min, max>> {
public:
    static auto property(const _<ANumberPicker>& view) {
        return view->value();
    }
    static void setup(const _<ANumberPicker>& view) {
        view->setMin(aui::ranged_number<UnderlyingType, min, max>::MIN);
        view->setMax(aui::ranged_number<UnderlyingType, min, max>::MAX);
    }
    // ...
};
A text field for numbers with increase/decrease buttons.
Definition ANumberPicker.h:24
std::add_lvalue_reference_t< T > value() const noexcept
Dereferences the stored pointer.
Definition SharedPtrTypes.h:294
static void setup(const _< View > &view)
Called then view linked with field.
Definition ADataBinding.h:43

As you can see, this specialization pulls the min and max values from aui::ranged_number type and sets them to ANumberPicker. This way ANumberPicker finds out the valid range of values by simply being bound to value that has constraints encoded inside its type.

_new<ANumberPicker>() && user->age,
Note
We're using operator&& here to set up bidirectional connection. For more info, go to Declarative bidirectional connection.

By creating this connection, we've done a little bit more. We've set ANumberPicker::setMin and ANumberPicker::setMax as well:

EXPECT_EQ(numberPicker->getMin(), 1);
EXPECT_EQ(numberPicker->getMax(), 99);

This example demonstrates how to use declarative binding to propagate strong types. aui::ranged_number propagates its constraints on ANumberPicker thanks to ADataBindingDefault specialization.

Label via declarative projection#

We can use projections in the same way as with let.

using namespace declarative;
struct User {
    AProperty<AString> name;
};
auto user = aui::ptr::manage(new User { .name = "Roza" });
class MyWindow: public AWindow {
public:
    MyWindow(const _<User>& user) {
        _<ALabel> label;
        setContents(Centered {
            _new<ALabel>() & user->name.readProjected(&AString::uppercase)
        });
    }
};
auto window = _new<MyWindow>(user);
window->show();

Note that the label already displays the projected value stored in User.

Projection applies to value changes as well. Let's change the name:

user->name = "Vasil";
EXPECT_EQ(user->name, "Vasil");
EXPECT_EQ(label->text(), "VASIL"); // projected

Declarative bidirectional connection#

In previous examples, we've used & to make one directional (one sided) connection. This is perfectly enough for ALabel because it cannot be changed by user.

In some cases, you might want to use property-to-property as it's bidirectional. It's used for populating view from model and obtaining data from view back to the model.

For this example, let's use ATextField instead of ALabel as it's an editable view. In this case, we'd want to use && because we do want user->name to be aware of changes of the view.

_new<ATextField>() && user->name

This gives the following result:

Let's change the name programmatically:

user->name = "Vasil";

ATextField will respond:

If the user changes the value from UI, these changes will reflect on user->model as well:

EXPECT_EQ(user->name, "Changed from UI");

This way we've set up bidirectional projection via && which makes user->name aware of UI changes.

Declarative bidirectional projection#

We can use projections in the same way as with let.

Let's repeat the Bidirectional projection sample in declarative way:

_new<ADropdownList>(gendersStr) && user->gender.biProjected(GENDER_INDEX_PROJECTION) > &ADropdownList::selectionId
auto selectionId() const
Selected id property.
Definition ADropdownList.h:50

Note
We used the && operator here instead of & because we want the connection work in both directions: user.gender -> ADropdownList and ADropdownList -> user.gender.
  • If we try to change user->gender programmatically, ADropdownList will respond:
    user->gender = Gender::FEMALE;
    EXPECT_EQ(dropdownList->getSelectedId(), 1); // second option
  • If the user changes the value of ADropdownList, it reflects on the model as well:
    EXPECT_EQ(user->gender, Gender::OTHER);

Classes#

class  AProperty< T >
 Basic easy-to-use property implementation containing T. More...
 
class  APropertyDef< M, Getter, Setter, SignalArg >
 Property implementation to use with custom getter/setter. More...
 
class  APropertyPrecomputed< T >
 Readonly property that holds a value computed by an expression. More...
 
class  aui::PropertyModifier< Property >
 Temporary transparent object that gains write access to underlying property's value, notifying about value changes when destructed. More...
 

Function Documentation#

◆ biConnect()#

template<APropertyWritable PropertySource, APropertyWritable PropertyDestination>
requires requires { { *propertySource } -> aui::convertible_to<std::decay_t<decltype(*propertyDestination)>>; { *propertyDestination } -> aui::convertible_to<std::decay_t<decltype(*propertySource)>>; }
static void AObject::biConnect ( PropertySource && propertySource,
PropertyDestination && propertyDestination )
inlinestatic

Connects propertySource.changed to the setter of propertyDestination . Additionally, sets the propertyDestination with the current value of the propertySource (pre-fire). Hence, initial dataflow is from left argument to the right argument.

After pre-fire, connects propertyDestination.changed to the setter of propertySource . This way, when propertyDestination changes (i.e, propertyDestination belongs to some view and it's value is changed due to user action) it immediately reflects on propertySource . So, propertySource is typically a property of some view model with prefilled interesting data, and propertyDestination is a property of some view whose value is unimportant at the moment of connection creation.

biConnect pulls AObject from propertySource and propertyDestination to maintain the connection.

See signal-slot system for more info.

Parameters
propertySourcesource property, whose value is preserved on connection creation.
propertyDestinationdestination property, whose value is overwritten on connection creation.
Examples
examples/7guis/flight_booker/src/main.cpp, examples/7guis/temperature_converter/src/main.cpp, and examples/app/notes/src/main.cpp.

◆ connect() [1/3]#

template<AAnyProperty Property, typename Object, ACompatibleSlotFor< Property > Function>
requires (!aui::derived_from<Object, AObject>)
static void AObject::connect ( const Property & property,
_< Object > object,
Function && function )
inlinestatic

See signal-slot system for more info.

struct User { AProperty<AString> name }; // user.name here is non-AObject type
connect(textField->text(), user->name.assignment());
Note
object arg is accepted by value intentionally – this way we ensure that it would not be destroyed during connection creation.
Parameters
propertysource property.
objectinstance of AObject.
functionslot. Can be lambda.

◆ connect() [2/3]#

template<AAnyProperty Property, aui::derived_from< AObjectBase > Object, typename Function>
static decltype(auto) AObject::connect ( const Property & property,
Object * object,
Function && function )
inlinestatic

Connects to "changed" signal of the property. Additionally, calls specified function with the current value of the property (pre-fire).

See signal-slot system for more info.

connect(textField->text(), slot(otherObjectRawPtr)::handleText);
#define slot(v)
Passes some variable and type of the variable separated by comma. It's convenient to use with the con...
Definition kAUI.h:88
Parameters
propertyproperty
objectinstance of AObject
functionslot. Can be lambda

◆ connect() [3/3]#

template<APropertyReadable PropertySource, APropertyWritable PropertyDestination>
requires requires { { *propertySource } -> aui::convertible_to<std::decay_t<decltype(*propertyDestination)>>; }
static void AObject::connect ( PropertySource && propertySource,
PropertyDestination && propertyDestination )
inlinestatic

Connects propertySource.changed to the setter of propertyDestination . Additionally, sets the propertyDestination with the current value of the propertySource (pre-fire). Hence, dataflow is from left argument to the right argument.

connect pulls AObject from propertyDestination to maintain the connection.

See signal-slot system for more info.

Parameters
propertySourcesource property, whose value is preserved on connection creation.
propertyDestinationdestination property, whose value is overwritten on connection creation.