AUI Framework
develop
Cross-platform base for C++ UI apps
|
A set of functionality that introduces non-standard syntax which seems like the language features.
Macros | |
#define | AUI_NULLSAFE(s) |
nullsafe call (see examples). | |
#define | AUI_ENTRY |
Application entry point. | |
#define | AUI_ENUM_VALUES(enum_t, ...) |
Defines all enum values for AEnumerate. | |
#define | AUI_NO_OPTIMIZE_OUT(object) |
Forbids object from being optimized out by compiler. | |
#define | AUI_MARK_AS_USED(variable) |
Marks the variable as being used. | |
#define | AUI_PP_CAT(_1, _2) |
Expands and concatenates the arguments. Constructed macros reevaluate. | |
#define | AUI_PP_STRINGIZE(...) |
Expands and stringifies the only argument. | |
#define | AUI_PP_EMPTY(...) |
Returns empty. Given a variadic number of arguments. | |
#define | AUI_PP_COMMA(...) |
Returns a comma. Given a variadic number of arguments. | |
#define | AUI_PP_IDENTITY(_1) |
Returns the only argument. | |
#define | AUI_PP_NARG(...) |
Evaluates to the number of arguments after expansion. | |
#define | AUI_PP_HAS_COMMA(...) |
Returns 1 if the expansion of arguments has an unprotected comma.Otherwise returns 0. | |
#define | AUI_PP_HEAD(...) |
Returns the first argument. | |
#define | AUI_PP_TAIL(...) |
Returns the tail. A variadic list of all arguments minus the first. | |
#define | AUI_PP_VARIADIC_CALL(_Macro, ...) |
Calls CAT(_Macro, NARG(VA_ARGS))(VA_ARGS) | |
#define | AUI_PP_IS_EMPTY(...) |
If the arguments after expansion have no tokens, evaluates to 1 . Otherwise evaluates to 0 . | |
#define | AUI_PP_IF(_Cond, _Then, _Else) |
Evaluates to _Then if _Cond is 1 and _Else if _Cond is 0. | |
#define | AUI_PP_GENERIC_IF(_Cond, _Then, _Else) |
Similar to AUI_PP_IF but takes _Then and _Else in parentheses. | |
#define | AUI_PP_NARG0(...) |
Evaluates to the number of arguments after expansion. Identifies 'empty' as 0. | |
#define | AUI_PP_IS_BEGIN_PARENS(...) |
Expands to 1 if the first argument starts with something in parentheses, otherwise to 0. | |
#define | AUI_PP_IS_ENCLOSED_PARENS(...) |
Expands to 1 is there is only one argument and it is enclosed in parentheses. | |
#define | AUI_PP_REMOVE_PARENS(...) |
Remove the parens, requires AUI_PP_IS_ENCLOSED_PARENS(args) => 1. | |
#define | AUI_PP_FOR_EACH(_Macro, _Data, _Tuple) |
Expands to _Macro(0, _Data, e1) _Macro(1, _Data, e2) ... _Macro(K -1, _Data, eK) as many of AUI_INTERNAL_NARG0 _Tuple. | |
#define | AUI_PP_REPEAT(_Macro, _Data, _N) |
Expands to _Macro(0, _Data, ) _Macro(1, _Data, ) ... _Macro(K - 1, _Data, ) | |
#define | AUI_PP_INC(_i) |
Increments the argument, requires the argument to be between 0 and 15. | |
#define | AUI_PP_COMMA_IF(_i) |
Returns comma if _i != 0 . Requires _i to be between 0 and 15. | |
#define | AUI_ASSERT(condition) |
Asserts that the passed condition evaluates to true. | |
#define | AUI_ASSERTX(condition, what) |
Asserts that the passed condition evaluates to true. Adds extra message string. | |
#define | AUI_ASSERT_NO_CONDITION(what) |
Always triggers assertion fail. | |
#define | AUI_BREAKPOINT() |
Stops program execution in attached debugger as if a breakpoint were set. | |
#define | AUI_PERFORM_AS_MEMBER(object, lambda) |
Performs multiple operations on a single object without repeating its name. | |
#define | AUI_DEFER ARaiiHelper AUI_PP_CAT($AUI_DEFER_at_line_, __LINE__) = [&] |
Defers execution of the next block to the end of current block (RAII scope). | |
#define | let ^ [&](const auto& it) |
Performs multiple operations on a single object without repeating its name (in place) This function can be used as an operator on object. | |
#define | with_style + ass::PropertyListRecursive |
Allows to define a style to the view right in place. | |
#define | async AThreadPool::global() * [=]() |
Executes following {} block asynchronously in the global thread pool. Unlike asyncX, does now allow to set lambda's capture. Lambda's capture is [=] . | |
#define | asyncX AThreadPool::global() * |
Executes following {} block asynchronously in the global thread pool. Unlike async, allows to set lambda's capture but you should always specify lambda's capture. | |
#define | do_once if(static bool _aui_once = false; (!_aui_once && (_aui_once = true))) |
Executes following function call or {} block once per program execution. | |
#define | ui_thread (*getThread()) * [=]() |
Executes lambda on current object's thread. | |
#define | ui_threadX (*getThread()) * |
Executes lambda on current object's thread. Allows to determine lambda's capture. | |
#define | AUI_ASSERT_UI_THREAD_ONLY() |
Asserts that the macro invocation has been performed in the UI thread. | |
#define | AUI_ASSERT_WORKER_THREAD_ONLY() |
Asserts that the macro invocation has not been performed in the UI thread. | |
#define async AThreadPool::global() * [=]() |
Executes following {} block asynchronously in the global thread pool. Unlike asyncX, does now allow to set lambda's capture. Lambda's capture is [=]
.
AFuture<T>
where T
is the return type of the lambda. AFuture<T>
is destroyed, the corresponding async
task is either cancelled or removed from the execution queue. Use AFutureSet or AAsyncHolder to keep multiple AFuture<T> alive.Example without a return value
Example with a return value
Lambda operators are supported:
#define asyncX AThreadPool::global() * |
Executes following {} block asynchronously in the global thread pool. Unlike async, allows to set lambda's capture but you should always specify lambda's capture.
AFuture<T>
where T
is the return type of the lambda. AFuture<T>
is destroyed, the corresponding async
task is either cancelled or removed from the execution queue. Use AFutureSet or AAsyncHolder to keep multiple AFuture<T> alive.Example without a return value
Example with a return value
#define AUI_ASSERT | ( | condition | ) |
Asserts that the passed condition evaluates to true.
condition | the expression |
If the condition evaluates to false, triggers default C++ assert behavior (that is, program termination) on debug build or throws AAssertionFailedException on release builds, so it can be handled and reported properly in production applications.
#define AUI_ASSERT_NO_CONDITION | ( | what | ) |
Always triggers assertion fail.
what | string literal which will be appended to the error message |
Triggers default C++ assert behavior (that is, program termination) on debug build or throws AAssertionFailedException on release builds, so it can be handled and reported properly in production applications.
#define AUI_ASSERT_UI_THREAD_ONLY | ( | ) |
Asserts that the macro invocation has been performed in the UI thread.
#define AUI_ASSERT_WORKER_THREAD_ONLY | ( | ) |
Asserts that the macro invocation has not been performed in the UI thread.
#define AUI_ASSERTX | ( | condition, | |
what ) |
Asserts that the passed condition evaluates to true. Adds extra message string.
condition | the expression |
what | string literal which will be appended to the error message |
If the condition evaluates to false, triggers default C++ assert behavior (that is, program termination) on debug build or throws AAssertionFailedException on release builds, so it can be handled and reported properly in production applications.
#define AUI_BREAKPOINT | ( | ) |
Stops program execution in attached debugger as if a breakpoint were set.
If debugger is not present, behaviour is undefined.
#define AUI_DEFER ARaiiHelper AUI_PP_CAT($AUI_DEFER_at_line_, __LINE__) = [&] |
Defers execution of the next block to the end of current block (RAII scope).
This example prints "Hello world\n":
#define AUI_ENTRY |
Application entry point.
Native entry point varies platform to platform (i.e. Windows requires int main
entry point for console applications and WinMain
for graphical applications; entry point of an Android application is located in Java code). AUI_ENTRY unifies your entry point, efficiently supporting every platform.
AUI_ENTRY of a graphical application should be non-blocking since on mobile platforms application's event loop is located outsize of the entry point. On desktop platforms an event loop is created outside AUI_ENTRY in order to unify the mobile and desktop behaviour. If there are no open windows, the event loop breaks causing the application to exit.
Application arguments (int argc, char** argv) are forwarded to AUI_ENTRY as AStringVector args
.
#define AUI_ENUM_VALUES | ( | enum_t, | |
... ) |
Defines all enum values for AEnumerate.
Defines all enum values to by used by AEnumerate.
#define AUI_MARK_AS_USED | ( | variable | ) |
Marks the variable as being used.
Silences the unused variable compiler warning.
Can be used to force [=] lambda to capture a variable.
#define AUI_NO_OPTIMIZE_OUT | ( | object | ) |
Forbids object from being optimized out by compiler.
object | Target object. Can be pointer-to-function, pointer-to-member, any variable. |
Performs compiler hacks in order to prevent the compiler/linker from optimizing out the specified object. It's useful for debugging purposes, when the function is expected to exist to be called.
The macro should be called in any function/method which is not being optimized out (i.e. in destructor).
Basic usage:
#define AUI_NULLSAFE | ( | s | ) |
nullsafe call (see examples).
without | with |
if (getAnimator()) getAnimator()->postRender(this);
| AUI_NULLSAFE(getAnimator())->postRender(this);
|
which is shorter, avoids code duplication and calls getAnimator()
only once because AUI_NULLSAFE
expands to:
Since AUI_NULLSAFE
is a macro that expands to if
, you can use else
keyword:
and even combine multiple AUI_NULLSAFE
statements:
#define AUI_PERFORM_AS_MEMBER | ( | object, | |
lambda ) |
Performs multiple operations on a single object without repeating its name.
object | object to perform operations on |
lambda | code executed in the context of an object (as its member function) |
with
, apply
in Kotlin.without | with |
class Worker {
public:
void buildHouse();
void plantTree();
void raiseSon();
};
...
auto worker = _new<Worker>();
worker->buildHouse();
worker->plantTree();
worker->raiseSon();
| class Worker {
public:
void buildHouse();
void plantTree();
void raiseSon();
};
...
auto worker = _new<Worker>();
AUI_PERFORM_AS_MEMBER(*worker, {
buildHouse();
plantTree();
raiseSon();
});
#define AUI_PERFORM_AS_MEMBER(object, lambda) Performs multiple operations on a single object without repeating its name. Definition kAUI.h:140 |
#define AUI_PP_CAT | ( | _1, | |
_2 ) |
Expands and concatenates the arguments. Constructed macros reevaluate.
#define AUI_PP_COMMA | ( | ... | ) |
Returns a comma. Given a variadic number of arguments.
#define AUI_PP_COMMA_IF | ( | _i | ) |
Returns comma if _i != 0
. Requires _i
to be between 0 and 15.
#define AUI_PP_FOR_EACH | ( | _Macro, | |
_Data, | |||
_Tuple ) |
Expands to _Macro(0, _Data, e1) _Macro(1, _Data, e2) ... _Macro(K -1, _Data, eK) as many of AUI_INTERNAL_NARG0 _Tuple.
Requires:
_Macro
can be called with 3 arguments._Tuple
expansion has no more than 15 elements. #define AUI_PP_GENERIC_IF | ( | _Cond, | |
_Then, | |||
_Else ) |
Similar to AUI_PP_IF but takes _Then and _Else in parentheses.
#define AUI_PP_HAS_COMMA | ( | ... | ) |
Returns 1 if the expansion of arguments has an unprotected comma.Otherwise returns 0.
Requires no more than 15 unprotected commas.
#define AUI_PP_HEAD | ( | ... | ) |
Returns the first argument.
#define AUI_PP_IDENTITY | ( | _1 | ) |
Returns the only argument.
#define AUI_PP_IF | ( | _Cond, | |
_Then, | |||
_Else ) |
Evaluates to _Then if _Cond is 1 and _Else if _Cond is 0.
#define AUI_PP_INC | ( | _i | ) |
Increments the argument, requires the argument to be between 0 and 15.
#define AUI_PP_IS_BEGIN_PARENS | ( | ... | ) |
Expands to 1 if the first argument starts with something in parentheses, otherwise to 0.
#define AUI_PP_IS_EMPTY | ( | ... | ) |
If the arguments after expansion have no tokens, evaluates to 1
. Otherwise evaluates to 0
.
Requires:
There is one case when it generates a compile error: if the argument is macro that cannot be called with one argument.
There are 4 cases tested:
__VA_ARGS__
possible expansion has no unparen'd commas. Expected 0.__VA_ARGS__
possible expansion is not enclosed in parenthesis. Expected 0.__VA_ARGS__
possible expansion is not a macro that ()-evaluates to a comma. Expected 0__VA_ARGS__
is empty, or has unparen'd commas, or is enclosed in parenthesis, or is a macro that ()-evaluates to comma. Expected 1.We trigger detection on '0001', i.e. on empty.
#define AUI_PP_IS_ENCLOSED_PARENS | ( | ... | ) |
Expands to 1 is there is only one argument and it is enclosed in parentheses.
#define AUI_PP_NARG | ( | ... | ) |
Evaluates to the number of arguments after expansion.
Requires: the number of arguments after expansion is at most 15.
#define AUI_PP_NARG0 | ( | ... | ) |
Evaluates to the number of arguments after expansion. Identifies 'empty' as 0.
Requires:
#define AUI_PP_REMOVE_PARENS | ( | ... | ) |
Remove the parens, requires AUI_PP_IS_ENCLOSED_PARENS(args) => 1.
#define AUI_PP_REPEAT | ( | _Macro, | |
_Data, | |||
_N ) |
Expands to _Macro(0, _Data, ) _Macro(1, _Data, ) ... _Macro(K - 1, _Data, )
Empty if _K = 0
. Requires:
_Macro
can be called with 3 arguments._K
literal between 0 and 15. #define AUI_PP_STRINGIZE | ( | ... | ) |
Expands and stringifies the only argument.
#define AUI_PP_TAIL | ( | ... | ) |
Returns the tail. A variadic list of all arguments minus the first.
Requires at least one argument.
#define AUI_PP_VARIADIC_CALL | ( | _Macro, | |
... ) |
Calls CAT(_Macro, NARG(VA_ARGS))(VA_ARGS)
#define do_once if(static bool _aui_once = false; (!_aui_once && (_aui_once = true))) |
Executes following function call or {} block once per program execution.
Guarantees that the following code block will be executed only once per application execution.
without | with |
static bool done = false;
if (!done) {
done = true;
std::printf("Only once!");
}
|
#define let ^ [&](const auto& it) |
Performs multiple operations on a single object without repeating its name (in place) This function can be used as an operator on object.
T | object type to perform operations on |
lambda | code executed in the context of an object (as its member function) |
with
, apply
in Kotlinlet
allows to call methods of newly created objects right in place. For example:
without | with |
auto tf = _new<ATextField>();
tf->setText("Hello!");
|
It's especially useful when building user interfaces:
without | with |
auto tf1 = _new<ATextField>();
tf1->setText("Hello!");
auto tf2 = _new<ATextField>();
tf2->setText("World!");
...
setContents(Vertical { // confusing
tf1,
tf2,
});
|
#define with_style + ass::PropertyListRecursive |
Allows to define a style to the view right in place.
Also applicable to declarative-style views: