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

A set of functionality that introduces non-standard syntax which seems like the language features. More...

Detailed Description#

Note
For build-time macros, see Macros.

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.
 

Macro Definition Documentation#

◆ async#

#define async   AThreadPool::global() * [=]()
Returns
AFuture<T> where T is the return type of the lambda.
Note
When 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

auto task = async {
  AThread::sleep(1000); // a long task
};
static void sleep(std::chrono::milliseconds duration)
Sleep for specified duration. Most operation systems guarantee that elasped time will be greater than...
#define async
Executes following {} block asynchronously in the global thread pool. Unlike asyncX,...
Definition kAUI.h:329

Example with a return value

auto futureStatus = async {
  int status;
  ...
  AThread::sleep(1000); // a long task
  ...
  return status;
};
int status = *futureStatus;

Lambda operators are supported:

auto futureStatus = async mutable noexcept {
  int status;
  ...
  AThread::sleep(1000); // a long task
  ...
  return status;
};
int status = *futureStatus;

◆ asyncX#

#define asyncX   AThreadPool::global() *
Returns
AFuture<T> where T is the return type of the lambda.
Note
When 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

auto task = asyncX [&] {
  AThread::sleep(1000); // a long task
};
#define asyncX
Executes following {} block asynchronously in the global thread pool. Unlike async,...
Definition kAUI.h:360

Example with a return value

auto futureStatus = asyncX [&] {
  int status;
  ...
  AThread::sleep(1000); // a long task
  ...
  return status;
};
int status = *futureStatus;

◆ AUI_ASSERT#

#define AUI_ASSERT ( condition)
Value:
AUI_IMPL_ASSERT(condition)
Parameters
conditionthe 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.

See also
AUI_ASSERTX
AUI_ASSERT_NO_CONDITION
int v = 2 + 2;
AUI_ASSERT(v >= 0); // does not trigger
AUI_ASSERT(v != 4); // triggers
#define AUI_ASSERT(condition)
Asserts that the passed condition evaluates to true.
Definition Assert.h:55
Examples
examples/7guis/cells/src/AST.cpp.

◆ AUI_ASSERT_NO_CONDITION#

#define AUI_ASSERT_NO_CONDITION ( what)
Value:
AUI_IMPL_FAIL(what)
Parameters
whatstring 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.

See also
AUI_ASSERT
switch (i) {
  case 0: // ...
  default:
    AUI_ASSERT_NO_CONDITION("invalid i");
}
#define AUI_ASSERT_NO_CONDITION(what)
Always triggers assertion fail.
Definition Assert.h:94
Examples
examples/ui/views/src/ExampleWindow.cpp.

◆ AUI_ASSERT_UI_THREAD_ONLY#

#define AUI_ASSERT_UI_THREAD_ONLY ( )
Value:
{ AUI_ASSERTX(AWindow::current() == nullptr || AThread::current() == AWindow::current()->getThread(), "this method should be used in ui thread only."); }
static _< AAbstractThread > current()
static AWindowBase * current()
#define AUI_ASSERTX(condition, what)
Asserts that the passed condition evaluates to true. Adds extra message string.
Definition Assert.h:74

◆ AUI_ASSERT_WORKER_THREAD_ONLY#

#define AUI_ASSERT_WORKER_THREAD_ONLY ( )
Value:
{ AUI_ASSERTX(AWindow::current() == nullptr || AThread::current() != AWindow::current()->getThread(), "this method should be used in worker thread only."); }

◆ AUI_ASSERTX#

#define AUI_ASSERTX ( condition,
what )
Value:
AUI_IMPL_ASSERTX(condition, what)
Parameters
conditionthe expression
whatstring 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.

See also
AUI_ASSERT
int v = 2 + 2;
AUI_ASSERTX(v >= 0, "positive value expected"); // does not trigger
AUI_ASSERTX(v != 4, "4 is unacceptable value"); // triggers
Examples
/home/runner/work/aui/aui/aui.views/src/AUI/View/AView.h.

◆ AUI_BREAKPOINT#

#define AUI_BREAKPOINT ( )
Value:
AUI_IMPL_BREAKPOINT

If debugger is not present, behaviour is undefined.

AUI_BREAKPOINT(); // stops debugger here.
#define AUI_BREAKPOINT()
Stops program execution in attached debugger as if a breakpoint were set.
Definition Breakpoint.h:26

◆ AUI_DECLARATIVE_FOR#

#define AUI_DECLARATIVE_FOR ( value,
model,
layout )

◆ AUI_DEFER#

#define AUI_DEFER   ARaiiHelper AUI_PP_CAT($AUI_DEFER_at_line_, __LINE__) = [&]

This example prints "Hello world\n":

AUI_DEFER { printf(" world\n") };
printf("Hello");
#define AUI_DEFER
Defers execution of the next block to the end of current block (RAII scope).
Definition kAUI.h:196
See also
ARaiiHelper

◆ AUI_ENTRY#

#define AUI_ENTRY
Value:
    AUI_EXPORT int aui_entry(const AStringVector& args); \
    AUI_EXPORT int aui_main(JavaVM* vm, int(*aui_entry)(const AStringVector&)); \
extern "C" \
JNIEXPORT jint JNICALL \
JNI_OnLoad(JavaVM* vm, void* reserved) { \
        aui_main(vm, aui_entry); \
        return JNI_VERSION_1_2;  \
    } \
    AUI_EXPORT int aui_entry(const AStringVector& args)
An AVector with string-related functions.
Definition AStringVector.h:22

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:

/*
* AUI Framework - Declarative UI toolkit for modern C++20
* Copyright (C) 2020-2025 Alex2772 and Contributors
*
* SPDX-License-Identifier: MPL-2.0
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include "ExampleWindow.h"
#include <AUI/Platform/Entry.h>
    _new<ExampleWindow>()->show();
    return 0;
}
#define AUI_ENTRY
Application entry point.
Definition Entry.h:90

Minimal entrypoint of a console application:

/*
* AUI Framework - Declarative UI toolkit for modern C++20
* Copyright (C) 2020-2022 Alex2772
*
* SPDX-License-Identifier: MPL-2.0
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include <AUI/Platform/Entry.h>
#include <AUI/Logging/ALogger.h>
static constexpr auto LOG_TAG = "MyApp";
    ALogger::info(LOG_TAG) << "Hello world!";
    return 0;
}

Prototype of AUI_ENTRY function you are actually implementing:

AUI_EXPORT int aui_entry(const AStringVector& args);
Android-specific
Only the main thread is able to initialize graphics context.
iOS-specific
Only the main thread is able to initialize graphics context.
macOS-specific
Only the main thread is able to initialize graphics context.

◆ AUI_ENUM_VALUES#

#define AUI_ENUM_VALUES ( enum_t,
... )
Value:
template<> \
struct AEnumerateAllValues<enum_t>{         \
    static inline constexpr AEnumerate<enum_t>::Values<__VA_ARGS__> get() {return {}; } \
};                                         \
namespace std { inline AString to_wstring(enum_t v) { return AEnumerate<enum_t>::valueToNameMap().optional(v).valueOr("<unknown enum value {}>"_format(int(v))); } } \
inline std::ostream& operator<<(std::ostream& o, enum_t v) { return o << std::to_wstring(v); }
static const AMap< enum_t, AString, enum_less > & valueToNameMap()
Map runtime enum value to name.
Definition AEnumerate.h:183
Definition AEnumerate.h:159
Definition AEnumerate.h:91

Defines all enum values to by used by AEnumerate.

enum class ATextOverflow {
    NONE,
    ELLIPSIS,
    CLIP
};
                ATextOverflow::ELLIPSIS,
                ATextOverflow::CLIP)
// AEnumerate<ATextOverflow>::toName(ATextOverflow::CLIP) -> "CLIP"
ATextOverflow
Controls behavior of the overflowed text. Relevant to AAbstractLabel and its derivatives only.
Definition AOverflow.h:47
@ NONE
Definition AFloat.h:23
#define AUI_ENUM_VALUES(enum_t,...)
Defines all enum values for AEnumerate.
Definition AEnumerate.h:208

◆ AUI_MARK_AS_USED#

#define AUI_MARK_AS_USED ( variable)
Value:
{ (void)variable; }

Silences the unused variable compiler warning.

Can be used to force [=] lambda to capture a variable.

◆ AUI_NO_OPTIMIZE_OUT#

#define AUI_NO_OPTIMIZE_OUT ( object)
Value:
{ auto unused = &object ; }
Parameters
objectTarget 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:

struct SomeClass {
public:
  ~SomeClass() {
      AUI_NO_OPTIMIZE_OUT(SomeClass::debuggerMethod)
  }
  int debuggerMethod() {
    ...
  }
}
#define AUI_NO_OPTIMIZE_OUT(object)
Forbids object from being optimized out by compiler.
Definition macros.h:40
Examples
/home/runner/work/aui/aui/aui.core/src/AUI/Traits/iterators.h.

◆ AUI_NULLSAFE#

#define AUI_NULLSAFE ( s)
Value:
if(decltype(auto) _tmp = (s))_tmp
without with
if (getAnimator()) getAnimator()->postRender(this);
  
AUI_NULLSAFE(getAnimator())->postRender(this);
#define AUI_NULLSAFE(s)
nullsafe call (see examples).
Definition SharedPtrTypes.h:517

which is shorter, avoids code duplication and calls getAnimator() only once because AUI_NULLSAFE expands to:

if (auto& _tmp = (getAnimator())) _tmp->postRender(this);

Since AUI_NULLSAFE is a macro that expands to if, you can use else keyword:

AUI_NULLSAFE(getWindow())->flagRedraw(); else ALogger::info("Window is null!");

and even combine multiple AUI_NULLSAFE statements:

AUI_NULLSAFE(getWindow())->flagRedraw(); else AUI_NULLSAFE(AWindow::current())->flagRedraw();

◆ AUI_PERFORM_AS_MEMBER#

#define AUI_PERFORM_AS_MEMBER ( object,
lambda )
Value:
    struct __apply ## __FUNCTION__ ## __LINE__   : std::decay_t<decltype(object)> { \
        void operator()() {                                                    \
            lambda;                                                            \
        }                                                                      \
    };                                                                         \
    (static_cast<__apply ## __FUNCTION__ ## __LINE__ &>(object))()
Parameters
objectobject to perform operations on
lambdacode executed in the context of an object (as its member function)
Note
It's an analogue to 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>();
  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

◆ AUI_PP_CAT#

#define AUI_PP_CAT ( _1,
_2 )
Value:
AUI_PP_INTERNAL_CAT(_1, _2)

◆ AUI_PP_COMMA#

#define AUI_PP_COMMA ( ...)
Value:
,

◆ AUI_PP_COMMA_IF#

#define AUI_PP_COMMA_IF ( _i)
Value:
AUI_PP_CAT(AUI_PP_INTERNAL_COMMA_IF_, _i)
#define AUI_PP_CAT(_1, _2)
Expands and concatenates the arguments. Constructed macros reevaluate.
Definition APreprocessor.h:22

◆ AUI_PP_FOR_EACH#

#define AUI_PP_FOR_EACH ( _Macro,
_Data,
_Tuple )
Value:
  AUI_PP_CAT(AUI_PP_INTERNAL_FOR_EACH_IMPL_, AUI_PP_NARG0 _Tuple) \
  (0, _Macro, _Data, _Tuple)
#define AUI_PP_NARG0(...)
Evaluates to the number of arguments after expansion. Identifies 'empty' as 0.
Definition APreprocessor.h:170

Requires:

  • _Macro can be called with 3 arguments.
  • _Tuple expansion has no more than 15 elements.

◆ AUI_PP_GENERIC_IF#

#define AUI_PP_GENERIC_IF ( _Cond,
_Then,
_Else )
Value:
  AUI_PP_REMOVE_PARENS(AUI_PP_IF(_Cond, _Then, _Else))
#define AUI_PP_REMOVE_PARENS(...)
Remove the parens, requires AUI_PP_IS_ENCLOSED_PARENS(args) => 1.
Definition APreprocessor.h:193
#define AUI_PP_IF(_Cond, _Then, _Else)
Evaluates to _Then if _Cond is 1 and _Else if _Cond is 0.
Definition APreprocessor.h:138
AUI_PP_GENERIC_IF(1, (a, b, c), (d, e, f)) => a, b, c
AUI_PP_GENERIC_IF(0, (a, b, c), (d, e, f)) => d, e, f
#define AUI_PP_GENERIC_IF(_Cond, _Then, _Else)
Similar to AUI_PP_IF but takes _Then and _Else in parentheses.
Definition APreprocessor.h:150

◆ AUI_PP_HAS_COMMA#

#define AUI_PP_HAS_COMMA ( ...)
Value:
  AUI_PP_INTERNAL_16TH(       \
      (__VA_ARGS__, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0))

Requires no more than 15 unprotected commas.

◆ AUI_PP_HEAD#

#define AUI_PP_HEAD ( ...)
Value:
AUI_PP_INTERNAL_HEAD((__VA_ARGS__, unusedArg))

◆ AUI_PP_IDENTITY#

#define AUI_PP_IDENTITY ( _1)
Value:
_1

◆ AUI_PP_IF#

#define AUI_PP_IF ( _Cond,
_Then,
_Else )
Value:
  AUI_PP_CAT(AUI_PP_INTERNAL_IF_, _Cond)(_Then, _Else)

◆ AUI_PP_INC#

#define AUI_PP_INC ( _i)
Value:
AUI_PP_CAT(AUI_PP_INTERNAL_INC_, _i)

◆ AUI_PP_IS_BEGIN_PARENS#

#define AUI_PP_IS_BEGIN_PARENS ( ...)
Value:
  AUI_PP_HEAD(AUI_PP_CAT(AUI_PP_INTERNAL_IBP_IS_VARIADIC_R_, \
                             AUI_PP_INTERNAL_IBP_IS_VARIADIC_C __VA_ARGS__))
#define AUI_PP_HEAD(...)
Returns the first argument.
Definition APreprocessor.h:80

◆ AUI_PP_IS_EMPTY#

#define AUI_PP_IS_EMPTY ( ...)
Value:
  AUI_PP_INTERNAL_IS_EMPTY(AUI_PP_HAS_COMMA(__VA_ARGS__),                \
                             AUI_PP_HAS_COMMA(AUI_PP_COMMA __VA_ARGS__), \
                             AUI_PP_HAS_COMMA(__VA_ARGS__()),              \
                             AUI_PP_HAS_COMMA(AUI_PP_COMMA __VA_ARGS__()))
#define AUI_PP_HAS_COMMA(...)
Returns 1 if the expansion of arguments has an unprotected comma.Otherwise returns 0.
Definition APreprocessor.h:72
#define AUI_PP_COMMA(...)
Returns a comma. Given a variadic number of arguments.
Definition APreprocessor.h:40

Requires:

  • the number of arguments after expansion is at most 15.
  • If the argument is a macro, it must be able to be called with one argument.

Implementation details#

There is one case when it generates a compile error: if the argument is macro that cannot be called with one argument.

#define M(a, b)  // it doesn't matter what it expands to
// Expected: expands to `0`.
// Actual: compile error.
#define AUI_PP_IS_EMPTY(...)
If the arguments after expansion have no tokens, evaluates to 1. Otherwise evaluates to 0.
Definition APreprocessor.h:128

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.

◆ AUI_PP_IS_ENCLOSED_PARENS#

#define AUI_PP_IS_ENCLOSED_PARENS ( ...)
Value:
  AUI_PP_IF(AUI_PP_IS_BEGIN_PARENS(__VA_ARGS__), \
              AUI_PP_IS_EMPTY(AUI_PP_EMPTY __VA_ARGS__), 0)
#define AUI_PP_EMPTY(...)
Returns empty. Given a variadic number of arguments.
Definition APreprocessor.h:34
#define AUI_PP_IS_BEGIN_PARENS(...)
Expands to 1 if the first argument starts with something in parentheses, otherwise to 0.
Definition APreprocessor.h:177

◆ AUI_PP_NARG#

#define AUI_PP_NARG ( ...)
Value:
  AUI_PP_INTERNAL_16TH(  \
      (__VA_ARGS__, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0))
#define PAIR x, y
AUI_PP_NARG(x) => 1
AUI_PP_NARG(x, y) => 2
AUI_PP_NARG(PAIR) => 2
#define AUI_PP_NARG(...)
Evaluates to the number of arguments after expansion.
Definition APreprocessor.h:63

Requires: the number of arguments after expansion is at most 15.

◆ AUI_PP_NARG0#

#define AUI_PP_NARG0 ( ...)
Value:
  AUI_PP_IF(AUI_PP_IS_EMPTY(__VA_ARGS__), 0, AUI_PP_NARG(__VA_ARGS__))
#define PAIR x, y
AUI_PP_NARG0(x) => 1
AUI_PP_NARG0(x, y) => 2
AUI_PP_NARG0(PAIR) => 2

Requires:

  • the number of arguments after expansion is at most 15.
  • If the argument is a macro, it must be able to be called with one argument.

◆ AUI_PP_REMOVE_PARENS#

#define AUI_PP_REMOVE_PARENS ( ...)
Value:
AUI_PP_INTERNAL_REMOVE_PARENS __VA_ARGS__

◆ AUI_PP_REPEAT#

#define AUI_PP_REPEAT ( _Macro,
_Data,
_N )
Value:
  AUI_PP_CAT(AUI_PP_INTERNAL_FOR_EACH_IMPL_, _N) \
  (0, _Macro, _Data, AUI_PP_INTENRAL_EMPTY_TUPLE)

Empty if _K = 0. Requires:

  • _Macro can be called with 3 arguments.
  • _K literal between 0 and 15.

◆ AUI_PP_STRINGIZE#

#define AUI_PP_STRINGIZE ( ...)
Value:
AUI_PP_INTERNAL_STRINGIZE(__VA_ARGS__)

◆ AUI_PP_TAIL#

#define AUI_PP_TAIL ( ...)
Value:
AUI_PP_INTERNAL_TAIL((__VA_ARGS__))

Requires at least one argument.

◆ AUI_PP_VARIADIC_CALL#

#define AUI_PP_VARIADIC_CALL ( _Macro,
... )
Value:
  AUI_PP_IDENTITY(                        \
      AUI_PP_CAT(_Macro, AUI_PP_NARG(__VA_ARGS__))(__VA_ARGS__))
#define AUI_PP_IDENTITY(_1)
Returns the only argument.
Definition APreprocessor.h:46

◆ do_once#

#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!");
}
  
    std::printf("Only once!");
}
#define do_once
Executes following function call or {} block once per program execution.
Definition kAUI.h:391

◆ let#

#define let   ^ [&](const auto& it)
Parameters
Tobject type to perform operations on
lambdacode executed in the context of an object (as its member function)
Note
analogue to with, apply in Kotlin

let allows to call methods of newly created objects right in place. For example:

without with
auto tf = _new<ATextField>();
tf->setText("Hello!");
  
auto tf = _new<ATextField>() let { it->setText("Hello!"); };
#define let
Performs multiple operations on a single object without repeating its name (in place) This function c...
Definition kAUI.h:262

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,
});
  
// clean, less code and easy to understand
setContents(Vertical {
    _new<ATextField>() let { it->setText("Hello!") },
    _new<ATextField>() let { it->setText("World!") },
});
Examples
examples/7guis/circle_drawer/src/main.cpp, examples/7guis/flight_booker/src/main.cpp, examples/7guis/temperature_converter/src/main.cpp, examples/app/game_of_life/src/main.cpp, examples/app/notes/src/main.cpp, examples/ui/contacts/src/main.cpp, and examples/ui/views/src/ExampleWindow.cpp.

◆ with_style#

#define with_style   & ass::PropertyListRecursive
#include <AUI/ASS/ASS.h>
using namespace ass;
...
setContents(Centered {
  _new<ALabel>("Red text!") with_style { TextColor { AColor::RED } },
});
#define with_style
Allows to define a style to the view right in place.
Definition kAUI.h:287
Controls the text color of AView.
Definition TextColor.h:26

Also applicable to declarative-style views:

#include <AUI/ASS/ASS.h>
using namespace ass;
...
setContents(Centered {
  Label { "Red text!" } with_style { TextColor { AColor::RED } },
});
Examples
examples/7guis/cells/src/main.cpp, examples/7guis/crud/src/main.cpp, examples/7guis/temperature_converter/src/main.cpp, examples/app/game_of_life/src/main.cpp, examples/app/notes/src/main.cpp, and examples/ui/contacts/src/main.cpp.