6#include "QskMetaInvokable.h"
7#include "QskMetaFunction.h"
8#include "QskInternalMacros.h"
10#include <qmetaobject.h>
12#include <qcoreapplication.h>
16#include <qsemaphore.h>
20#include <private/qobject_p.h>
23static void qskRegisterMetaInvokable()
25 qRegisterMetaType< QskMetaInvokable >();
28Q_CONSTRUCTOR_FUNCTION( qskRegisterMetaInvokable )
32 using CallFunction = QObjectPrivate::StaticMetaCallFunction;
37 inline Function(
void* functionCall )
42 static inline void ref(
void* functionCall )
45 static_cast< FunctionCall*
>( functionCall )->ref();
48 static inline void deref(
void* functionCall )
51 static_cast< FunctionCall*
>( functionCall )->destroyIfLastRef();
55 class MetaCallEvent final :
public QMetaCallEvent
58 MetaCallEvent( QMetaObject::Call call,
const QMetaObject* metaObject,
59 ushort offset, ushort index,
void* args[], QSemaphore* semaphore )
60 : QMetaCallEvent( offset, index,
61 metaObject->d.static_metacall, nullptr, -1, args, semaphore )
63 , m_callFunction( metaObject->d.static_metacall )
68 MetaCallEvent( QMetaObject::Call call,
const QMetaObject* metaObject,
69 ushort offset, ushort index,
int argc )
70 : QMetaCallEvent( offset, index,
71 metaObject->d.static_metacall, nullptr, -1, argc )
73 , m_callFunction( metaObject->d.static_metacall )
78 void placeMetaCall( QObject*
object )
override
80 m_callFunction(
object, m_call, m_index, args() );
84 const QMetaObject::Call m_call;
87 CallFunction m_callFunction;
92QMetaMethod qskMetaMethod(
const QObject*
object,
const char* methodName )
94 return object ? qskMetaMethod( object->metaObject(), methodName ) : QMetaMethod();
97QMetaMethod qskMetaMethod(
const QMetaObject* metaObject,
const char* methodName )
99 if ( metaObject ==
nullptr || methodName ==
nullptr )
100 return QMetaMethod();
102 constexpr char signalIndicator =
'0' + QSIGNAL_CODE;
103 constexpr char slotIndicator =
'0' + QSLOT_CODE;
107 if ( methodName[ 0 ] == signalIndicator )
109 auto signature = QMetaObject::normalizedSignature( methodName + 1 );
110 index = metaObject->indexOfSignal( signature );
112 else if ( methodName[ 0 ] == slotIndicator )
114 auto signature = QMetaObject::normalizedSignature( methodName + 1 );
115 index = metaObject->indexOfSlot( signature );
119 auto signature = QMetaObject::normalizedSignature( methodName );
120 index = metaObject->indexOfMethod( signature );
123 return ( index >= 0 ) ? metaObject->method( index ) : QMetaMethod();
126QMetaMethod qskNotifySignal(
const QObject*
object,
const char* propertyName )
128 return object ? qskNotifySignal( object->metaObject(), propertyName ) : QMetaMethod();
131QMetaMethod qskNotifySignal(
const QMetaObject* metaObject,
const char* propertyName )
133 if ( metaObject ==
nullptr || propertyName ==
nullptr )
134 return QMetaMethod();
136 const int propertyIndex = metaObject->indexOfProperty( propertyName );
139 const auto property = metaObject->property( propertyIndex );
140 return property.notifySignal();
143 return QMetaMethod();
146static void qskInvokeMetaCall(
147 QObject*
object,
const QMetaObject* metaObject,
148 QMetaObject::Call call, ushort offset, ushort index,
void* args[],
149 Qt::ConnectionType connectionType )
151 QPointer< QObject > receiver(
object );
153 int invokeType = connectionType & 0x3;
155 if ( invokeType == Qt::AutoConnection )
157 invokeType = (
object &&
object->thread() != QThread::currentThread() )
158 ? Qt::QueuedConnection : Qt::DirectConnection;
161 switch ( invokeType )
163 case Qt::DirectConnection:
165 if ( receiver.isNull() )
178 metaObject->d.static_metacall( receiver, call, index, args );
181 case Qt::BlockingQueuedConnection:
183 if ( receiver.isNull() ||
184 ( receiver->thread() == QThread::currentThread() ) )
191 QSemaphore semaphore;
193 auto event =
new MetaCallEvent( call, metaObject,
194 offset, index, args, &semaphore );
197 auto event =
new MetaCallEvent( call, metaObject,
198 offset, index, args,
nullptr );
201 QCoreApplication::postEvent( receiver, event );
209 case Qt::QueuedConnection:
211 if ( receiver ==
nullptr )
214 MetaCallEvent*
event =
nullptr;
216 if ( call == QMetaObject::InvokeMetaMethod )
220 const auto method = metaObject->method( offset + index );
222 const int argc = method.parameterCount() + 1;
224 event =
new MetaCallEvent( call, metaObject, offset, index, argc );
231 auto types =
event->types();
232 auto arguments =
event->args();
234#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
235 types[0] = QMetaType();
236 arguments[ 0 ] =
nullptr;
239 arguments[ 0 ] =
nullptr;
241 for (
int i = 1; i < argc; i++ )
243 if ( args[ i ] ==
nullptr )
245 Q_ASSERT( args[ i ] !=
nullptr );
250 const auto type = method.parameterType( i - 1 );
251 const auto arg = args[ i ];
253#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
254 types[ i ] = QMetaType( type );
255 arguments[ i ] = QMetaType( type ).create( arg );
258 arguments[ i ] = QMetaType::create( type, arg );
265 const auto property = metaObject->property( offset + index );
267 event =
new MetaCallEvent( call, metaObject, offset, index, 1 );
269 const auto type =
property.userType();
270 const auto arg = args[ 0 ];
272#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
273 event->types()[0] = QMetaType( type );
274 event->args()[ 0 ] = QMetaType( type ).create( arg );
276 event->types()[0] = type;
277 event->args()[ 0 ] = QMetaType::create( type, arg );
282 QCoreApplication::postEvent( receiver, event );
291void qskInvokeMetaPropertyWrite( QObject* context,
const QMetaProperty& property,
292 void* args[], Qt::ConnectionType connectionType )
294 qskInvokeMetaPropertyWrite( context, property.enclosingMetaObject(),
295 property.propertyIndex(), args, connectionType );
298void qskInvokeMetaPropertyWrite( QObject* context,
const QMetaObject* metaObject,
299 int propertyIndex,
void* args[], Qt::ConnectionType connectionType )
303 if ( metaObject && ( propertyIndex >= 0 ) &&
304 ( propertyIndex < metaObject->propertyCount() ) )
306 const auto offset = metaObject->propertyOffset();
307 const auto index = propertyIndex - offset;
309 qskInvokeMetaCall( context, metaObject, QMetaObject::WriteProperty,
310 offset, index, args + 1, connectionType );
314void qskInvokeMetaMethod( QObject*
object,
315 const QMetaMethod& method,
void* args[],
316 Qt::ConnectionType connectionType )
318 qskInvokeMetaMethod(
object, method.enclosingMetaObject(),
319 method.methodIndex(), args, connectionType );
322void qskInvokeMetaMethod( QObject*
object,
323 const QMetaObject* metaObject,
int methodIndex,
void* args[],
324 Qt::ConnectionType connectionType )
326 if ( metaObject && ( methodIndex >= 0 ) &&
327 ( methodIndex < metaObject->methodCount() ) )
329 const auto offset = metaObject->methodOffset();
330 const auto index = methodIndex - offset;
332 qskInvokeMetaCall(
object, metaObject, QMetaObject::InvokeMetaMethod,
333 offset, index, args, connectionType );
337QskMetaInvokable::QskMetaInvokable(
const QMetaMethod& method )
338 : m_metaData{ method.enclosingMetaObject(), method.methodIndex() }
339 , m_type( MetaMethod )
343QskMetaInvokable::QskMetaInvokable(
const QObject*
object,
const char* methodName )
348QskMetaInvokable::QskMetaInvokable(
const QMetaObject* metaObject,
const char* methodName )
353QskMetaInvokable::QskMetaInvokable(
const QMetaProperty& property )
354 : m_metaData{ property.enclosingMetaObject(), property.propertyIndex() }
355 , m_type( MetaProperty )
360 : m_functionData{ function.functionCall() }
361 , m_type( MetaFunction )
363 Function::ref( m_functionData.functionCall );
367 : m_type( other.m_type )
374 m_metaData.metaObject = other.m_metaData.metaObject;
375 m_metaData.index = other.m_metaData.index;
381 m_functionData.functionCall = other.m_functionData.functionCall;
382 Function::ref( m_functionData.functionCall );
392QskMetaInvokable::~QskMetaInvokable()
394 if ( m_type == MetaFunction )
395 Function::deref( m_functionData.functionCall );
400 switch ( other.m_type )
405 if ( m_type == MetaFunction )
406 Function::deref( m_functionData.functionCall );
408 m_metaData.metaObject = other.m_metaData.metaObject;
409 m_metaData.index = other.m_metaData.index;
415 if ( m_type == MetaFunction )
416 Function::deref( m_functionData.functionCall );
418 m_functionData.functionCall = other.m_functionData.functionCall;
419 Function::ref( m_functionData.functionCall );
425 if ( m_type == MetaFunction )
426 Function::deref( m_functionData.functionCall );
429 m_type = other.m_type;
436 if ( m_type != other.m_type )
444 return ( m_metaData.metaObject == other.m_metaData.metaObject ) &&
445 ( m_metaData.index == other.m_metaData.index );
449 return m_functionData.functionCall == other.m_functionData.functionCall;
458bool QskMetaInvokable::isNull()
const
465 const auto& d = m_metaData;
467 if ( d.metaObject ==
nullptr || d.index < 0 )
470 const int count = ( m_type == MetaMethod )
471 ? d.metaObject->methodCount() : d.metaObject->propertyCount();
473 return d.index >= count;
477 return m_functionData.functionCall ==
nullptr;
485void QskMetaInvokable::reset()
487 if ( m_type == MetaFunction )
488 Function::deref( m_functionData.functionCall );
493int QskMetaInvokable::parameterCount()
const
500 const auto method = QskMetaInvokable::method();
501 return method.parameterCount();
509 return function().parameterCount();
518int QskMetaInvokable::parameterType(
int index )
const
524 const auto method = QskMetaInvokable::method();
525 return method.parameterType( index );
529 const auto property = QskMetaInvokable::property();
530 return property.userType();
534 auto types = function().parameterTypes();
535 return types[ index ];
539 return QMetaType::UnknownType;
544int QskMetaInvokable::returnType()
const
550 return method().returnType();
554 return function().returnType();
558 return QMetaType::Void;
562 return QMetaType::Void;
567QByteArray QskMetaInvokable::name()
const
573 return method().name();
577 return property().name();
591QMetaMethod QskMetaInvokable::method()
const
593 if ( m_type == MetaMethod && m_metaData.metaObject )
594 return m_metaData.metaObject->method( m_metaData.index );
596 return QMetaMethod();
599QMetaProperty QskMetaInvokable::property()
const
601 if ( m_type == MetaProperty && m_metaData.metaObject )
602 return m_metaData.metaObject->property( m_metaData.index );
604 return QMetaProperty();
609 if ( m_type == MetaFunction && m_functionData.functionCall )
611 Function function( m_functionData.functionCall );
618void QskMetaInvokable::invoke( QObject*
object,
void* args[],
619 Qt::ConnectionType connectionType )
628 qskInvokeMetaMethod(
object, m_metaData.metaObject,
629 m_metaData.index, args, connectionType );
635 qskInvokeMetaPropertyWrite(
636 object, m_metaData.metaObject, m_metaData.index, args, connectionType );
642 if ( m_functionData.functionCall )
644 Function function( m_functionData.functionCall );
645 function.invoke(
object, args, connectionType );