AUI Framework
master
Cross-platform base for C++ UI apps
|
A set of functionality that introduces non-standard syntax which seems like the language features. More...
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 (*AThread::main()) * [=]() |
Executes lambda on main thread. | |
#define | ui_threadX (*AThread::main()) * |
Executes lambda on main 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 | AUI_DECLARATIVE_FOR(value, model, layout) |
ranged-for-loop style wrapped for AForEachUI. | |
#define async AThreadPool::global() * [=]() |
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() * |
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 | ) |
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 | ) |
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 | ( | ) |
#define AUI_ASSERT_WORKER_THREAD_ONLY | ( | ) |
#define AUI_ASSERTX | ( | condition, | |
what ) |
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 | ( | ) |
If debugger is not present, behaviour is undefined.
#define AUI_DECLARATIVE_FOR | ( | value, | |
model, | |||
layout ) |
See AForEachUI
#define AUI_DEFER ARaiiHelper AUI_PP_CAT($AUI_DEFER_at_line_, __LINE__) = [&] |
This example prints "Hello world\n":
#define AUI_ENTRY |
Entry point of any AUI application.
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, effectively 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 with the exit code returned by AUI_ENTRY
earlier, or 0
.
Application arguments (int argc, char** argv)
are forwarded to AUI_ENTRY
as AStringVector args
.
Minimal entrypoint of an UI application:
Minimal entrypoint of a console application:
Prototype of AUI_ENTRY
function you are actually implementing:
#define AUI_ENUM_VALUES | ( | enum_t, | |
... ) |
Defines all enum values to by used by AEnumerate.
#define AUI_MARK_AS_USED | ( | variable | ) |
Silences the unused variable compiler warning.
Can be used to force [=] lambda to capture a variable.
#define AUI_NO_OPTIMIZE_OUT | ( | object | ) |
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 | ) |
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 ) |
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 ) |
#define AUI_PP_COMMA | ( | ... | ) |
#define AUI_PP_COMMA_IF | ( | _i | ) |
#define AUI_PP_FOR_EACH | ( | _Macro, | |
_Data, | |||
_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 ) |
#define AUI_PP_HAS_COMMA | ( | ... | ) |
Requires no more than 15 unprotected commas.
#define AUI_PP_HEAD | ( | ... | ) |
#define AUI_PP_IDENTITY | ( | _1 | ) |
#define AUI_PP_IF | ( | _Cond, | |
_Then, | |||
_Else ) |
#define AUI_PP_INC | ( | _i | ) |
#define AUI_PP_IS_BEGIN_PARENS | ( | ... | ) |
#define AUI_PP_IS_EMPTY | ( | ... | ) |
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 | ( | ... | ) |
#define AUI_PP_NARG | ( | ... | ) |
Requires: the number of arguments after expansion is at most 15.
#define AUI_PP_NARG0 | ( | ... | ) |
Requires:
#define AUI_PP_REMOVE_PARENS | ( | ... | ) |
#define AUI_PP_REPEAT | ( | _Macro, | |
_Data, | |||
_N ) |
Empty if _K = 0
. Requires:
_Macro
can be called with 3 arguments._K
literal between 0 and 15. #define AUI_PP_STRINGIZE | ( | ... | ) |
#define AUI_PP_TAIL | ( | ... | ) |
Requires at least one argument.
#define AUI_PP_VARIADIC_CALL | ( | _Macro, | |
... ) |
#define do_once if(static bool _aui_once = false; (!_aui_once && (_aui_once = true))) |
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) |
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 |
Also applicable to declarative-style views: