AUI Framework  master
Cross-platform base for C++ UI apps
Loading...
Searching...
No Matches
aui::PropertyModifier< Property > Class Template Reference

Temporary transparent object that gains write access to underlying property's value, notifying about value changes when destructed. More...

#include </home/runner/work/aui/aui/doxygen/intermediate/property_modifier.h>

Detailed Description#

template<typename Property>
class aui::PropertyModifier< Property >

PropertyModifier is a result of writeScope() method of writeable properties. Also, it is used inside non-const operator implementations (see below). It gains transparent writeable handle to property's value, and calls notify() method on associated property upon PropertyModifier destruction.

Non-const operators of properties such as non-const versions of operator=, operator+=, operator-= have a side effect of emitting changed signal upon operation completion. This ensures that modifying access to the property can be observed.

AProperty<int> counter = 0;
AObject::connect(counter.changed, slot(observer)::observeInt);
EXPECT_CALL(observer, observeInt(1)).Times(1);
counter += 1; // observable by observeInt
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 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
Basic easy-to-use property implementation containing T.
Definition AProperty.h:30
emits< T > changed
Signal that notifies data changes.
Definition AProperty.h:47
AProperty<AString> name = "Hello";
AObject::connect(name.changed, slot(observer)::observeString);
EXPECT_CALL(observer, observeString("Hello world"_as)).Times(1);
name += " world"; // observable by observeString
Note
Make sure your read-only operators (such as operator+, operator-) have marked const, otherwise property would treat them as a writing access, resulting in unwanted signaling changed upon each access.
AProperty<int> counter = 0;
AObject::connect(counter.changed, slot(observer)::observeInt);
EXPECT_CALL(observer, observeInt(testing::_)).Times(0);
int nextCounter = counter + 1; // read-only access; noone is notified

Member access operator (operator->)#

operator-> is a special case. operator-> having both non-const and const versions is a common practice, so there's should be a way to distinguish between non-const access and const access, preferring the latter if possible. The const version of operator-> can be used directly on property:

EXPECT_CALL(observer, observeString(testing::_)).Times(0);
AProperty<AString> name = "Hello";
AObject::connect(name.changed, slot(observer)::observeString);
// returns const pointer
auto data = name->data();

Property System is designed in such a way you would explicitly express a modifying operation via binary equals operator (and favours such as +=, -=):

AProperty<AString> name = "Hello";
AObject::connect(name.changed, slot(observer)::observeString);
EXPECT_CALL(observer, observeString("Test"_as)).Times(1);
name = "Test";

However, it is still possible to achieve non-const version of operator->. To do this, you need a aui::PropertyModifier object that grants such access:

AProperty<AString> name = "Hello";
AObject::connect(name.changed, slot(observer)::observeString);
EXPECT_CALL(observer, observeString("Hell"_as)).Times(1);
name.writeScope()->removeAll('o');

You need to be careful when performing multiple operations at once. Design rationale behind writeScope() method makes it painful (by intention) performing multiple accesses, since it would lead to unwanted change notifications during the process:

// WRONG WAY
EXPECT_CALL(observer, observeString("Hell"_as)).Times(1);
name.writeScope()->removeAll('o');
EXPECT_CALL(observer, observeString("He"_as)).Times(1);
name.writeScope()->removeAll('l');

The right way is to create aui::PropertyModifier just once. This will produce exactly one notification, ensuring that modifications to the property are performed atomically. This means that all operations within the scope of aui::PropertyModifier produced by writeScope() will be treated as one unit, and only one change notification will be emitted.

// RIGHT WAY
EXPECT_CALL(observer, observeString("He"_as)).Times(1);
auto nameWriteable = name.writeScope();
nameWriteable->removeAll('o');
nameWriteable->removeAll('l');