2017-08-06 00:14:35 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <algorithm>
|
2017-08-26 13:28:45 +02:00
|
|
|
#include <cassert>
|
2017-08-06 00:14:35 +02:00
|
|
|
#include <functional>
|
|
|
|
#include <map>
|
|
|
|
#include <memory>
|
2017-08-26 13:28:45 +02:00
|
|
|
#include <string>
|
|
|
|
#include <typeinfo>
|
2017-08-06 00:14:35 +02:00
|
|
|
#include <vector>
|
|
|
|
|
2017-08-06 17:14:14 +02:00
|
|
|
#if __cplusplus < 201103L
|
|
|
|
#error This library needs at least a C++11 compliant compiler
|
2017-08-06 00:14:35 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
namespace Dexode
|
|
|
|
{
|
|
|
|
|
|
|
|
class EventBus
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
EventBus() = default;
|
|
|
|
|
|
|
|
~EventBus()
|
|
|
|
{
|
|
|
|
_callbacks.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
EventBus(const EventBus&) = delete;
|
|
|
|
EventBus(EventBus&&) = delete;
|
|
|
|
|
|
|
|
EventBus& operator=(EventBus&&) = delete;
|
|
|
|
EventBus& operator=(const EventBus&) = delete;
|
|
|
|
|
|
|
|
/**
|
2017-08-06 17:24:13 +02:00
|
|
|
* Register listener for event. Returns token used for unlisten.
|
2017-08-06 00:14:35 +02:00
|
|
|
*
|
2017-08-26 13:28:45 +02:00
|
|
|
* @tparam Event - type you want to listen for
|
2017-08-06 17:24:13 +02:00
|
|
|
* @param callback - your callback to handle event
|
2017-08-06 00:14:35 +02:00
|
|
|
* @return token used for unlisten
|
|
|
|
*/
|
2017-08-26 13:28:45 +02:00
|
|
|
template<typename Event>
|
|
|
|
int listen(const std::function<void(const Event&)>& callback)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
|
|
|
const int token = ++_tokener;
|
2017-08-26 13:28:45 +02:00
|
|
|
listen<Event>(token, callback);
|
2017-08-06 00:14:35 +02:00
|
|
|
return token;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-26 13:28:45 +02:00
|
|
|
* @tparam Event - type you want to listen for
|
2017-08-06 00:14:35 +02:00
|
|
|
* @param token - unique token for identification receiver. Simply pass token from @see EventBus::listen
|
2017-08-06 17:24:13 +02:00
|
|
|
* @param callback - your callback to handle event
|
2017-08-06 00:14:35 +02:00
|
|
|
*/
|
2017-08-26 13:28:45 +02:00
|
|
|
template<typename Event>
|
|
|
|
void listen(const int token, const std::function<void(const Event&)>& callback)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-26 13:28:45 +02:00
|
|
|
using Vector = VectorImpl<Event>;
|
2017-08-06 00:14:35 +02:00
|
|
|
|
2017-08-06 17:24:13 +02:00
|
|
|
assert(callback && "callback should be valid");//Check for valid object
|
2017-08-06 00:14:35 +02:00
|
|
|
|
2017-08-26 13:28:45 +02:00
|
|
|
std::unique_ptr<VectorInterface>& vector = _callbacks[getTypeId<Event>()];
|
2017-08-06 00:14:35 +02:00
|
|
|
if (vector == nullptr)
|
|
|
|
{
|
|
|
|
vector.reset(new Vector{});
|
|
|
|
}
|
|
|
|
assert(dynamic_cast<Vector*>(vector.get()));
|
|
|
|
Vector* vectorImpl = static_cast<Vector*>(vector.get());
|
2017-08-27 22:59:40 +02:00
|
|
|
vectorImpl->add(token, callback);
|
2017-08-06 00:14:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param token - token from EventBus::listen
|
|
|
|
*/
|
|
|
|
void unlistenAll(const int token)
|
|
|
|
{
|
|
|
|
for (auto& element : _callbacks)
|
|
|
|
{
|
|
|
|
element.second->remove(token);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-26 13:28:45 +02:00
|
|
|
* @tparam Event - type you want to unlisten. @see Notiier::listen
|
2017-08-06 00:14:35 +02:00
|
|
|
* @param token - token from EventBus::listen
|
|
|
|
*/
|
2017-08-26 13:28:45 +02:00
|
|
|
template<typename Event>
|
|
|
|
void unlisten(const int token)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-26 13:28:45 +02:00
|
|
|
auto found = _callbacks.find(getTypeId<Event>());
|
2017-08-06 00:14:35 +02:00
|
|
|
if (found != _callbacks.end())
|
|
|
|
{
|
2017-08-26 13:28:45 +02:00
|
|
|
found->second->remove(token);
|
2017-08-06 00:14:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-06 17:24:13 +02:00
|
|
|
/**
|
2017-08-26 13:28:45 +02:00
|
|
|
* Notify all listeners for event
|
2017-08-06 17:24:13 +02:00
|
|
|
*
|
2017-08-26 13:28:45 +02:00
|
|
|
* @param event your event struct
|
2017-08-06 17:24:13 +02:00
|
|
|
*/
|
2017-08-26 13:28:45 +02:00
|
|
|
template<typename Event>
|
|
|
|
void notify(const Event& event)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-26 13:28:45 +02:00
|
|
|
using Vector = VectorImpl<Event>;
|
2017-08-27 22:59:40 +02:00
|
|
|
const auto typeId = getTypeId<Event>();//TODO think about constexpr
|
2017-08-26 13:28:45 +02:00
|
|
|
auto found = _callbacks.find(typeId);
|
2017-08-06 00:14:35 +02:00
|
|
|
if (found == _callbacks.end())
|
|
|
|
{
|
|
|
|
return;// no such notifications
|
|
|
|
}
|
|
|
|
|
|
|
|
std::unique_ptr<VectorInterface>& vector = found->second;
|
|
|
|
assert(dynamic_cast<Vector*>(vector.get()));
|
|
|
|
Vector* vectorImpl = static_cast<Vector*>(vector.get());
|
|
|
|
|
2017-08-27 22:59:40 +02:00
|
|
|
vectorImpl->beginTransaction();
|
2017-08-26 13:28:45 +02:00
|
|
|
for (const auto& element : vectorImpl->container)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-27 22:59:40 +02:00
|
|
|
element.second(event);
|
2017-08-06 00:14:35 +02:00
|
|
|
}
|
2017-08-27 22:59:40 +02:00
|
|
|
vectorImpl->commitTransaction();
|
2017-08-06 00:14:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct VectorInterface
|
|
|
|
{
|
2017-08-15 20:26:38 +02:00
|
|
|
virtual ~VectorInterface() = default;
|
2017-08-06 00:14:35 +02:00
|
|
|
|
|
|
|
virtual void remove(const int token) = 0;
|
|
|
|
};
|
|
|
|
|
2017-08-26 13:28:45 +02:00
|
|
|
template<typename Event>
|
2017-08-06 00:14:35 +02:00
|
|
|
struct VectorImpl : public VectorInterface
|
|
|
|
{
|
2017-08-26 13:28:45 +02:00
|
|
|
using CallbackType = std::function<void(const Event&)>;
|
2017-08-27 22:59:40 +02:00
|
|
|
using ContainerElement = std::pair<int, CallbackType>;
|
|
|
|
using ContainerType = std::vector<ContainerElement>;
|
|
|
|
ContainerType container;
|
|
|
|
ContainerType toAdd;
|
|
|
|
std::vector<int> toRemove;
|
|
|
|
bool inTransaction = false;
|
2017-08-06 00:14:35 +02:00
|
|
|
|
|
|
|
virtual void remove(const int token) override
|
|
|
|
{
|
2017-08-27 22:59:40 +02:00
|
|
|
if (inTransaction)
|
|
|
|
{
|
|
|
|
toRemove.push_back(token);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Invalidation rules: https://stackoverflow.com/questions/6438086/iterator-invalidation-rules
|
2017-08-06 00:14:35 +02:00
|
|
|
auto removeFrom = std::remove_if(container.begin(), container.end()
|
2017-08-27 22:59:40 +02:00
|
|
|
, [token](const ContainerElement& element)
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-27 22:59:40 +02:00
|
|
|
return element.first == token;
|
2017-08-06 00:14:35 +02:00
|
|
|
});
|
2017-08-06 17:30:06 +02:00
|
|
|
if (removeFrom != container.end())
|
2017-08-06 00:14:35 +02:00
|
|
|
{
|
2017-08-06 17:30:06 +02:00
|
|
|
container.erase(removeFrom, container.end());
|
2017-08-06 00:14:35 +02:00
|
|
|
}
|
|
|
|
}
|
2017-08-27 22:59:40 +02:00
|
|
|
|
|
|
|
void add(const int token, const CallbackType& callback)
|
|
|
|
{
|
|
|
|
if (inTransaction)
|
|
|
|
{
|
|
|
|
toAdd.emplace_back(token, callback);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
container.emplace_back(token, callback);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void beginTransaction()
|
|
|
|
{
|
|
|
|
inTransaction = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void commitTransaction()
|
|
|
|
{
|
|
|
|
inTransaction = false;
|
|
|
|
|
|
|
|
if (toAdd.empty() == false)
|
|
|
|
{
|
|
|
|
container.insert(container.end(), toAdd.begin(), toAdd.end());
|
|
|
|
toAdd.clear();
|
|
|
|
}
|
|
|
|
if (toRemove.empty() == false)
|
|
|
|
{
|
|
|
|
for (auto token : toRemove)
|
|
|
|
{
|
|
|
|
remove(token);
|
|
|
|
}
|
|
|
|
toRemove.clear();
|
|
|
|
}
|
|
|
|
}
|
2017-08-06 00:14:35 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
int _tokener = 0;
|
|
|
|
std::map<std::size_t, std::unique_ptr<VectorInterface>> _callbacks;
|
2017-08-26 13:28:45 +02:00
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
static const std::size_t getTypeId()
|
|
|
|
{
|
|
|
|
//std::hash<std::string>{}(typeid(T).name() is slower
|
|
|
|
return typeid(T).hash_code();
|
|
|
|
}
|
2017-08-06 00:14:35 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
} /* namespace Dexode */
|