/* Copyright (C) 2013 * swift project Community / Contributors * * This file is part of swift project. It is subject to the license terms in the LICENSE file found in the top-level * directory of this distribution. No part of swift project, including this file, may be copied, modified, propagated, * or distributed except according to the terms contained in the LICENSE file. */ //! \file #ifndef BLACKMISC_GENERICDBUSINTERFACE_H #define BLACKMISC_GENERICDBUSINTERFACE_H #include "blackmisc/logmessage.h" #include "blackmisc/promise.h" #include #include #include #include #include #include #include #ifndef Q_MOC_RUN /*! * Any signals tagged with this macro will be ignored by BlackMisc::CGenericDBusInterface::relayParentSignals(). * \see QMetaMethod::tag */ #define BLACK_NO_RELAY #endif namespace BlackMisc { /*! * Used for hand written interface based on virtual methods. * Allows to relay a message to DBus in a single code line */ class BLACKMISC_EXPORT CGenericDBusInterface : public QDBusAbstractInterface { Q_OBJECT public: //! Constructor CGenericDBusInterface(const QString &serviceName, const QString &path, const QString &interfaceName, const QDBusConnection &connection, QObject *parent = nullptr) : QDBusAbstractInterface(serviceName, path, interfaceName.toUtf8().constData(), connection, parent) { } //! For each signal in parent, attempt to connect to it an interface signal of the same name. //! \see BLACK_NO_RELAY void relayParentSignals() { const QMetaObject *metaObject = this->parent()->metaObject(); const QMetaObject *superMetaObject = metaObject; while (strcmp(superMetaObject->superClass()->className(), "QObject") != 0) { superMetaObject = superMetaObject->superClass(); } for (int i = superMetaObject->methodOffset(), count = metaObject->methodCount(); i < count; ++i) { const QMetaMethod method = metaObject->method(i); if (method.methodType() != QMetaMethod::Signal) { continue; } if (method.tag() && strcmp(method.tag(), "BLACK_NO_RELAY") == 0) { continue; } const QByteArray signature = method.methodSignature().prepend("2"); // the reason for this "2" can be found in the definition of SIGNAL() macro const bool c = this->connection().connect(this->service(), this->path(), this->interface(), method.name(), this->parent(), signature); if (!c) { CLogMessage(this).error(u"Cannot connect signal: %1") << QString(signature); } } } //! Call DBus, no return value template void callDBus(QLatin1String method, Args &&... args) { const QList argumentList { QVariant::fromValue(std::forward(args))... }; this->callWithArgumentList(QDBus::NoBlock, method, argumentList); } //! Call DBus with synchronous return value template Ret callDBusRet(QLatin1String method, Args &&... args) { QList argumentList { QVariant::fromValue(std::forward(args))... }; QDBusPendingReply pr = this->asyncCallWithArgumentList(method, argumentList); pr.waitForFinished(); if(pr.isError()) { CLogMessage(this).debug(u"CGenericDBusInterface::callDBusRet(%1) returned: %2") << method << pr.error().message(); } return pr; } //! Call DBus with asynchronous return value //! Callback can be any callable object taking a single argument of type QDBusPendingCallWatcher*. template QDBusPendingCallWatcher *callDBusAsync(QLatin1String method, Func callback, Args &&... args) { QList argumentList { QVariant::fromValue(std::forward(args))... }; QDBusPendingCall pc = this->asyncCallWithArgumentList(method, argumentList); auto pcw = new QDBusPendingCallWatcher(pc, this); connect(pcw, &QDBusPendingCallWatcher::finished, callback); return pcw; } //! Call DBus with asynchronous return as a future template QFuture callDBusFuture(QLatin1String method, Args&&... args) { auto sharedPromise = QSharedPointer>::create(); this->callDBusAsync(method, [ = ](auto pcw) { sharedPromise->setResult(QDBusPendingReply(*pcw)); }, std::forward(args)...); return sharedPromise->future(); } //! Cancel all asynchronous DBus calls which are currently pending //! \warning Don't call this from inside an async callback! void cancelAllPendingAsyncCalls() { auto watchers = this->findChildren(QString(), Qt::FindDirectChildrenOnly); for (auto w : watchers) { delete w; } } }; } // ns #endif // guard