From 4d650160a388b0192e04c454ba1a3de21b0d4cc3 Mon Sep 17 00:00:00 2001 From: crupest Date: Thu, 4 Apr 2019 17:52:15 +0800 Subject: ... --- include/cru/common/event.hpp | 56 +++++++++++++++++++++++++++++++++++++------- 1 file changed, 47 insertions(+), 9 deletions(-) (limited to 'include/cru/common') diff --git a/include/cru/common/event.hpp b/include/cru/common/event.hpp index ce014fb8..52d75a7b 100644 --- a/include/cru/common/event.hpp +++ b/include/cru/common/event.hpp @@ -3,33 +3,65 @@ #include #include +#include #include namespace cru { +using EventHandlerRevoker = std::function; + // A non-copyable non-movable Event class. // It stores a list of event handlers. template class Event { + private: + using EventResolver = std::function; + class EventHandlerRevokerImpl { + public: + EventHandlerRevokerImpl(const std::shared_ptr& resolver, + long token) + : resolver_(resolver), token_(token) {} + EventHandlerRevokerImpl(const EventHandlerRevokerImpl& other) = default; + EventHandlerRevokerImpl(EventHandlerRevokerImpl&& other) = default; + EventHandlerRevokerImpl& operator=(EventHandlerRevokerImpl&& other) = + default; + EventHandlerRevokerImpl& operator=(EventHandlerRevokerImpl&& other) = + default; + ~EventHandlerRevokerImpl() = default; + + void operator()() { + const auto true_resolver = resolver_.lock(); + if (true_resolver) { + true_resolver()->RemoveHandler(token_); + } + } + + private: + std::weak_ptr resolver_; + long token_; + }; + public: using EventHandler = std::function; - using EventHandlerToken = long; - Event() = default; + Event() + : event_resolver_(new std::function([this] { return this; })) {} Event(const Event&) = delete; Event& operator=(const Event&) = delete; Event(Event&&) = delete; Event& operator=(Event&&) = delete; ~Event() = default; - EventHandlerToken AddHandler(const EventHandler& handler) { + EventHandlerRevoker AddHandler(const EventHandler& handler) { const auto token = current_token_++; handlers_.emplace(token, handler); - return token; + return EventHandlerRevoker(EventHandlerRevokerImpl(event_resolver_, token)); } - void RemoveHandler(const EventHandlerToken token) { - auto find_result = handlers_.find(token); - if (find_result != handlers_.cend()) handlers_.erase(find_result); + template + EventHandlerRevoker AddHandler(Args&& args...) { + const auto token = current_token_++; + handlers_.emplace(token, EventHandler(std::forward(args)...)); + return EventHandlerRevoker(EventHandlerRevokerImpl(event_resolver_, token)); } template @@ -39,8 +71,14 @@ class Event { } private: - std::map handlers_; + void RemoveHandler(const long token) { + auto find_result = handlers_.find(token); + if (find_result != handlers_.cend()) handlers_.erase(find_result); + } - EventHandlerToken current_token_ = 0; + private: + std::map handlers_{}; + long current_token_ = 0; + std::shared_ptr event_resolver_; }; } // namespace cru -- cgit v1.2.3