/* This file is part of Clementine. Copyright 2011, David Sansome Clementine is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. Clementine is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with Clementine. If not, see . */ #ifndef CLOSURE_H #define CLOSURE_H #include #include #include #include #include #include #include "core/logging.h" namespace _detail { class ClosureArgumentWrapper { public: virtual ~ClosureArgumentWrapper() {} virtual QGenericArgument arg() const = 0; }; template class ClosureArgument : public ClosureArgumentWrapper { public: explicit ClosureArgument(const T& data) : data_(data) {} virtual QGenericArgument arg() const { return Q_ARG(T, data_); } private: T data_; }; class Closure : public QObject, boost::noncopyable { Q_OBJECT public: Closure(QObject* sender, const char* signal, QObject* receiver, const char* slot, const ClosureArgumentWrapper* val0 = 0, const ClosureArgumentWrapper* val1 = 0, const ClosureArgumentWrapper* val2 = 0, const ClosureArgumentWrapper* val3 = 0); Closure(QObject* sender, const char* signal, std::tr1::function callback); virtual ~Closure(); private slots: void Invoked(); void Cleanup(); private: void Connect(QObject* sender, const char* signal); QMetaMethod slot_; std::tr1::function callback_; boost::scoped_ptr val0_; boost::scoped_ptr val1_; boost::scoped_ptr val2_; boost::scoped_ptr val3_; }; class SharedPointerWrapper { public: virtual ~SharedPointerWrapper() {} virtual QObject* data() const = 0; }; template class SharedPointer : public SharedPointerWrapper { public: explicit SharedPointer(QSharedPointer ptr) : ptr_(ptr) { } QObject* data() const { return ptr_.data(); } private: QSharedPointer ptr_; }; // For use with a QSharedPointer as a sender. class SharedClosure : public Closure { Q_OBJECT public: SharedClosure(SharedPointerWrapper* sender, const char* signal, QObject* receiver, const char* slot, const ClosureArgumentWrapper* val0 = 0, const ClosureArgumentWrapper* val1 = 0, const ClosureArgumentWrapper* val2 = 0, const ClosureArgumentWrapper* val3 = 0) : Closure(sender->data(), signal, receiver, slot, val0, val1, val2, val3), shared_sender_(sender) { } private: boost::scoped_ptr shared_sender_; }; } // namespace _detail #define C_ARG(type, data) new _detail::ClosureArgument(data) _detail::Closure* NewClosure( QObject* sender, const char* signal, QObject* receiver, const char* slot); template _detail::Closure* NewClosure( QObject* sender, const char* signal, QObject* receiver, const char* slot, const T& val0) { return new _detail::Closure( sender, signal, receiver, slot, C_ARG(T, val0)); } template _detail::Closure* NewClosure( QObject* sender, const char* signal, QObject* receiver, const char* slot, const T0& val0, const T1& val1) { return new _detail::Closure( sender, signal, receiver, slot, C_ARG(T0, val0), C_ARG(T1, val1)); } template _detail::Closure* NewClosure( QObject* sender, const char* signal, QObject* receiver, const char* slot, const T0& val0, const T1& val1, const T2& val2) { return new _detail::Closure( sender, signal, receiver, slot, C_ARG(T0, val0), C_ARG(T1, val1), C_ARG(T2, val2)); } template _detail::Closure* NewClosure( QObject* sender, const char* signal, QObject* receiver, const char* slot, const T0& val0, const T1& val1, const T2& val2, const T3& val3) { return new _detail::Closure( sender, signal, receiver, slot, C_ARG(T0, val0), C_ARG(T1, val1), C_ARG(T2, val2), C_ARG(T3, val3)); } template _detail::Closure* NewClosure( QSharedPointer sender, const char* signal, QObject* receiver, const char* slot) { return new _detail::SharedClosure( new _detail::SharedPointer(sender), signal, receiver, slot); } template _detail::Closure* NewClosure( QSharedPointer sender, const char* signal, QObject* receiver, const char* slot, const T0& val0) { return new _detail::SharedClosure( new _detail::SharedPointer(sender), signal, receiver, slot, C_ARG(T0, val0)); } template _detail::Closure* NewClosure( QSharedPointer sender, const char* signal, QObject* receiver, const char* slot, const T0& val0, const T1& val1) { return new _detail::SharedClosure( new _detail::SharedPointer(sender), signal, receiver, slot, C_ARG(T0, val0), C_ARG(T1, val1)); } #endif // CLOSURE_H