//This file is part of Bertini 2.
//
//events.hpp is free software: you can redistribute it and/or modify
//it under the terms of the GNU General Public License as published by
//the Free Software Foundation, either version 3 of the License, or
//(at your option) any later version.
//
//events.hpp is distributed in the hope that it will be useful,
//but WITHOUT ANY WARRANTY; without even the implied warranty of
//MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
//GNU General Public License for more details.
//
//You should have received a copy of the GNU General Public License
//along with events.hpp. If not, see .
//
// Copyright(C) 2015 - 2021 by Bertini2 Development Team
//
// See for a copy of the license,
// as well as COPYING. Bertini2 is provided with permitted
// additional terms in the b2/licenses/ directory.
// individual authors of this file include:
// silviana amethyst, university of wisconsin-eau claire
//
// detail/events.hpp
/**
\file detail/events.hpp
\brief Contains the detail/events base types
*/
#ifndef BERTINI_DETAIL_EVENTS_HPP
#define BERTINI_DETAIL_EVENTS_HPP
#include
namespace bertini {
/**
\brief Strawman Event type, enabling polymorphism.
This class is abstract, and should never actually be created.
*/
class AnyEvent
{ BOOST_TYPE_INDEX_REGISTER_CLASS
public:
virtual ~AnyEvent() = default;
};
template
class Event;
/**
\brief For emission of events from observables.
An observable object probably wants to emit events to notify observers that things are happening. The observers filter the events, at this time by dynamic casting (slow, I know, but it works for now. Premature optimization, etc.) A better method than dynamic casting would be to filter based on a union of bits or something. If you, the reader, want to help make this better, please contact a developer!
Say I am an observable object, and I want to emit an event. Events attach the type of object emitting them, and in fact (a refence to) the emitter itself. So if my type is `T`, I would do something like `NotifyObservers(Event(*this))`. Then an Observer can filter based on a heirarchy of event types, etc.
\tparam ObsT The Observed type. When emitting an event, you pass in the type of object emitting the event, and the object itself. Then the observer can `Get` the emitting object, and do (const) stuff to it.
\see Observable, Observable::NotifyObservers, ADD_BERTINI_EVENT_TYPE, AMPPathAccumulator, AnyEvent
*/
template
class Event : public AnyEvent
{ BOOST_TYPE_INDEX_REGISTER_CLASS
public:
/**
\brief Constructor for an event.
\param obs The observable emitting the event to observers. Passed by const reference, this permits arbitrary `const` function calls by the observer.
*/
Event(ObsT const& obs) : current_observable_(obs)
{}
virtual ~Event() = default;
/**
\brief Get the emitting object, by `const` reference.
\return The observable who emitted the event. This permits calls of arbitrary const functions, particularly getters.
\see AMPPathAccumulator for simple example of filtering using `dynamic_cast`s
*/
ObsT const& Get() const
{return current_observable_;}
Event() = delete;
using HeldT = const ObsT&;
protected:
const ObsT& current_observable_;
};
/**
\brief For emission of events from observables.
An observable object probably wants to emit events to notify observers that things are happening. The observers filter the events, at this time by dynamic casting (slow, I know, but it works for now. Premature optimization, etc.) A better method than dynamic casting would be to filter based on a union of bits or something. If you, the reader, want to help make this better, please contact a developer!
Say I am an observable object, and I want to emit an event. Events attach the type of object emitting them, and in fact (a refence to) the emitter itself. So if my type is `T`, I would do something like `NotifyObservers(Event(*this))`. Then an Observer can filter based on a heirarchy of event types, etc.
\tparam ObsT The Observed type. When emitting an event, you pass in the type of object emitting the event, and the object itself. Then the observer can `Get` the emitting object, and do (const) stuff to it.
\see Observable, Observable::NotifyObservers, ADD_BERTINI_EVENT_TYPE, AMPPathAccumulator, AnyEvent
*/
template
class Event : public AnyEvent
{ BOOST_TYPE_INDEX_REGISTER_CLASS
public:
/**
\brief Constructor for an event.
\param obs The observable emitting the event to observers. Passed by const reference, this permits arbitrary `const` function calls by the observer.
*/
Event(ObsT & obs) : current_observable_(obs)
{}
virtual ~Event() = default;
/**
\brief Get the emitting object, by `const` reference.
\return The observable who emitted the event. This permits calls of arbitrary const functions, particularly getters.
\see AMPPathAccumulator for simple example of filtering using `dynamic_cast`s
*/
ObsT & Get()
{return current_observable_;}
Event() = delete;
using HeldT = ObsT&;
protected:
ObsT& current_observable_;
};
template
using ConstEvent = Event;
template
using MutableEvent = Event;
/**
\brief Defines a new event type in a hierarchy
\param event_name The name of the new Event type you are making.
\param event_parenttype The name of the parent Event type in the heirarchy.
*/
#define ADD_BERTINI_EVENT_TYPE(event_name,event_parenttype) template \
class event_name : public event_parenttype \
{ BOOST_TYPE_INDEX_REGISTER_CLASS \
public: \
using HeldT = typename event_parenttype::HeldT; \
event_name(HeldT obs) : event_parenttype(obs){} \
virtual ~event_name() = default; \
event_name() = delete; }
} //re: namespace bertini
#endif