2015-10-07 18:11:14 +02:00
# EventBus
2017-08-13 14:37:19 +02:00
2017-08-15 20:35:56 +02:00
[![Join the chat at https://gitter.im/EventBusCpp/Lobby ](https://badges.gitter.im/EventBusCpp/Lobby.svg )](https://gitter.im/EventBusCpp/Lobby?utm_source=badge& utm_medium=badge& utm_campaign=pr-badge& utm_content=badge) [![Build Status ](https://travis-ci.org/gelldur/EventBus.svg?branch=master )](https://travis-ci.org/gelldur/EventBus)
2017-08-13 14:37:19 +02:00
2017-08-06 11:22:59 +02:00
Simple and very fast event bus.
The EventBus library is a convenient realization of the observer pattern.
It works perfectly to supplement the implementation of MVC logic (model-view-controller) in event-driven UIs
![EventBus Diagram ](EventBusDiagram.png )
EventBus was created because I want something easy to use and faster than [CCNotificationCenter ](https://github.com/cocos2d/cocos2d-x/blob/v2/cocos2dx/support/CCNotificationCenter.h )
from [cocos2d-x ](https://github.com/cocos2d/cocos2d-x ) library. Of course C++11 support was mandatory.
2017-08-06 18:16:41 +02:00
EventBus main goals:
2017-08-06 11:22:59 +02:00
- Fast
- Easy to use
2017-08-06 16:32:09 +02:00
- Strongly typed
2017-08-06 11:22:59 +02:00
- Free
2017-09-10 14:32:21 +02:00
- tiny (~15 KB)
2017-08-06 11:22:59 +02:00
- Decouples notification senders and receivers
2018-07-02 13:26:13 +02:00
- on every platform you need (crossplatform)
# Brief @ presentation
Presentation [google docs ](https://docs.google.com/presentation/d/1apAlKcVWo9FcqkPqL8108a1Fy9LGmhgLT56hSVpoI3w/edit?usp=sharing )
2017-08-06 11:22:59 +02:00
2017-09-20 20:42:01 +02:00
# Sample
You can checkout [sample/ ](sample/ )
If you want to play with sample online checkout this link: [wandbox.org ](https://wandbox.org/permlink/p3xeQaosuxv6w1rv )
2017-08-06 11:22:59 +02:00
# Usage
2017-09-10 14:32:21 +02:00
0. Store bus
2017-08-06 11:22:59 +02:00
```cpp
2018-07-02 13:26:13 +02:00
// store it in controller / singleton / std::sharted_ptr whatever you want
2017-08-06 11:22:59 +02:00
Dexode::EventBus bus;
```
2017-09-10 14:32:21 +02:00
1. Define events
2017-08-06 11:22:59 +02:00
```cpp
2017-09-10 14:32:21 +02:00
namespace Event // optional namespace
{
struct Gold
{
int goldReceived = 0;
};
struct OK {}; // Simple event when user press "OK" button
}
```
2. Subscribe
```cpp
// ...
bus.listen< Event::Gold >
([](const auto& event) // listen with lambda
{
std::cout < < "I received gold: " < < event.goldReceived < < " ! " < < std::endl ;
});
HudLayer* hudLayer;
// Hud layer will receive info about gold
bus.listen< Event::Gold > (std::bind(&HudLayer::onGoldReceived,hudLayer,std::placeholders::_1));
```
3. Notify
```cpp
//Inform listeners about event
bus.notify(Event::Gold{12}); // 1 way
bus.notify< Event::Gold > ({12}); // 2 way
Event::Gold myGold{12};
bus.notify(myGold); // 3 way
2017-08-06 11:22:59 +02:00
```
2017-08-06 16:32:09 +02:00
Lambda listener:
2017-08-06 11:22:59 +02:00
```cpp
2017-09-10 14:32:21 +02:00
struct SampleEvent {};
2017-08-06 11:22:59 +02:00
Dexode::EventBus bus;
//...
2017-09-10 14:32:21 +02:00
int token = bus.listen< SampleEvent > ([](const SampleEvent& event) // register listener
2017-08-06 11:22:59 +02:00
{
});
2017-09-10 14:32:21 +02:00
2017-08-06 11:22:59 +02:00
//If we want unlisten exact listener we can use token for it
bus.unlistenAll(token);
```
Listener is identified by `token` . Token is returned from EventBus::listen methods.
2017-08-06 16:32:09 +02:00
We can register multiple listeners on one token:
2017-08-06 11:22:59 +02:00
```cpp
Dexode::EventBus bus;
2017-09-10 14:32:21 +02:00
struct SampleEvent {};
2017-08-06 11:22:59 +02:00
//...
2017-09-10 14:32:21 +02:00
int token = bus.listen< SimpleEvent > ([](const auto& event) // register listener
2017-08-06 11:22:59 +02:00
{
});
2017-09-10 14:32:21 +02:00
bus.listen< SimpleEvent > (token, [](const auto& event) // another listener
2017-08-06 11:22:59 +02:00
{
});
bus.unlistenAll(token);//Now those two lambdas will be removed from listeners
```
2017-08-06 18:16:41 +02:00
If you don't want to handle manually with `token` you can use `EventCollector` class.
It is useful when we want to have multiple listeners in one class. So above example could look like this:
2017-08-06 11:22:59 +02:00
```cpp
Dexode::EventBus bus;
2017-09-10 14:32:21 +02:00
struct SampleEvent {};
2017-08-06 11:22:59 +02:00
Dexode::EventCollector collector{&bus};
//...
2017-09-10 14:32:21 +02:00
collector.listen< SampleEvent > ([](const SampleEvent& event) // register listener
2017-08-06 11:22:59 +02:00
{
});
2017-09-10 14:32:21 +02:00
collector.listen< SampleEvent > ([](const SampleEvent& event) // another listener
2017-08-06 11:22:59 +02:00
{
});
collector.unlistenAll();//Now those two lambdas will be removed from listeners
```
2017-08-06 16:32:09 +02:00
Or as component of class:
2017-08-06 11:22:59 +02:00
```cpp
class Example
{
public:
2017-09-10 14:32:21 +02:00
Example(const std::shared_ptr< Dexode::EventBus > & bus)
: _collector{bus}
2017-08-06 11:22:59 +02:00
{
2017-09-10 14:32:21 +02:00
_collector.listen< SimpleEvent > (std::bind(& Example::onEvent1, this, std::placeholders::_1));
_collector.listen< OtherEvent > (std::bind(& Example::onEvent2, this, std::placeholders::_1));
2017-08-06 11:22:59 +02:00
}
2017-09-10 14:32:21 +02:00
void onEvent1(const SimpleEvent& event)
2017-08-06 11:22:59 +02:00
{
}
2017-09-10 14:32:21 +02:00
void onEvent2(const OtherEvent& event)
2017-08-06 11:22:59 +02:00
{
}
private:
Dexode::EventCollector _collector;// use RAII
};
//EventCollector sample
2017-09-10 14:32:21 +02:00
std::shared_ptr< Dexode::EventBus > bus;
2017-08-06 11:22:59 +02:00
Example ex{bus};
//...
bus.notify< int > ("event1", 2);
```
# Add to your project
EventBus can be added as `ADD_SUBDIRECTORY` to your cmake file.
Then simply link it via `TARGET_LINK_LIBRARIES`
Example:
```
ADD_SUBDIRECTORY(lib/EventBus)
ADD_EXECUTABLE(MyExecutable
main.cpp
)
TARGET_LINK_LIBRARIES(MyExecutable PUBLIC Dexode::EventBus)
```
2017-08-15 21:50:49 +02:00
Also if you want you can install library and add it at any way you want.
Eg.
```commandline
2018-07-31 11:33:35 +02:00
mkdir -p lib/build/
cd lib/build
cmake -DCMAKE_BUILD_TYPE=Relase -DCMAKE_INSTALL_PREFIX=~/.local/ ..
cmake --build . --target install
# OR
2017-08-15 21:50:49 +02:00
make & & make install
```
2017-08-06 11:22:59 +02:00
2017-09-10 14:32:21 +02:00
Now in `Release/install` library is placed.
2017-08-06 11:22:59 +02:00
# Performance
I have prepared some performance results. You can read about them [here ](performance/README.md )
Small example:
```
2017-08-26 13:28:45 +02:00
check10NotificationsFor1kListeners 263 ns 263 ns 2668786 sum=-1.76281G
2017-08-06 11:22:59 +02:00
check10NotificationsFor1kListeners_CCNotificationCenter 11172 ns 11171 ns 62865 sum=54.023M
2017-08-26 13:28:45 +02:00
checkNotifyFor10kListenersWhenNoOneListens 18 ns 18 ns 38976599 sum=0
checkNotifyFor10kListenersWhenNoOneListens_CCNotificationCenter 127388 ns 127378 ns 5460 sum=0
2017-08-06 11:22:59 +02:00
```
2017-08-05 00:36:05 +02:00
2017-08-26 13:28:45 +02:00
# Future plans
- Write more and better tests
2017-09-10 14:32:21 +02:00
- Thread safe EventBus ?
- Verbose messages for easy debugging ?
- Generating graph flow ?
2018-07-31 11:33:35 +02:00
- Add nice documentation [like in POCO ](https://pocoproject.org/slides/090-NotificationsEvents.pdf )
2017-08-26 13:28:45 +02:00
- ...
2018-07-02 13:26:13 +02:00
# Issues ?
Please report here issue / question / whatever in 99% I will answer ;)
2017-08-05 00:36:05 +02:00
# Thanks to
2017-08-20 21:03:19 +02:00
- [staakk ](https://github.com/stanislawkabacinski ) for fixing windows ;) [53d5026 ](https://github.com/gelldur/EventBus/commit/53d5026cad24810e82cd8d4a43d58cbfe329c502 )
2017-08-05 02:36:12 +02:00
- [kuhar ](https://github.com/kuhar ) for his advice and suggestions for EventBus
2017-08-06 16:32:09 +02:00
- [swietlana ](https://github.com/swietlana ) for english correction and support ;)
2017-08-05 02:36:12 +02:00
- [ruslo ](https://github.com/ruslo ) for this great example: https://github.com/forexample/package-example
2017-08-06 11:22:59 +02:00
2018-07-31 11:33:35 +02:00
## For modern cmake refer
- https://github.com/forexample/package-example
- https://www.youtube.com/watch?v=6sWec7b0JIc
2017-08-06 11:22:59 +02:00
# License
2017-08-15 21:53:35 +02:00
EventBus source code can be used according to the **Apache License, Version 2.0** .
2017-08-06 11:22:59 +02:00
For more information see [LICENSE ](LICENSE ) file
2018-07-02 13:26:13 +02:00
If you don't like to read to much here is sumup about license [Apache 2.0 ](https://tldrlegal.com/license/apache-license-2.0-(apache-2.0 )#summary )