summaryrefslogtreecommitdiffstats
path: root/src/Event.cpp
blob: ff1617871d82b028b3db66a544987e5650ff51af (plain) (blame)
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
#include "Event.hpp"
#include <easylogging++.h>

std::queue <Event> EventAgregator::eventsToHandle;
std::mutex EventAgregator::queueMutex;
bool EventAgregator::isStarted = false;
std::vector<EventListener*> EventAgregator::listeners;
std::mutex EventAgregator::listenersMutex;

EventListener::EventListener() {
    EventAgregator::RegisterListener(*this);
}

EventListener::~EventListener() {
    EventAgregator::UnregisterListener(*this);
}

void EventListener::PushEvent(Event event) {
    eventsMutex.lock();
    events.push(event);
    eventsMutex.unlock();
}

bool EventListener::IsEventsQueueIsNotEmpty() {
    eventsMutex.lock();
    bool value = !events.empty();
    eventsMutex.unlock();
    return value;
}


void EventListener::RegisterHandler(EventType type, EventListener::HandlerFunc handler) {
	handlers[type] = handler;
}

void EventListener::HandleEvent() {
    eventsMutex.lock();
    if (events.empty()) {
        eventsMutex.unlock();
        return;
    }
    Event event = events.front();
    events.pop();
    eventsMutex.unlock();
    auto function = handlers[event.type];
    function(event.data);
}



void EventAgregator::RegisterListener(EventListener &listener) {
    listenersMutex.lock();
    LOG(WARNING)<<"Registered handler "<<&listener;
    listeners.push_back(&listener);
    listenersMutex.unlock();
}

void EventAgregator::UnregisterListener(EventListener &listener) {
    listenersMutex.lock();
    LOG(WARNING)<<"Unregistered handler "<<&listener;
    listeners.erase(std::find(listeners.begin(), listeners.end(), &listener));
    listenersMutex.unlock();
}

void EventAgregator::PushEvent(EventType type, EventData data) {
    Event event;
    event.type = type;
    event.data = data;
    eventsToHandle.push(event);
    if (!isStarted) {
        isStarted = true;
        std::thread(&EventAgregator::EventHandlingLoop).detach();
    }
}

void EventAgregator::EventHandlingLoop() {
    while (true) {
        queueMutex.lock();
        if (!eventsToHandle.empty()) {
            auto queue = eventsToHandle;
            while (!eventsToHandle.empty())
                eventsToHandle.pop();
            queueMutex.unlock();

            while (!queue.empty()) {
                auto event = queue.front();
                listenersMutex.lock();
                for (auto& listener : listeners) {
                    LOG(ERROR)<<"Listener notified about event";
                    listener->PushEvent(event);
                }
                listenersMutex.unlock();
                queue.pop();
            }

            queueMutex.lock();
        }
        queueMutex.unlock();
    }
}