1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
|
#ifndef LIBFILEZILLA_EVENT_HEADER
#define LIBFILEZILLA_EVENT_HEADER
#include "libfilezilla.hpp"
#include <tuple>
#include <typeinfo>
/** \file
* \brief Declares event_base and simple_event<>
*/
namespace fz {
/**
\brief Common base class for all events.
If possible, use simple_event<> below instead of deriving from event_base directly.
Keep events as simple as possible. Avoid mutexes in your events.
*/
class FZ_PUBLIC_SYMBOL event_base
{
public:
event_base() = default;
virtual ~event_base() = default;
event_base(event_base const&) = default;
event_base& operator=(event_base const&) = default;
event_base(event_base &&) = default;
event_base& operator=(event_base &&) = default;
/**
The returned pointer must be unique for the derived type such that:
event_base& a = ...
event_base& b = ...
assert((a.derived_type() == b.derived_type()) == (typeid(a) == typeid(b)));
\warning Using &typeid is tempting, but unspecifined (sic)
\warning According to the C++ standard, the address of a static member function is unique
for each type. Unfortunately this does not prevent optimizing compilers to pool identical
functions.
Best solution is to have your derived type return the address of a static data member of it, as
done in \ref fz::simple_event "simple_event".
*/
virtual size_t derived_type() const = 0;
};
/**
* Maps a type info to a unique identifier even across DLL boundaries
*/
size_t FZ_PUBLIC_SYMBOL get_unique_type_id(std::type_info const& id);
/**
\brief This is the recommended event class.
Instantiate the template with a unique type to identify the type of the event and a number of types for the values.
Keep the values simple, in particular avoid mutexes in your values.
\see event_handler for usage example.
*/
template<typename UniqueType, typename...Values>
class simple_event final : public event_base
{
public:
typedef UniqueType unique_type;
typedef std::tuple<Values...> tuple_type;
using event_base::event_base;
template<typename First_Value, typename...Remaining_Values>
explicit simple_event(First_Value&& value, Remaining_Values&& ...values)
: v_(std::forward<First_Value>(value), std::forward<Remaining_Values>(values)...)
{
}
/// \brief Returns a unique id for the type such that can be used directly in derived_type.
inline static size_t type() {
// Exporting templates from DLLs is problematic to say the least. It breaks
// ODR, so we use this trick that goes over the type name.
static size_t const v = get_unique_type_id(typeid(UniqueType*));
return v;
}
/// \brief Simply returns \ref type()
virtual size_t derived_type() const override {
return type();
}
/** \brief The event value, gets built from the arguments passed in the constructor.
*
* You don't need to access this member directly if you use the \ref dispatch mechanism.
*/
mutable tuple_type v_;
};
class event_source{};
/// \private
class event_with_source_base : public event_base
{
public:
using event_base::event_base;
virtual event_source* source() const = 0;
};
/**
\brief Events with associated source
*
* This is similar to simple_event, except that the type of the first event value
* is derived from event_source.
*
* Pending events from a given source can easily be removed
* using \ref event_handler::remove_events
*/
template<typename UniqueType, typename...Values>
class event_with_source final : public event_with_source_base
{
public:
typedef UniqueType unique_type;
typedef std::tuple<Values...> tuple_type;
using event_with_source_base::event_with_source_base;
template<typename First_Value, typename...Remaining_Values>
explicit event_with_source(First_Value&& value, Remaining_Values&& ...values)
: v_(std::forward<First_Value>(value), std::forward<Remaining_Values>(values)...)
{
}
inline static size_t type() {
static size_t const v = get_unique_type_id(typeid(UniqueType*));
return v;
}
virtual size_t derived_type() const override {
return type();
}
virtual event_source* source() const override {
return static_cast<event_source*>(std::get<0>(v_));
}
mutable tuple_type v_;
};
/// Used as lightweight RTTI alternative during \ref dispatch
/// \return true iff T& t = ...; t.derived_type() == ev.derived_type()
template<typename T>
bool same_type(event_base const& ev)
{
return ev.derived_type() == T::type();
}
typedef unsigned long long timer_id;
/// \private
struct timer_event_type{};
/** \brief All timer events have this type.
*
* All timer events have one arguments of type \c timer_id which is the id of the timer that triggered.
*/
typedef simple_event<timer_event_type, timer_id> timer_event;
}
#endif
|