emailrelay/src/glib/gslot.h
Graeme Walker 7f88cace83 v1.1
2003-05-18 12:00:00 +00:00

416 lines
11 KiB
C++

//
// Copyright (C) 2001-2003 Graeme Walker <graeme_walker@users.sourceforge.net>
//
// This program 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 2 of the License, or (at your option) any later
// version.
//
// This program 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 this program; if not, write to the Free Software
// Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
//
// ===
//
// gslot.h
//
// Inspired by libsigc++, but simplified by:
// * not doing multicast
// * not detecting dangling references
// * not supporting global function callbacks
// * using only void returns
//
// The key to the implementation is that SlotImp<> is
// templated on the callback parameter P and the callback
// sink class T, but Slot<> is templated only on P, so
// the event-source class does not need to know the type
// of the event-sink. The SlotOp<> classes are used to
// resolve this mismatch, by downcasting to the T-specific
// SlotImp<> class. The Slot<> classes are reference-counting
// handles to SlotImp<> (via SlotBase), with an additional
// function pointer for the relevant SlotOp<> downcasting
// method.
//
// The overloaded template function slot<>() creates
// a Slot<> handle, passing it a suitable SlotBase body
// and casting function. The dynamic type of the SlotBase
// reference is a SlotImp<> class which knows about the
// specific sink class T, and the casting function knows
// how to access the derived class. The combination of
// SlotBase polymorphism and casting function isolates the
// slot class from the sink class.
//
// Usage:
/// class Source
/// {
/// public:
/// Signal1<int> m_signal ;
/// private:
/// void Source::raiseEvent()
/// {
/// int n = 123 ;
/// m_signal.emit( n ) ;
/// }
/// } ;
///
/// class Sink
/// {
/// public:
/// void onEvent( int n ) ;
/// Sink( Source & source )
/// {
/// source.m_signal.connect( slot(*this,&Sink::onEvent) ) ;
/// }
/// } ;
//
#ifndef G_SLOT_H
#define G_SLOT_H
#include "gdef.h"
#include "gexception.h"
#include "gnoncopyable.h"
namespace G
{
// Class: SlotBase
// Description: Part of the slot/signal system.
// Used as a base class to all SlotImp<> classes
// allowing them to be used as bodies to the
// Slot<> reference-counting handle.
//
class SlotBase
{
public:
virtual ~SlotBase() ;
// Destructor.
SlotBase() ;
// Default constuctor.
void up() ;
// Increments the reference count.
void down() ;
// Decrements the reference count
// and does "delete this" on zero.
private:
SlotBase( const SlotBase & ) ; // not implemented
void operator=( const SlotBase & ) ; // not implemented
private:
unsigned long m_ref_count ;
} ;
// Class: SignalImp
// Description: Part of the slot/signal system.
// A static helper class used by Signal<> classes.
//
class SignalImp
{
public:
G_EXCEPTION( AlreadyConnected , "signal already connected to a slot" ) ;
static void check( const SlotBase * p ) ;
private:
SignalImp() ; // not implemented
} ;
template <class T>
inline
void swap_( T & t1 , T & t2 ) // no std::swap in gcc2.95
{
T temp( t1 ) ;
t1 = t2 ;
t2 = temp ;
}
// ===
// Class: SlotImp0
// Description: Part of the slot/signal system.
// An implementation class for Slot0<>. An instance
// is created by the slot<>()
//
template <class T>
class SlotImp0 : public SlotBase
{
T & m_object ;
void (T::*m_fn)() ;
public:
SlotImp0( T & object , void (T::*fn)() ) : m_object(object) , m_fn(fn) {}
void callback() { (m_object.*m_fn)() ; }
} ;
// Class: SlotOp0
// Description: Part of the slot/signal system.
//
template <class T>
class SlotOp0
{
public:
static void callback( SlotBase * imp )
{ static_cast<SlotImp0<T>*>(imp)->callback() ; }
} ;
// Class: Slot0
// Description: Part of the slot/signal system.
//
class Slot0
{
SlotBase * m_imp ;
void (*m_op)( SlotBase * ) ;
public:
Slot0() : m_imp(0) , m_op(0) {}
Slot0( SlotBase * imp , void (*op)(SlotBase*) ) : m_imp(imp) , m_op(op) {}
~Slot0() { if( m_imp ) m_imp->down() ; }
void callback() { if( m_imp ) (*m_op)( m_imp ) ; }
Slot0( const Slot0 & other ) : m_imp(other.m_imp) , m_op(other.m_op) { if(m_imp) m_imp->up() ; }
void swap( Slot0 & rhs ) { swap_(m_imp,rhs.m_imp) ; swap_(m_op,rhs.m_op) ; }
void operator=( const Slot0 & rhs ) { Slot0 tmp(rhs) ; swap(tmp) ; }
const SlotBase * base() const { return m_imp ; }
} ;
// Class: Signal0
// Description: Part of the slot/signal system.
//
class Signal0 : public noncopyable
{
Slot0 m_slot ;
public:
Signal0() {}
void emit() { m_slot.callback() ; }
void connect( Slot0 slot ) { SignalImp::check(m_slot.base()) ; m_slot = slot ; }
void disconnect() { m_slot = Slot0() ; }
} ;
// Function: slot
// Description: Part of the slot/signal system.
//
template <class T>
inline
Slot0 slot( T & object , void (T::*fn)() )
{
return Slot0( new SlotImp0<T>(object,fn) , SlotOp0<T>::callback ) ;
}
// ===
// Class: SlotImp1
// Description: Part of the slot/signal system.
//
template <class T, class P>
class SlotImp1 : public SlotBase
{
T & m_object ;
void (T::*m_fn)( P ) ;
public:
SlotImp1( T & object , void (T::*fn)(P) ) : m_object(object) , m_fn(fn) {}
void callback( P p ) { (m_object.*m_fn)(p) ; }
} ;
// Class: SlotOp1
// Description: Part of the slot/signal system.
//
template <class T, class P>
class SlotOp1
{
public:
static void callback( SlotBase * imp , P p )
{ static_cast<SlotImp1<T,P>*>(imp)->callback( p ) ; }
} ;
// Class: Slot1
// Description: Part of the slot/signal system.
//
template <class P>
class Slot1
{
SlotBase * m_imp ;
void (*m_op)( SlotBase * , P ) ;
public:
Slot1() : m_imp(0) , m_op(0) {}
Slot1( SlotBase * imp , void (*op)(SlotBase*,P) ) : m_imp(imp) , m_op(op) {}
~Slot1() { if( m_imp ) m_imp->down() ; }
void callback( P p ) { if( m_imp ) (*m_op)( m_imp , p ) ; }
Slot1( const Slot1<P> & other ) : m_imp(other.m_imp) , m_op(other.m_op) { if(m_imp) m_imp->up() ; }
void swap( Slot1<P> & rhs ) { swap_(m_imp,rhs.m_imp) ; swap_(m_op,rhs.m_op) ; }
void operator=( const Slot1<P> & rhs ) { Slot1 tmp(rhs) ; swap(tmp) ; }
const SlotBase * base() const { return m_imp ; }
} ;
// Class: Signal1
// Description: Part of the slot/signal system.
//
template <class P>
class Signal1 : public noncopyable
{
Slot1<P> m_slot ;
public:
Signal1() {}
void emit( P p ) { m_slot.callback( p ) ; }
void connect( Slot1<P> slot ) { SignalImp::check(m_slot.base()) ; m_slot = slot ; }
void disconnect() { m_slot = Slot1<P>() ; }
} ;
// Function: slot
// Description: Part of the slot/signal system.
//
template <class T,class P>
inline
Slot1<P> slot( T & object , void (T::*fn)(P) )
{
return Slot1<P>( new SlotImp1<T,P>(object,fn) , SlotOp1<T,P>::callback ) ;
}
// ===
// Class: SlotImp2
// Description: Part of the slot/signal system.
//
template <class T, class P1, class P2>
class SlotImp2 : public SlotBase
{
T & m_object ;
void (T::*m_fn)( P1 , P2 ) ;
public:
SlotImp2( T & object , void (T::*fn)(P1,P2) ) : m_object(object) , m_fn(fn) {}
void callback( P1 p1 , P2 p2 ) { (m_object.*m_fn)(p1,p2) ; }
} ;
// Class: SlotOp2
// Description: Part of the slot/signal system.
//
template <class T, class P1 , class P2>
class SlotOp2
{
public:
static void callback( SlotBase * imp , P1 p1 , P2 p2 )
{ static_cast<SlotImp2<T,P1,P2>*>(imp)->callback( p1 , p2 ) ; }
} ;
// Class: Slot2
// Description: Part of the slot/signal system.
//
template <class P1, class P2>
class Slot2
{
SlotBase * m_imp ;
void (*m_op)( SlotBase * , P1 , P2 ) ;
public:
Slot2() : m_imp(0) , m_op(0) {}
Slot2( SlotBase * imp , void (*op)(SlotBase*,P1,P2) ) : m_imp(imp) , m_op(op) {}
~Slot2() { if( m_imp ) m_imp->down() ; }
void callback( P1 p1 , P2 p2 ) { if( m_imp ) (*m_op)( m_imp , p1 , p2 ) ; }
Slot2( const Slot2<P1,P2> & other ) : m_imp(other.m_imp) , m_op(other.m_op) { if(m_imp) m_imp->up() ; }
void swap( Slot2<P1,P2> & rhs ) { swap_(m_imp,rhs.m_imp) ; swap_(m_op,rhs.m_op) ; }
void operator=( const Slot2<P1,P2> & rhs ) { Slot2 tmp(rhs) ; swap(tmp) ; }
const SlotBase * base() const { return m_imp ; }
} ;
// Class: Signal2
// Description: Part of the slot/signal system.
//
template <class P1, class P2>
class Signal2 : public noncopyable
{
Slot2<P1,P2> m_slot ;
public:
Signal2() {}
void emit( P1 p1 , P2 p2 ) { m_slot.callback( p1 , p2 ) ; }
void connect( Slot2<P1,P2> slot ) { SignalImp::check(m_slot.base()) ; m_slot = slot ; }
void disconnect() { m_slot = Slot2<P1,P2>() ; }
} ;
// Function: slot
// Description: Part of the slot/signal system.
//
template <class T, class P1, class P2>
inline
Slot2<P1,P2> slot( T & object , void (T::*fn)(P1,P2) )
{
return Slot2<P1,P2>( new SlotImp2<T,P1,P2>(object,fn) , SlotOp2<T,P1,P2>::callback ) ;
}
// ===
// Class: SlotImp3
// Description: Part of the slot/signal system.
//
template <class T, class P1, class P2, class P3>
class SlotImp3 : public SlotBase
{
T & m_object ;
void (T::*m_fn)( P1 , P2 , P3 ) ;
public:
SlotImp3( T & object , void (T::*fn)(P1,P2,P3) ) : m_object(object) , m_fn(fn) {}
void callback( P1 p1 , P2 p2 , P3 p3 ) { (m_object.*m_fn)(p1,p2,p3) ; }
} ;
// Class: SlotOp3
// Description: Part of the slot/signal system.
//
template <class T, class P1 , class P2, class P3>
class SlotOp3
{
public:
static void callback( SlotBase * imp , P1 p1 , P2 p2 , P3 p3 )
{ static_cast<SlotImp3<T,P1,P2,P3>*>(imp)->callback( p1 , p2 , p3 ) ; }
} ;
// Class: Slot3
// Description: Part of the slot/signal system.
//
template <class P1, class P2, class P3>
class Slot3
{
SlotBase * m_imp ;
void (*m_op)( SlotBase * , P1 , P2 , P3 ) ;
public:
Slot3() : m_imp(0) , m_op(0) {}
Slot3( SlotBase * imp , void (*op)(SlotBase*,P1,P2,P3) ) : m_imp(imp) , m_op(op) {}
~Slot3() { if( m_imp ) m_imp->down() ; }
void callback( P1 p1 , P2 p2 , P3 p3 ) { if( m_imp ) (*m_op)( m_imp , p1 , p2 , p3 ) ; }
Slot3( const Slot3<P1,P2,P3> & other ) : m_imp(other.m_imp) , m_op(other.m_op) { if(m_imp) m_imp->up() ; }
void swap( Slot3<P1,P2,P3> & rhs ) { swap_(m_imp,rhs.m_imp) ; swap_(m_op,rhs.m_op) ; }
void operator=( const Slot3<P1,P2,P3> & rhs ) { Slot3 tmp(rhs) ; swap(tmp) ; }
const SlotBase * base() const { return m_imp ; }
} ;
// Class: Signal3
// Description: Part of the slot/signal system.
//
template <class P1, class P2, class P3>
class Signal3 : public noncopyable
{
Slot3<P1,P2,P3> m_slot ;
public:
Signal3() {}
void emit( P1 p1 , P2 p2 , P3 p3 ) { m_slot.callback( p1 , p2 , p3 ) ; }
void connect( Slot3<P1,P2,P3> slot ) { SignalImp::check(m_slot.base()) ; m_slot = slot ; }
void disconnect() { m_slot = Slot3<P1,P2,P3>() ; }
} ;
// Function: slot
// Description: Part of the slot/signal system.
//
template <class T, class P1, class P2, class P3>
inline
Slot3<P1,P2,P3> slot( T & object , void (T::*fn)(P1,P2,P3) )
{
return Slot3<P1,P2,P3>( new SlotImp3<T,P1,P2,P3>(object,fn) , SlotOp3<T,P1,P2,P3>::callback ) ;
}
} // namespace
#endif