qskinny/src/common/QskMetaFunction.h

175 lines
4.6 KiB
C
Raw Normal View History

/******************************************************************************
* QSkinny - Copyright (C) 2016 Uwe Rathmann
* This file may be used under the terms of the QSkinny License, Version 1.0
*****************************************************************************/
#ifndef QSK_META_FUNCTION_H
#define QSK_META_FUNCTION_H 1
#include "QskGlobal.h"
2018-02-28 09:43:15 +00:00
#include "QskMetaInvokable.h"
#include <QMetaType>
2018-02-28 09:43:15 +00:00
namespace QskMetaFunctionTraits
{
using namespace QtPrivate;
template< typename T >
using IsMemberFunction = typename std::enable_if< FunctionPointer< T >::IsPointerToMemberFunction,
std::true_type >::type;
template< typename T >
using IsFunction = typename std::enable_if< !FunctionPointer< T >::IsPointerToMemberFunction
&& FunctionPointer< T >::ArgumentCount >= 0, std::true_type >::type;
template< typename T >
using IsFunctor = typename std::enable_if< !FunctionPointer< T >::IsPointerToMemberFunction
&& FunctionPointer< T >::ArgumentCount == -1, std::true_type >::type;
}
class QSK_EXPORT QskMetaFunction
{
Q_GADGET
public:
enum Type
{
Invalid = -1,
// a non static method of class
Member,
// a static function, or static method of a class
Function,
// a functor or lambda
Functor
};
Q_ENUM( Type )
QskMetaFunction();
QskMetaFunction( const QskMetaFunction& );
QskMetaFunction( QskMetaFunction&& );
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsMemberFunction< T >* = nullptr >
QskMetaFunction( T );
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsFunction< T >* = nullptr >
QskMetaFunction( T );
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsFunctor< T >* = nullptr >
QskMetaFunction( T );
~QskMetaFunction();
QskMetaFunction& operator=( const QskMetaFunction& );
QskMetaFunction& operator=( QskMetaFunction&& );
const int* parameterTypes() const;
// including the return type !
size_t parameterCount() const;
void invoke( QObject*, void* args[],
Qt::ConnectionType = Qt::AutoConnection );
Type functionType() const;
2018-02-28 15:48:46 +00:00
protected:
friend class QskMetaCallback;
QskMetaFunction( QskMetaInvokable*, const int* );
2018-02-28 09:43:15 +00:00
void init( QskMetaInvokable*, const int* );
2018-02-28 15:48:46 +00:00
QskMetaInvokable* invokable() const;
private:
template< typename T, typename F >
static inline QskMetaInvokable* findInvokable( F function )
{
QskMetaInvokable* invokable;
#if QSK_SHARED_META_INVOKABLE
invokable = QskMetaInvokable::find( typeid( T ) );
if ( invokable )
{
invokable->ref();
}
else
{
invokable = new T( function );
QskMetaInvokable::insert( typeid( T ), invokable );
}
#else
invokable = new T( function );
#endif
return invokable;
}
2018-02-28 09:43:15 +00:00
QskMetaInvokable* m_invokable;
const int* m_parameterTypes;
};
2018-02-28 15:48:46 +00:00
inline QskMetaInvokable* QskMetaFunction::invokable() const
{
return m_invokable;
}
inline const int* QskMetaFunction::parameterTypes() const
{
return m_parameterTypes;
}
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsMemberFunction< T >* >
inline QskMetaFunction::QskMetaFunction( T function )
{
using namespace QtPrivate;
using Traits = FunctionPointer< T >;
const int Argc = Traits::ArgumentCount;
using Args = typename List_Left< typename Traits::Arguments, Argc >::Value;
2018-02-28 15:48:46 +00:00
using Invokable = QskMetaMemberInvokable< T, Args, void >;
init( findInvokable<Invokable>( function ),
ConnectionTypes< typename Traits::Arguments >::types() );
}
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsFunction< T >* >
inline QskMetaFunction::QskMetaFunction( T function )
{
using namespace QtPrivate;
using Traits = FunctionPointer< T >;
const int Argc = Traits::ArgumentCount;
using Args = typename List_Left< typename Traits::Arguments, Argc >::Value;
2018-02-28 15:48:46 +00:00
using Invokable = QskMetaFunctionInvokable< T, Args, void >;
2018-02-28 15:48:46 +00:00
init( findInvokable<Invokable>( function ),
ConnectionTypes< typename Traits::Arguments >::types() );
}
2018-02-28 09:43:15 +00:00
template< typename T, QskMetaFunctionTraits::IsFunctor< T >* >
inline QskMetaFunction::QskMetaFunction( T functor )
{
using namespace QtPrivate;
using Traits = FunctionPointer< decltype( &T::operator() ) >;
const int Argc = Traits::ArgumentCount;
using Args = typename List_Left< typename Traits::Arguments, Argc >::Value;
2018-02-28 15:48:46 +00:00
using Invokable = QskMetaFunctorInvokable< T, Argc, Args, void >;
2018-02-28 15:48:46 +00:00
init( findInvokable<Invokable>( functor ),
ConnectionTypes< typename Traits::Arguments >::types() );
}
Q_DECLARE_METATYPE( QskMetaFunction )
#endif