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
|
// SPDX-License-Identifier: LGPL-3.0-or-later
// Author: Kristian Lytje
#pragma once
#include <utility/observer_ptr.h>
#include <utility/Observer.h>
#include <list>
#include <memory>
namespace ausaxs::utility {
namespace detail {
template<typename T> concept observer_type = requires(T t, const T& value) {
{ t.notify(value) } -> std::same_as<void>;
};
}
template<typename T>
class Observable {
public:
Observable() = default;
~Observable() {
for (auto& o : observers) {
o->on_delete();
o->on_delete = [] () {}; // clear the on_delete callback to avoid dangling pointer to this
}
}
/**
* @brief Notify all observers of a new value.
*/
void notify(const T& value) {
for (auto& o : observers) {
o->notify(value);
}
}
/**
* @brief Create a new observer and attach it to this observable.
*
* When the returned observer is destroyed, it will detach itself from this observable.
*/
std::unique_ptr<Observer<T>> make_observer() {
auto o = std::make_unique<Observer<T>>();
o->on_delete = [this, ptr=o.get()] () {
this->detach_observer(ptr);
};
this->attach_observer(o.get());
return o;
}
private:
void attach_observer(observer_ptr<Observer<T>> observer) {
observers.push_back(observer);
}
void detach_observer(observer_ptr<Observer<T>> observer) {
observers.remove(observer);
}
std::list<observer_ptr<Observer<T>>> observers;
};
}
|