17 #ifndef NURIA_CALLBACK_HPP
18 #define NURIA_CALLBACK_HPP
20 #include <type_traits>
23 #include <QSharedData>
28 #include "essentials.hpp"
29 #include "variant.hpp"
32 class CallbackPrivate;
35 namespace CallbackHelper {
36 template<
int ... Index >
44 typedef typename CreateIndexTuple< Size - 1 >::type::template append< Size - 1 > type;
52 template<
typename... Types >
53 using buildIndexTuple =
typename CreateIndexTuple<
sizeof...(Types) + 1 >::type;
56 template<
typename T >
57 constexpr
int typeId () {
return qMetaTypeId< T > (); }
60 constexpr
int typeId< void > () {
return 0; }
63 template<
typename T >
64 void getArguments (
void **list,
int *types,
int idx, T *cur) {
66 types[idx] = qMetaTypeId< typename std::remove_const< T >::type > ();
69 template<
typename T,
typename T2,
typename ... Args >
70 void getArguments (
void **list,
int *types,
int idx,
71 T *cur, T2 *next, Args * ... args) {
72 getArguments (list, types, idx, cur);
73 getArguments (list, types, idx + 1, next, args ...);
143 struct TrampolineBase;
144 template<
typename Ret,
typename ... Args >
struct MethodHelper;
145 template<
typename Class,
typename Ret,
typename ... Args >
struct MemberMethodHelper;
146 template<
typename FullType,
typename Ret,
typename ... Args >
struct LambdaHelper;
187 Callback (QObject *receiver,
const char *slot,
bool variadic =
false,
188 Qt::ConnectionType connectionType = Qt::AutoConnection);
194 template<
typename Ret,
typename ... Args >
195 Callback (Ret (*func)(Args ...),
bool variadic =
false)
196 : d (0) { setCallback (func); setVariadic (variadic); }
199 template<
typename Class,
typename Ret,
typename ... Args >
200 Callback (Class *instance, Ret (Class::*func)(Args ...),
bool variadic =
false)
201 : d (0) { setCallback (instance, func); setVariadic (variadic); }
204 template<
typename Class,
typename Ret,
typename ... Args >
205 Callback (Class *instance, Ret (Class::*func)(Args ...)
const,
bool variadic =
false)
206 : d (0) { setCallback (instance, func); setVariadic (variadic); }
208 template<
typename Ret,
typename ... Args >
209 Callback (
const std::function< Ret(Args ...) > &func,
bool variadic =
false)
210 : d (0) { setCallback (func); setVariadic (variadic); }
213 template<
typename Lambda >
215 Callback cb = fromLambdaImpl (l, &Lambda::operator());
235 bool operator== (
const Callback &other)
const;
238 bool isValid ()
const;
244 bool isVariadic ()
const;
247 void setVariadic (
bool variadic);
250 int returnType ()
const;
253 QList< int > argumentTypes ()
const;
260 template<
typename Ret,
typename ... Args >
261 bool setCallback (Ret (*func)(Args ...)) {
263 buildArgList< Args ... > (args);
264 return initBase (
new MethodHelper< Ret, Args ... > (func),
265 CallbackHelper::typeId< Ret > (), args);
269 template<
typename Class,
typename Ret,
typename ... Args >
270 bool setCallback (Class *instance, Ret (Class::*func)(Args ...)
const) {
271 typedef Ret(Class::*WithoutConst)(Args ...);
272 return setCallback (instance, (WithoutConst)func);
275 template<
typename Class,
typename Ret,
typename ... Args >
276 bool setCallback (Class *instance, Ret (Class::*func)(Args ...)) {
278 buildArgList< Args ... > (args);
279 return initBase (
new MemberMethodHelper< Class, Ret, Args ... > (instance, func),
280 CallbackHelper::typeId< Ret > (), args);
284 template<
typename Ret,
typename ... Args >
285 bool setCallback (std::function< Ret(Args ...) > func) {
287 buildArgList< Args ... > (args);
288 return initBase (
new LambdaHelper< std::function< Ret(Args ...) >, Ret, Args ... > (func),
289 CallbackHelper::typeId< Ret > (), args);
303 bool setCallback (QObject *receiver,
const char *slot,
304 Qt::ConnectionType connectionType = Qt::AutoConnection);
307 template<
typename Ret,
typename ... Args >
308 Callback &operator= (Ret (*func)(Args ...)) { setCallback (func);
return *
this; }
319 void bindList (
const QVariantList &arguments = QVariantList());
324 template<
typename Lambda,
typename ... Args >
326 Callback cb = fromLambdaImpl (func, &Lambda::operator());
336 template<
typename ... Args >
347 QVariant invoke (
const QVariantList &arguments)
const;
352 template<
typename ... Args >
353 QVariant operator() (
const Args &... args)
const {
354 void *list[
sizeof... (Args)];
355 int types[
sizeof... (Args)];
356 CallbackHelper::getArguments (list, types, 0, const_cast< Args * > (&args) ...);
357 return invoke (
sizeof... (Args), list, types);
361 inline QVariant operator() () {
362 return invoke (0, 0, 0);
367 friend class CallbackPrivate;
369 QVariant invoke (
int count,
void **args,
int *types)
const;
370 QVariant invokePrepared (
const QVariantList &arguments)
const;
373 QVariant invokeInternal (
int count,
void **args,
int *types)
const;
378 template<
typename T >
struct removeRef {
typedef T type; };
379 template<
typename T >
struct removeRef< T & >;
380 template<
typename T >
struct removeRef< const T & > {
typedef T type; };
381 template<
typename T >
struct removeRef< const T * > {
typedef T* type; };
386 template<
typename Lambda,
typename Ret,
typename ... Args >
387 inline static Callback fromLambdaImpl (Lambda l, Ret (Lambda::*)(Args ...)
const) {
388 return Callback (std::function< Ret(Args ...) > (l));
395 bool initBase (TrampolineBase *base,
int retType,
const QList< int > &args);
397 template<
typename T1,
typename ... T2 >
398 inline void buildArgListDo (QList< int > &list, T1 cur, T2 ... next) {
400 buildArgListDo (list, next ...);
403 template<
typename T >
404 inline void buildArgListDo (QList< int > &list, T cur)
405 { list.append (cur); }
408 inline void buildArgListDo (QList< int > &) { }
415 template<
typename ... Args >
416 inline void buildArgList (QList< int > &list) {
417 buildArgListDo (list, qMetaTypeId<
typename removeRef< Args >::type > () ...);
425 struct TrampolineBase {
427 TrampolineBase (Type t) : type (t) {}
429 virtual ~TrampolineBase () {}
430 virtual void trampoline (
void **) = 0;
433 struct MemberTrampolineBase : TrampolineBase {
434 MemberTrampolineBase (
void *inst) : TrampolineBase (MemberMethod), instance (inst) {}
439 template<
typename Ret,
typename ... Args >
440 struct MethodHelper :
public TrampolineBase {
441 typedef Ret (*Prototype)(Args ...);
444 MethodHelper (Prototype f) : TrampolineBase (StaticMethod), func (f) {}
446 template<
int ... Index >
447 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
448 (*
reinterpret_cast< typename removeRef< Ret >::type *
> (args[0])) =
449 func (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
452 void trampoline (
void **args)
453 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
457 template<
typename ... Args >
458 struct MethodHelper< void, Args ... > :
public TrampolineBase {
459 typedef void (*Prototype)(Args ...);
462 MethodHelper (Prototype f) : TrampolineBase (StaticMethod), func (f) {}
464 template<
int ... Index >
465 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
467 func (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
470 void trampoline (
void **args)
471 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
476 template<
typename Class,
typename Ret,
typename ... Args >
477 struct MemberMethodHelper :
public MemberTrampolineBase {
478 typedef Ret (Class::*Prototype)(Args ...);
481 MemberMethodHelper (Class *inst, Prototype f) : MemberTrampolineBase (inst), func (f) {}
483 template<
int ... Index >
484 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
485 (*
reinterpret_cast< typename removeRef< Ret >::type *
> (args[0])) =
486 (
reinterpret_cast< Class *
> (instance)->*func)
487 (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
490 void trampoline (
void **args)
491 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
495 template<
typename Class,
typename ... Args >
496 struct MemberMethodHelper< Class, void, Args ... > :
public MemberTrampolineBase {
497 typedef void (Class::*Prototype)(Args ...);
500 MemberMethodHelper (Class *inst, Prototype f) : MemberTrampolineBase (inst), func (f) {}
502 template<
int ... Index >
503 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
505 (
reinterpret_cast< Class *
> (instance)->*func)
506 (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
509 void trampoline (
void **args)
510 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
515 template<
typename FullType,
typename Ret,
typename ... Args >
516 struct LambdaHelper :
public TrampolineBase {
519 LambdaHelper (
const FullType &f) : TrampolineBase (Lambda), func (f) {}
521 template<
int ... Index >
522 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
523 (*
reinterpret_cast< typename removeRef< Ret >::type *
> (args[0])) =
524 func (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
527 void trampoline (
void **args)
528 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
532 template<
typename FullType,
typename ... Args >
533 struct LambdaHelper< FullType, void, Args ... > :
public TrampolineBase {
536 LambdaHelper (
const FullType &f) : TrampolineBase (Lambda), func (f) {}
538 template<
int ... Index >
539 inline void trampolineImpl (
void **args, CallbackHelper::IndexTuple< Index... >) {
541 func (*
reinterpret_cast< typename removeRef< Args >::type *
>(args[Index]) ...);
544 void trampoline (
void **args)
545 { trampolineImpl (args, CallbackHelper::buildIndexTuple< Args ... > ()); }
557 Q_DECLARE_METATYPE(
Nuria::Callback::Placeholder)
559 #endif // NURIA_CALLBACK_HPP
Placeholder
Definition: callback.hpp:164
void setVariadic(bool variadic)
A modern style callback mechanism which can be bound to various method types including slots...
Definition: callback.hpp:140
static Callback boundLambda(Lambda func, const Args &...args)
Definition: callback.hpp:325
Callback(Class *instance, Ret(Class::*func)(Args...) const, bool variadic=false)
Definition: callback.hpp:205
static QVariantList buildList(const Items &...items)
Definition: variant.hpp:42
static Callback fromLambda(Lambda l, bool variadic=false)
Definition: callback.hpp:214
Definition: abstractsessionmanager.hpp:24
Callback(Ret(*func)(Args...), bool variadic=false)
Definition: callback.hpp:195
Callback & bind(const Args &...args)
Definition: callback.hpp:337
Type
Definition: callback.hpp:153
Definition: callback.hpp:37
Definition: callback.hpp:43
Callback(Class *instance, Ret(Class::*func)(Args...), bool variadic=false)
Definition: callback.hpp:200