255 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
		
			7.7 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/* Copyright 2013-2015 Jan Dalheimer
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
#pragma once
 | 
						|
 | 
						|
#include <QObject>
 | 
						|
#include <QMetaMethod>
 | 
						|
#include <QThread>
 | 
						|
#include <QCoreApplication>
 | 
						|
#include <QSemaphore>
 | 
						|
#include <memory>
 | 
						|
 | 
						|
#if QT_VERSION == QT_VERSION_CHECK(5, 1, 1)
 | 
						|
#include <5.1.1/QtCore/private/qobject_p.h>
 | 
						|
#elif QT_VERSION == QT_VERSION_CHECK(5, 2, 0)
 | 
						|
#include <5.2.0/QtCore/private/qobject_p.h>
 | 
						|
#elif QT_VERSION == QT_VERSION_CHECK(5, 2, 1)
 | 
						|
#include <5.2.1/QtCore/private/qobject_p.h>
 | 
						|
#elif QT_VERSION == QT_VERSION_CHECK(5, 3, 0)
 | 
						|
#include <5.3.0/QtCore/private/qobject_p.h>
 | 
						|
#elif QT_VERSION == QT_VERSION_CHECK(5, 3, 1)
 | 
						|
#include <5.3.1/QtCore/private/qobject_p.h>
 | 
						|
#else
 | 
						|
#error Please add support for this version of Qt
 | 
						|
#endif
 | 
						|
 | 
						|
class Bindable
 | 
						|
{
 | 
						|
	friend class tst_LogicalGui;
 | 
						|
 | 
						|
public:
 | 
						|
	Bindable(Bindable *parent = 0) : m_parent(parent)
 | 
						|
	{
 | 
						|
	}
 | 
						|
	virtual ~Bindable()
 | 
						|
	{
 | 
						|
	}
 | 
						|
 | 
						|
	void setBindableParent(Bindable *parent)
 | 
						|
	{
 | 
						|
		m_parent = parent;
 | 
						|
	}
 | 
						|
 | 
						|
	void bind(const QString &id, const QObject *receiver, const char *methodSignature)
 | 
						|
	{
 | 
						|
		auto mo = receiver->metaObject();
 | 
						|
		Q_ASSERT_X(mo, "Bindable::bind",
 | 
						|
				   "Invalid metaobject. Did you forget the QObject macro?");
 | 
						|
		const QMetaMethod method = mo->method(mo->indexOfMethod(
 | 
						|
			QMetaObject::normalizedSignature(methodSignature + 1).constData()));
 | 
						|
		Q_ASSERT_X(method.isValid(), "Bindable::bind", "Invalid method signature");
 | 
						|
		m_bindings.insert(id, Binding(receiver, method));
 | 
						|
	}
 | 
						|
	template <typename Func>
 | 
						|
	void bind(const QString &id,
 | 
						|
			  const typename QtPrivate::FunctionPointer<Func>::Object *receiver, Func slot)
 | 
						|
	{
 | 
						|
		typedef QtPrivate::FunctionPointer<Func> SlotType;
 | 
						|
		m_bindings.insert(
 | 
						|
			id,
 | 
						|
			Binding(receiver, new QtPrivate::QSlotObject<Func, typename SlotType::Arguments,
 | 
						|
														 typename SlotType::ReturnType>(slot)));
 | 
						|
	}
 | 
						|
	template <typename Func> void bind(const QString &id, Func slot)
 | 
						|
	{
 | 
						|
		typedef QtPrivate::FunctionPointer<Func> SlotType;
 | 
						|
		m_bindings.insert(
 | 
						|
			id,
 | 
						|
			Binding(nullptr, new QtPrivate::QSlotObject<Func, typename SlotType::Arguments,
 | 
						|
														typename SlotType::ReturnType>(slot)));
 | 
						|
	}
 | 
						|
	void unbind(const QString &id)
 | 
						|
	{
 | 
						|
		m_bindings.remove(id);
 | 
						|
	}
 | 
						|
 | 
						|
private:
 | 
						|
	struct Binding
 | 
						|
	{
 | 
						|
		Binding(const QObject *receiver, const QMetaMethod &method)
 | 
						|
			: receiver(receiver), method(method)
 | 
						|
		{
 | 
						|
		}
 | 
						|
		Binding(const QObject *receiver, QtPrivate::QSlotObjectBase *object)
 | 
						|
			: receiver(receiver), object(object)
 | 
						|
		{
 | 
						|
		}
 | 
						|
		Binding()
 | 
						|
		{
 | 
						|
		}
 | 
						|
		const QObject *receiver;
 | 
						|
		QMetaMethod method;
 | 
						|
		QtPrivate::QSlotObjectBase *object = nullptr;
 | 
						|
	};
 | 
						|
	QMap<QString, Binding> m_bindings;
 | 
						|
 | 
						|
	Bindable *m_parent;
 | 
						|
 | 
						|
private:
 | 
						|
	inline Qt::ConnectionType connectionType(const QObject *receiver)
 | 
						|
	{
 | 
						|
		return receiver == nullptr ? Qt::DirectConnection
 | 
						|
								   : (QThread::currentThread() == receiver->thread()
 | 
						|
										  ? Qt::DirectConnection
 | 
						|
										  : Qt::BlockingQueuedConnection);
 | 
						|
	}
 | 
						|
 | 
						|
protected:
 | 
						|
	template <typename Ret, typename... Params> Ret wait(const QString &id, Params... params)
 | 
						|
	{
 | 
						|
		static_assert(!std::is_same<Ret, void>::value, "You need to use Bindable::waitVoid");
 | 
						|
 | 
						|
		if (!m_bindings.contains(id) && m_parent)
 | 
						|
		{
 | 
						|
			return m_parent->wait<Ret, Params...>(id, params...);
 | 
						|
		}
 | 
						|
		Q_ASSERT(m_bindings.contains(id));
 | 
						|
		const auto binding = m_bindings[id];
 | 
						|
		const Qt::ConnectionType type = connectionType(binding.receiver);
 | 
						|
		Ret ret;
 | 
						|
		if (binding.object)
 | 
						|
		{
 | 
						|
			void *args[] = {&ret,
 | 
						|
							const_cast<void *>(reinterpret_cast<const void *>(¶ms))...};
 | 
						|
			if (type == Qt::BlockingQueuedConnection)
 | 
						|
			{
 | 
						|
				QSemaphore semaphore;
 | 
						|
				QMetaCallEvent *ev =
 | 
						|
					new QMetaCallEvent(binding.object, nullptr, -1, 0, 0, args, &semaphore);
 | 
						|
				QCoreApplication::postEvent(const_cast<QObject *>(binding.receiver), ev);
 | 
						|
				semaphore.acquire();
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				binding.object->call(const_cast<QObject *>(binding.receiver), args);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			const QMetaMethod method = binding.method;
 | 
						|
			Q_ASSERT_X(method.parameterCount() == sizeof...(params), "Bindable::wait",
 | 
						|
					   qPrintable(QString("Incompatible argument count (expected %1, got %2)")
 | 
						|
									  .arg(method.parameterCount(), sizeof...(params))));
 | 
						|
			Q_ASSERT_X(
 | 
						|
				qMetaTypeId<Ret>() != QMetaType::UnknownType, "Bindable::wait",
 | 
						|
				"Requested return type is not registered, please use the Q_DECLARE_METATYPE "
 | 
						|
				"macro to make it known to Qt's meta-object system");
 | 
						|
			Q_ASSERT_X(
 | 
						|
				method.returnType() == qMetaTypeId<Ret>() ||
 | 
						|
					QMetaType::hasRegisteredConverterFunction(method.returnType(),
 | 
						|
															  qMetaTypeId<Ret>()),
 | 
						|
				"Bindable::wait",
 | 
						|
				qPrintable(
 | 
						|
					QString(
 | 
						|
						"Requested return type (%1) is incompatible method return type (%2)")
 | 
						|
						.arg(QMetaType::typeName(qMetaTypeId<Ret>()),
 | 
						|
							 QMetaType::typeName(method.returnType()))));
 | 
						|
			const auto retArg = QReturnArgument<Ret>(
 | 
						|
				QMetaType::typeName(qMetaTypeId<Ret>()),
 | 
						|
				ret); // because Q_RETURN_ARG doesn't work with templates...
 | 
						|
			method.invoke(const_cast<QObject *>(binding.receiver), type, retArg,
 | 
						|
						  Q_ARG(Params, params)...);
 | 
						|
		}
 | 
						|
		return ret;
 | 
						|
	}
 | 
						|
	template <typename... Params>
 | 
						|
	typename std::enable_if<sizeof...(Params) != 0, void>::type waitVoid(const QString &id,
 | 
						|
																		 Params... params)
 | 
						|
	{
 | 
						|
		if (!m_bindings.contains(id) && m_parent)
 | 
						|
		{
 | 
						|
			m_parent->waitVoid<Params...>(id, params...);
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		Q_ASSERT(m_bindings.contains(id));
 | 
						|
		const auto binding = m_bindings[id];
 | 
						|
		const Qt::ConnectionType type = connectionType(binding.receiver);
 | 
						|
		if (binding.object)
 | 
						|
		{
 | 
						|
			void *args[] = {0, const_cast<void *>(reinterpret_cast<const void *>(¶ms))...};
 | 
						|
			if (type == Qt::BlockingQueuedConnection)
 | 
						|
			{
 | 
						|
				QSemaphore semaphore;
 | 
						|
				QMetaCallEvent *ev =
 | 
						|
					new QMetaCallEvent(binding.object, nullptr, -1, 0, 0, args, &semaphore);
 | 
						|
				QCoreApplication::postEvent(const_cast<QObject *>(binding.receiver), ev);
 | 
						|
				semaphore.acquire();
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				binding.object->call(const_cast<QObject *>(binding.receiver), args);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			const QMetaMethod method = binding.method;
 | 
						|
			Q_ASSERT_X(method.parameterCount() == sizeof...(params), "Bindable::wait",
 | 
						|
					   qPrintable(QString("Incompatible argument count (expected %1, got %2)")
 | 
						|
									  .arg(method.parameterCount(), sizeof...(params))));
 | 
						|
			method.invoke(const_cast<QObject *>(binding.receiver), type,
 | 
						|
						  Q_ARG(Params, params)...);
 | 
						|
		}
 | 
						|
	}
 | 
						|
	void waitVoid(const QString &id)
 | 
						|
	{
 | 
						|
		if (!m_bindings.contains(id) && m_parent)
 | 
						|
		{
 | 
						|
			m_parent->waitVoid(id);
 | 
						|
			return;
 | 
						|
		}
 | 
						|
		Q_ASSERT(m_bindings.contains(id));
 | 
						|
		const auto binding = m_bindings[id];
 | 
						|
		const Qt::ConnectionType type = connectionType(binding.receiver);
 | 
						|
		if (binding.object)
 | 
						|
		{
 | 
						|
			void *args[] = {0};
 | 
						|
			if (type == Qt::BlockingQueuedConnection)
 | 
						|
			{
 | 
						|
				QSemaphore semaphore;
 | 
						|
				QMetaCallEvent *ev =
 | 
						|
					new QMetaCallEvent(binding.object, nullptr, -1, 0, 0, args, &semaphore);
 | 
						|
				QCoreApplication::postEvent(const_cast<QObject *>(binding.receiver), ev);
 | 
						|
				semaphore.acquire();
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				binding.object->call(const_cast<QObject *>(binding.receiver), args);
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			const QMetaMethod method = binding.method;
 | 
						|
			Q_ASSERT_X(method.parameterCount() == 0, "Bindable::wait",
 | 
						|
					   qPrintable(QString("Incompatible argument count (expected %1, got %2)")
 | 
						|
									  .arg(method.parameterCount(), 0)));
 | 
						|
			method.invoke(const_cast<QObject *>(binding.receiver), type);
 | 
						|
		}
 | 
						|
	}
 | 
						|
};
 | 
						|
 | 
						|
// used frequently
 | 
						|
Q_DECLARE_METATYPE(bool *)
 |