ResInsight/Fwk/AppFwk/cafProjectDataModel/cafPdmCore/cafSignal.h
Magne Sjaastad bfda6519db
Cleanup includes
Use new tools in Visual Studio to help cleaning up includes and remove redundant includes.
2024-03-07 19:45:43 +01:00

193 lines
6.8 KiB
C++

//##################################################################################################
//
// Custom Visualization Core library
// Copyright (C) 2020- Ceetron Solutions AS
//
// This library may be used under the terms of either the GNU General Public License or
// the GNU Lesser General Public License as follows:
//
// GNU General Public License Usage
// This library 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.
//
// This library 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 at <<http://www.gnu.org/licenses/gpl.html>>
// for more details.
//
// GNU Lesser General Public License Usage
// This library is free software; you can redistribute it and/or modify
// it under the terms of the GNU Lesser General Public License as published by
// the Free Software Foundation; either version 2.1 of the License, or
// (at your option) any later version.
//
// This library 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 Lesser General Public License at <<http://www.gnu.org/licenses/lgpl-2.1.html>>
// for more details.
//
//##################################################################################################
#pragma once
#include "cafAssert.h"
#include <functional>
#include <list>
#include <map>
#include <type_traits>
namespace caf
{
class SignalEmitter;
class SignalObserver;
class AbstractSignal
{
public:
virtual ~AbstractSignal() = default;
virtual void disconnect( SignalObserver* observer ) = 0;
};
//==================================================================================================
/// SignalEmitter
/// Should be inherited by any object that emits signals.
//==================================================================================================
class SignalEmitter
{
public:
SignalEmitter();
virtual ~SignalEmitter();
void addEmittedSignal( AbstractSignal* signalToAdd ) const;
std::list<AbstractSignal*> emittedSignals() const;
private:
mutable std::list<AbstractSignal*> m_signals;
};
//==================================================================================================
/// SignalObserver.
/// Should be inherited by any object that observes and reacts to signals.
//==================================================================================================
class SignalObserver
{
public:
SignalObserver();
virtual ~SignalObserver();
std::list<AbstractSignal*> observedSignals() const;
void addObservedSignal( AbstractSignal* signalToAdd ) const;
void removeObservedSignal( AbstractSignal* signalToRemove ) const;
private:
void disconnectAllSignals();
private:
mutable std::list<AbstractSignal*> m_signals;
};
//==================================================================================================
/// General signal class.
/// Connect any member function with the signature void(const Signal*, const SignalData* data)
/// Connect with .connect(this, &Class::nameOfMethod)
/// The method should accept that data may be nullptr
//==================================================================================================
template <typename... Args>
class Signal : public AbstractSignal
{
public:
using MemberCallback = std::function<void( const SignalEmitter*, Args... args )>;
using MemberCallbackAndActiveFlag = std::pair<MemberCallback, bool>;
public:
Signal( const SignalEmitter* emitter )
: m_emitter( emitter )
{
m_emitter->addEmittedSignal( this );
}
virtual ~Signal()
{
for ( auto observerCallbackPair : m_observerCallbacks )
{
observerCallbackPair.first->removeObservedSignal( this );
}
}
template <typename ClassType>
void connect( ClassType* observer, void ( ClassType::*method )( const SignalEmitter*, Args... args ) )
{
MemberCallback lambda = [=]( const SignalEmitter* emitter, Args... args )
{
// Call method
( observer->*method )( emitter, args... );
};
connect( observer, lambda );
}
template <typename ClassType>
void connect( ClassType* observer, const MemberCallback& callback )
{
static_assert( std::is_convertible<ClassType*, SignalObserver*>::value,
"Only classes that inherit SignalObserver can connect as an observer of a Signal." );
// Connecting a signal twice is not supported
// CAF_ASSERT( m_observerCallbacks.find( observer ) == m_observerCallbacks.end() );
if ( m_observerCallbacks.find( observer ) != m_observerCallbacks.end() ) return;
m_observerCallbacks[observer] = std::make_pair( callback, true );
observer->addObservedSignal( this );
}
// Disconnect an observer from the signal. Do this only when the relationship between the
// observer and emitter is severed but the object kept alive.
// There's no need to do this when deleting the observer.
void disconnect( SignalObserver* observer ) override
{
m_observerCallbacks.erase( observer );
observer->removeObservedSignal( this );
}
void send( Args... args ) const
{
auto observerCallBacksCopy = m_observerCallbacks;
for ( auto observerCallbackPair : observerCallBacksCopy )
{
if ( observerCallbackPair.second.second )
{
observerCallbackPair.second.first( m_emitter, args... );
}
}
}
void block( SignalObserver* observer )
{
auto it = m_observerCallbacks.find( observer );
CAF_ASSERT( it != m_observerCallbacks.end() );
it->second.second = false;
}
void unblock( SignalObserver* observer )
{
auto it = m_observerCallbacks.find( observer );
CAF_ASSERT( it != m_observerCallbacks.end() );
it->second.second = true;
}
size_t observerCount() const { return m_observerCallbacks.size(); }
private:
Signal() = delete;
Signal( const Signal& rhs ) = delete;
Signal& operator=( const Signal& rhs ) = delete;
private:
std::map<SignalObserver*, MemberCallbackAndActiveFlag> m_observerCallbacks;
const SignalEmitter* m_emitter;
};
} // namespace caf