/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the Qt Mobility Components. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qsignalintercepter_p.h" #include #include #include #include /*! \class QSignalIntercepter \inpublicgroup QtBaseModule \internal \brief The QSignalIntercepter class provides an interface for intercepting signals as meta-calls \since 1.1 IPC mechanisms need to intercept signals and convert them into protocol messages, but it is generally impractical to create a slot for every signal that needs to be dispatched. The QSignalIntercepter class allows signals to be intercepted as meta-calls so that IPC dispatching can be implemented in a generic fashion. The activated() method is called whenever the signal is emitted, with the arguments in a typed list. \sa QSlotInvoker \ingroup objectmodel */ QTM_BEGIN_NAMESPACE class QSignalIntercepterPrivate { public: QObject *sender; QByteArray signal; int signalIndex; int destroyIndex; int slotIndex; int *types; int numArgs; ~QSignalIntercepterPrivate() { if ( types ) qFree( types ); } }; /*! Create a new signal intercepter which traps \a signal on \a sender. The object will be attached to \a parent, if present. \since 1.1 */ QSignalIntercepter::QSignalIntercepter ( QObject *sender, const QByteArray& signal, QObject *parent ) : QObject( parent ) { // Initialize the private members. d = new QSignalIntercepterPrivate(); d->sender = sender; d->signal = signal; d->signalIndex = -1; d->destroyIndex = -1; d->slotIndex = -1; d->types = 0; // Resolve the indices of the signals we are interested in. if ( sender && signal.size() > 0 ) { // '2' is QSIGNAL_CODE in Qt 4.4 and below, // '6' is QSIGNAL_CODE in Qt 4.5 and higher. if ( signal[0] != '2' && signal[0] != '6' ) { qWarning( "QSignalIntercepter: `%s' is not a valid signal " "specification", signal.constData() ); return; } QByteArray name = QMetaObject::normalizedSignature ( signal.constData() + 1 ); d->signalIndex = sender->metaObject()->indexOfSignal( name.constData() ); if ( d->signalIndex < 0 ) { qWarning( "QSignalIntercepter: no such signal: %s::%s", sender->metaObject()->className(), signal.constData() ); return; } d->destroyIndex = sender->metaObject()->indexOfSignal( "destroyed()" ); d->types = connectionTypes( name, d->numArgs ); } // Derive a fake slot index to use in our manual qt_metacall implementation. d->slotIndex = staticMetaObject.methodCount(); // Connect up the signals. if ( d->signalIndex >= 0 ) { QMetaObject::connect( sender, d->signalIndex, this, d->slotIndex, Qt::DirectConnection, 0 ); } if ( d->destroyIndex >= 0 ) { QMetaObject::connect( sender, d->destroyIndex, this, d->slotIndex + 1, Qt::DirectConnection, 0 ); } } /*! Destroy a signal intercepter. */ QSignalIntercepter::~QSignalIntercepter() { if ( d->signalIndex >= 0 ) { QMetaObject::disconnect( d->sender, d->signalIndex, this, d->slotIndex ); } if ( d->destroyIndex >= 0 ) { QMetaObject::disconnect( d->sender, d->destroyIndex, this, d->slotIndex + 1 ); } delete d; } /*! Returns the sender that this signal interceptor is attached to. \since 1.1 */ QObject *QSignalIntercepter::sender() const { return d->sender; } /*! Returns the name of the signal that this signal interceptor is attached to. \since 1.1 */ QByteArray QSignalIntercepter::signal() const { return d->signal; } /*! Returns true if this signal intercepter is valid; that is, there was a signal present with the specified parameters when this object was constructed. \since 1.1 */ bool QSignalIntercepter::isValid() const { return ( d->signalIndex != -1 ); } /*! \internal \since 1.1 */ int QSignalIntercepter::qt_metacall(QMetaObject::Call c, int id, void **a) { id = QObject::qt_metacall(c, id, a); if (id < 0) return id; if (c == QMetaObject::InvokeMetaMethod) { switch (id) { case 0: { // The signal we are interested in has been activated. if ( d->types ) { QList args; for ( int i = 0; i < d->numArgs; ++i ) { if ( d->types[i] != QVariantId ) { QVariant arg( d->types[i], a[i + 1] ); args.append( arg ); } else { args.append( *((const QVariant *)( a[i + 1] )) ); } } activated( args ); } } break; case 1: { // The sender has been destroyed. Clear the signal indices // so that we don't try to do a manual disconnect when our // own destructor is called. d->signalIndex = -1; d->destroyIndex = -1; } break; } id -= 2; } return id; } /*! \fn void QSignalIntercepter::activated( const QList& args ) Called when the signal that is being intercepted is activated. The arguments to the signal are passed in the list \a args. \since 1.1 */ // Get the QVariant type number for a type name. int QSignalIntercepter::typeFromName( const QByteArray& type ) { int id; if (type.endsWith('*')) return QMetaType::VoidStar; else if ( type.size() == 0 || type == "void" ) return QMetaType::Void; else if ( type == "QVariant" ) return QSignalIntercepter::QVariantId; id = QMetaType::type( type.constData() ); if ( id != (int)QMetaType::Void ) return id; return QVariant::nameToType(type); } /*! Returns the connection types associated with a signal or slot \a member specification. The array of types is returned from this function, and the number of arguments is returned in \a nargs. Returns null if \a member is invalid. The return value must be freed with qFree(). \since 1.1 */ int *QSignalIntercepter::connectionTypes( const QByteArray& member, int& nargs ) { // Based on Qt's internal queuedConnectionTypes function. nargs = 0; int *types = 0; const char *s = member.constData(); while (*s != '\0' && *s != '(') { ++s; } if ( *s == '\0' ) return 0; ++s; const char *e = s; while (*e != ')') { ++e; if (*e == ')' || *e == ',') ++nargs; } types = (int *) qMalloc((nargs+1)*sizeof(int)); types[nargs] = 0; for (int n = 0; n < nargs; ++n) { e = s; while (*s != ',' && *s != ')') ++s; QByteArray type(e, s-e); ++s; types[n] = typeFromName(type); if (!types[n]) { qWarning("QSignalIntercepter::connectionTypes: Cannot marshal arguments of type '%s'", type.data()); qFree(types); return 0; } } return types; } QTM_END_NAMESPACE