123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 |
- /****************************************************************************
- **
- ** Copyright (C) 2016 The Qt Company Ltd.
- ** Contact: https://www.qt.io/licensing/
- **
- ** This file is part of the QtTest module of the Qt Toolkit.
- **
- ** $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 The Qt Company. For licensing terms
- ** and conditions see https://www.qt.io/terms-conditions. For further
- ** information use the contact form at https://www.qt.io/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 3 as published by the Free Software
- ** Foundation and appearing in the file LICENSE.LGPL3 included in the
- ** packaging of this file. Please review the following information to
- ** ensure the GNU Lesser General Public License version 3 requirements
- ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
- **
- ** GNU General Public License Usage
- ** Alternatively, this file may be used under the terms of the GNU
- ** General Public License version 2.0 or (at your option) the GNU General
- ** Public license version 3 or any later version approved by the KDE Free
- ** Qt Foundation. The licenses are as published by the Free Software
- ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
- ** included in the packaging of this file. Please review the following
- ** information to ensure the GNU General Public License requirements will
- ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
- ** https://www.gnu.org/licenses/gpl-3.0.html.
- **
- ** $QT_END_LICENSE$
- **
- ****************************************************************************/
- #ifndef QSIGNALSPY_H
- #define QSIGNALSPY_H
- #include <QtCore/qbytearray.h>
- #include <QtCore/qlist.h>
- #include <QtCore/qobject.h>
- #include <QtCore/qmetaobject.h>
- #include <QtCore/qvariant.h>
- #include <QtCore/qvector.h>
- #include <QtTest/qtesteventloop.h>
- QT_BEGIN_NAMESPACE
- class QVariant;
- class QSignalSpy: public QObject, public QList<QList<QVariant> >
- {
- public:
- explicit QSignalSpy(const QObject *obj, const char *aSignal)
- : m_waiting(false)
- {
- #ifdef Q_CC_BOR
- const int memberOffset = QObject::staticMetaObject.methodCount();
- #else
- static const int memberOffset = QObject::staticMetaObject.methodCount();
- #endif
- if (!obj) {
- qWarning("QSignalSpy: Cannot spy on a null object");
- return;
- }
- if (!aSignal) {
- qWarning("QSignalSpy: Null signal name is not valid");
- return;
- }
- if (((aSignal[0] - '0') & 0x03) != QSIGNAL_CODE) {
- qWarning("QSignalSpy: Not a valid signal, use the SIGNAL macro");
- return;
- }
- const QByteArray ba = QMetaObject::normalizedSignature(aSignal + 1);
- const QMetaObject * const mo = obj->metaObject();
- const int sigIndex = mo->indexOfMethod(ba.constData());
- if (sigIndex < 0) {
- qWarning("QSignalSpy: No such signal: '%s'", ba.constData());
- return;
- }
- if (!QMetaObject::connect(obj, sigIndex, this, memberOffset,
- Qt::DirectConnection, Q_NULLPTR)) {
- qWarning("QSignalSpy: QMetaObject::connect returned false. Unable to connect.");
- return;
- }
- sig = ba;
- initArgs(mo->method(sigIndex), obj);
- }
- #ifdef Q_QDOC
- QSignalSpy(const QObject *object, PointerToMemberFunction signal);
- #else
- template <typename Func>
- QSignalSpy(const typename QtPrivate::FunctionPointer<Func>::Object *obj, Func signal0)
- : m_waiting(false)
- {
- #ifdef Q_CC_BOR
- const int memberOffset = QObject::staticMetaObject.methodCount();
- #else
- static const int memberOffset = QObject::staticMetaObject.methodCount();
- #endif
- if (!obj) {
- qWarning("QSignalSpy: Cannot spy on a null object");
- return;
- }
- if (!signal0) {
- qWarning("QSignalSpy: Null signal name is not valid");
- return;
- }
- const QMetaObject * const mo = obj->metaObject();
- const QMetaMethod signalMetaMethod = QMetaMethod::fromSignal(signal0);
- const int sigIndex = signalMetaMethod.methodIndex();
- if (!signalMetaMethod.isValid() ||
- signalMetaMethod.methodType() != QMetaMethod::Signal) {
- qWarning("QSignalSpy: Not a valid signal: '%s'",
- signalMetaMethod.methodSignature().constData());
- return;
- }
- if (!QMetaObject::connect(obj, sigIndex, this, memberOffset,
- Qt::DirectConnection, 0)) {
- qWarning("QSignalSpy: QMetaObject::connect returned false. Unable to connect.");
- return;
- }
- sig = signalMetaMethod.methodSignature();
- initArgs(mo->method(sigIndex), obj);
- }
- #endif // Q_QDOC
- inline bool isValid() const { return !sig.isEmpty(); }
- inline QByteArray signal() const { return sig; }
- bool wait(int timeout = 5000)
- {
- Q_ASSERT(!m_waiting);
- const int origCount = count();
- m_waiting = true;
- m_loop.enterLoopMSecs(timeout);
- m_waiting = false;
- return count() > origCount;
- }
- int qt_metacall(QMetaObject::Call call, int methodId, void **a) Q_DECL_OVERRIDE
- {
- methodId = QObject::qt_metacall(call, methodId, a);
- if (methodId < 0)
- return methodId;
- if (call == QMetaObject::InvokeMetaMethod) {
- if (methodId == 0) {
- appendArgs(a);
- }
- --methodId;
- }
- return methodId;
- }
- private:
- void initArgs(const QMetaMethod &member, const QObject *obj)
- {
- args.reserve(member.parameterCount());
- for (int i = 0; i < member.parameterCount(); ++i) {
- int tp = member.parameterType(i);
- if (tp == QMetaType::UnknownType && obj) {
- void *argv[] = { &tp, &i };
- QMetaObject::metacall(const_cast<QObject*>(obj),
- QMetaObject::RegisterMethodArgumentMetaType,
- member.methodIndex(), argv);
- if (tp == -1)
- tp = QMetaType::UnknownType;
- }
- if (tp == QMetaType::UnknownType) {
- qWarning("QSignalSpy: Unable to handle parameter '%s' of type '%s' of method '%s',"
- " use qRegisterMetaType to register it.",
- member.parameterNames().at(i).constData(),
- member.parameterTypes().at(i).constData(),
- member.name().constData());
- }
- args << tp;
- }
- }
- void appendArgs(void **a)
- {
- QList<QVariant> list;
- list.reserve(args.count());
- for (int i = 0; i < args.count(); ++i) {
- const QMetaType::Type type = static_cast<QMetaType::Type>(args.at(i));
- if (type == QMetaType::QVariant)
- list << *reinterpret_cast<QVariant *>(a[i + 1]);
- else
- list << QVariant(type, a[i + 1]);
- }
- append(list);
- if (m_waiting)
- m_loop.exitLoop();
- }
- // the full, normalized signal name
- QByteArray sig;
- // holds the QMetaType types for the argument list of the signal
- QVector<int> args;
- QTestEventLoop m_loop;
- bool m_waiting;
- };
- QT_END_NAMESPACE
- #endif
|