// // FunctionDelegate.h // // Library: Foundation // Package: Events // Module: FunctionDelegate // // Implementation of the FunctionDelegate template. // // Copyright (c) 2006-2025, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #ifndef Foundation_FunctionDelegate_INCLUDED #define Foundation_FunctionDelegate_INCLUDED #include "Poco/Foundation.h" #include "Poco/AbstractDelegate.h" #include "Poco/Mutex.h" #include namespace Poco { namespace Detail { /// Helper to select the correct function pointer type based on template parameters. template struct FunctionDelegateNotifyType; template struct FunctionDelegateNotifyType { using Type = void (*)(const void*, TArgs&); }; template struct FunctionDelegateNotifyType { using Type = void (*)(void*, TArgs&); }; template struct FunctionDelegateNotifyType { using Type = void (*)(TArgs&); }; template <> struct FunctionDelegateNotifyType { using Type = void (*)(const void*); }; template <> struct FunctionDelegateNotifyType { using Type = void (*)(void*); }; template struct FunctionDelegateNotifyType { using Type = void (*)(); }; } // namespace Detail template class FunctionDelegate: public AbstractDelegate /// Wraps a freestanding function or static member function /// for use as a Delegate. { public: using NotifyFunction = typename Detail::FunctionDelegateNotifyType::Type; FunctionDelegate(NotifyFunction function): _function(function) { } FunctionDelegate(const FunctionDelegate& delegate): AbstractDelegate(delegate), _function(delegate._function) { } ~FunctionDelegate() = default; FunctionDelegate() = delete; FunctionDelegate& operator = (const FunctionDelegate& delegate) { if (&delegate != this) { this->_function = delegate._function; } return *this; } bool notify(const void* sender, TArgs& arguments) { Mutex::ScopedLock lock(_mutex); if (_function) { if constexpr (hasSender) { if constexpr (senderIsConst) { (*_function)(sender, arguments); } else { (*_function)(const_cast(sender), arguments); } } else { (*_function)(arguments); } return true; } else return false; } bool equals(const AbstractDelegate& other) const { const FunctionDelegate* pOtherDelegate = dynamic_cast(other.unwrap()); return pOtherDelegate && _function == pOtherDelegate->_function; } AbstractDelegate* clone() const { return new FunctionDelegate(*this); } void disable() { Mutex::ScopedLock lock(_mutex); _function = nullptr; } protected: NotifyFunction _function; Mutex _mutex; }; template class FunctionDelegate: public AbstractDelegate /// Specialization for void arguments (no TArgs parameter). { public: using NotifyFunction = typename Detail::FunctionDelegateNotifyType::Type; FunctionDelegate(NotifyFunction function): _function(function) { } FunctionDelegate(const FunctionDelegate& delegate): AbstractDelegate(delegate), _function(delegate._function) { } ~FunctionDelegate() override = default; FunctionDelegate() = delete; FunctionDelegate& operator = (const FunctionDelegate& delegate) { if (&delegate != this) { this->_function = delegate._function; } return *this; } bool notify(const void* sender) override { Mutex::ScopedLock lock(_mutex); if (_function) { if constexpr (hasSender) { if constexpr (senderIsConst) { (*_function)(sender); } else { (*_function)(const_cast(sender)); } } else { (*_function)(); } return true; } else return false; } bool equals(const AbstractDelegate& other) const override { const FunctionDelegate* pOtherDelegate = dynamic_cast(other.unwrap()); return pOtherDelegate && _function == pOtherDelegate->_function; } AbstractDelegate* clone() const override { return new FunctionDelegate(*this); } void disable() override { Mutex::ScopedLock lock(_mutex); _function = nullptr; } protected: NotifyFunction _function; Mutex _mutex; }; } // namespace Poco #endif // Foundation_FunctionDelegate_INCLUDED