Files
FC1/Editor/Util/functor.h
romkazvo 34d6c5d489 123
2023-08-07 19:29:24 +08:00

1413 lines
42 KiB
C++

// Modified by Timur Davidenko...
//**************** callback.hpp **********************
// Copyright 1994 Rich Hickey
/* Permission to use, copy, modify, distribute and sell this software
* for any purpose is hereby granted without fee,
* provided that the above copyright notice appear in all copies and
* that both that copyright notice and this permission notice appear
* in supporting documentation. Rich Hickey makes no
* representations about the suitability of this software for any
* purpose. It is provided "as is" without express or implied warranty.
*/
// 08/31/96 Rich Hickey
// Added ==, != and <
// They are not inline, source is in file callback.cpp
// Note: You must compile and link in callback.obj if you use them
// C++ doesn't allow ptr-to-func to void * anymore -> changed to void (*)(void)
// Added compiler workarounds for MS VC++ 4.2
// Prefixed all macros with RHCB
// Note: derivation from FunctorBase is now public, and access functions
// (for func, callee etc) are provided >>for implementation use only<<
// 06/12/94 Rich Hickey
// 3rd major revision
// Now functors are concrete classes, and should be held by value
// Virtual function mechanism removed
// Generic functor() mechanism added for building functors
// from both stand-alone functions and object/ptr-to-mem-func pairs
#ifndef __functor_h__
#define __functor_h__
/*
To use:
If you wish to build a component that provides/needs a callback, simply
specify and hold a Functor of the type corresponding to the args
you wish to pass and the return value you need. There are 10 Functors
from which to choose:
Functor0
Functor1<P1>
Functor2<P1,P2>
Functor3<P1,P2,P3>
Functor4<P1,P2,P3,P4>
Functor0wRet<RT>
Functor1wRet<P1,RT>
Functor2wRet<P1,P2,RT>
Functor3wRet<P1,P2,P3,RT>
Functor4wRet<P1,P2,P3,P4,RT>
These are parameterized by their args and return value if any. Each has
a default ctor and an operator() with the corresponding signature.
They can be treated and used just like ptr-to-functions.
If you want to be a client of a component that uses callbacks, you
create a Functor by calling functor().
There are three flavors of functor - you can create a functor from:
a ptr-to-stand-alone function
an object and a pointer-to-member function.
a pointer-to-member function (which will be called on first arg of functor)
Note: this last was not covered in the article - see CBEXAM3.CPP
functor( ptr-to-function)
functor( reference-to-object,ptr-to-member-function)
functor( ptr-to-member-function)
The functor system is 100% type safe. It is also type flexible. You can
build a functor out of a function that is 'type compatible' with the
target functor - it need not have an exactly matching signature. By
type compatible I mean a function with the same number of arguments, of
types reachable from the functor's argument types by implicit conversion.
The return type of the function must be implicitly convertible to the
return type of the functor. A functor with no return can be built from a
function with a return - the return value is simply ignored.
(See ethel example below)
All the correct virtual function behavior is preserved. (see ricky
example below).
If you somehow try to create something in violation
of the type system you will get a compile-time or template-instantiation-
time error.
The Functor base class and translator
classes are artifacts of this implementation. You should not write
code that relies upon their existence. Nor should you rely on the return
value of functor being anything in particular.
All that is guaranteed is that the Functor classes have a default ctor,
a ctor that can accept 0 as an initializer,
an operator() with the requested argument types and return type, an
operator that will allow it to be evaluated in a conditional (with
'true' meaning the functor is set, 'false' meaning it is not), and that
Functors can be constructed from the result of functor(), given
you've passed something compatible to functor(). In addition you
can compare 2 functors with ==, !=, and <. 2 functors with the same
'callee' (function, object and/or ptr-to-mem-func) shall compare
equal. op < forms an ordering relation across all callee types -> the
actual order is not meaningful or to be depended upon.
/////////////////////// BEGIN Example 1 //////////////////////////
#include <iostream.h>
#include "callback.hpp"
//do5Times() is a function that takes a functor and invokes it 5 times
void do5Times(const Functor1<int> &doIt)
{
for(int i=0;i<5;i++)
doIt(i);
}
//Here are some standalone functions
void fred(int i){cout << "fred: " << i<<endl;}
int ethel(long l){cout << "ethel: " << l<<endl;return l;}
//Here is a class with a virtual function, and a derived class
class B{
public:
virtual void ricky(int i)
{cout << "B::ricky: " << i<<endl;}
};
class D:public B{
public:
void ricky(int i)
{cout << "D::ricky: " << i<<endl;}
};
void main()
{
//create a typedef of the functor type to simplify dummy argument
typedef Functor1<int> *FtorType;
Functor1<int> ftor; //a functor variable
//make a functor from ptr-to-function
ftor = functor( fred );
do5Times(ftor);
//note ethel is not an exact match - ok, is compatible
ftor = functor( ethel );
do5Times(ftor);
//create a D object to be a callback target
D myD;
//make functor from object and ptr-to-member-func
ftor = functor( myD,&B::ricky );
do5Times(ftor);
}
/////////////////////// END of example 1 //////////////////////////
/////////////////////// BEGIN Example 2 //////////////////////////
#include <iostream.h>
#include "callback.hpp"
//Button is a component that provides a functor-based
//callback mechanism, so you can wire it up to whatever you wish
class Button{
public:
//ctor takes a functor and stores it away in a member
Button(const Functor0 &uponClickDoThis):notify(uponClickDoThis)
{}
void click()
{
//invoke the functor, thus calling back client
notify();
}
private:
//note this is a data member with a verb for a name - matches its
//function-like usage
Functor0 notify;
};
class CDPlayer{
public:
void play()
{cout << "Playing"<<endl;}
void stop()
{cout << "Stopped"<<endl;}
};
void main()
{
CDPlayer myCD;
Button playButton(functor((Functor0*)0,myCD,&CDPlayer::play));
Button stopButton(functor((Functor0*)0,myCD,&CDPlayer::stop));
playButton.click(); //calls myCD.play()
stopButton.click(); //calls myCD.stop()
}
/////////////////////// END of example 2 //////////////////////////
*/
//******************************************************************
///////////////////////////////////////////////////////////////////*
//WARNING - no need to read past this point, lest confusion ensue. *
//Only the curious need explore further - but remember *
//about that cat! *
///////////////////////////////////////////////////////////////////*
//******************************************************************
//////////////////////////////
// COMPILER BUG WORKAROUNDS:
// As of version 4.02 Borland has a code generation bug
// returning the result of a call via a ptr-to-function in a template
#ifdef __BORLANDC__
#define RHCB_BC4_RET_BUG(x) RT(x)
#else
#define RHCB_BC4_RET_BUG(x) x
#endif
// MS VC++ 4.2 still has many bugs relating to templates
// This version works around them as best I can - however note that
// MS will allow 'void (T::*)()const' to bind to a non-const member function
// of T. In addition, they do not support overloading template functions
// based on constness of ptr-to-mem-funcs.
// When _MSC_VER is defined I provide only the const versions,however it is on
// the user's head, when calling functor with a const T, to make sure
// that the pointed-to member function is also const since MS won't enforce it!
// Other than that the port is completely functional under VC++ 4.2
// One MS bug you may encounter during _use_ of the callbacks:
// If you pass them by reference you can't call op() on the reference
// Workaround is to pass by value.
/*
// MS unable to call operator() on template class reference
template <class T>
class Functor{
public:
void operator()(T t)const{};
};
void foo(const Functor<int> &f)
{
f(1); //error C2064: term does not evaluate to a function
//works when f is passed by value
}
*/
// Note: if you are porting to another compiler that is having trouble you
// can try defining some of these flags as well:
#if defined(_MSC_VER)
#define RHCB_CANT_PASS_MEMFUNC_BY_REFERENCE //like it says
//#define RHCB_CANT_OVERLOAD_ON_CONSTNESS //of mem funcs
#define RHCB_CANT_INIT_REFERENCE_CTOR_STYLE //int i;int &ir(i); //MS falls down
//#define RHCB_WONT_PERFORM_PTR_CONVERSION //of 0 to ptr-to-any-type
#endif
// Don't touch this stuff
#if defined(RHCB_CANT_PASS_MEMFUNC_BY_REFERENCE)
#define RHCB_CONST_REF
#else
#define RHCB_CONST_REF const &
#endif
#if defined(RHCB_CANT_INIT_REFERENCE_CTOR_STYLE)
#define RHCB_CTOR_STYLE_INIT =
#else
#define RHCB_CTOR_STYLE_INIT
#endif
#if defined(RHCB_WONT_PERFORM_PTR_CONVERSION)
#define RHCB_DUMMY_INIT int
#else
#define RHCB_DUMMY_INIT DummyInit *
#endif
////////////////////////////// THE CODE //////////////////////////
//change these when your compiler gets bool
typedef bool RHCB_BOOL;
#include <string.h> //for memstuff
#include <stddef.h> //for size_t
//typeless representation of a function and optional object
class FunctorBase {
public:
//Note: ctors are protected
//for evaluation in conditionals - can the functor be called?
operator RHCB_BOOL()const{return callee||func;}
//Now you can put them in containers _and_ remove them!
//Source for these 3 is in callback.cpp
friend bool operator==(const FunctorBase &lhs,const FunctorBase &rhs);
friend bool operator!=(const FunctorBase &lhs,const FunctorBase &rhs);
friend bool operator<(const FunctorBase &lhs,const FunctorBase &rhs);
//The rest below for implementation use only !
class DummyInit{
};
typedef void (FunctorBase::*PMemFunc)();
typedef void (*PFunc)();
enum {MEM_FUNC_SIZE = sizeof(PMemFunc)};
PFunc getFunc() const {return func;}
void *getCallee() const {return callee;}
const char *getMemFunc() const {return memFunc;}
protected:
////////////////////////////////////////////////////////////////
// Note: this code depends on all ptr-to-mem-funcs being same size
// If that is not the case then make memFunc as large as largest
////////////////////////////////////////////////////////////////
union{
PFunc func;
char memFunc[MEM_FUNC_SIZE*2]; // Make sure we support multiple inheritace.
};
void *callee;
FunctorBase() : callee(0),func(0) {}
FunctorBase(const void *c,PFunc f, const void *mf,size_t sz) : callee((void *)c)
{
if (c) //must be callee/memfunc
{
memcpy(memFunc,mf,sz);
if(sz<MEM_FUNC_SIZE) //zero-out the rest, if any, so comparisons work
{
memset(memFunc+sz,0,MEM_FUNC_SIZE-sz);
}
}
else //must be ptr-to-func
{
func = f;
}
}
};
/************************* no arg - no return *******************/
class Functor0:public FunctorBase{
public:
Functor0(RHCB_DUMMY_INIT = 0){}
void operator()()const
{
thunk(*this);
}
protected:
typedef void (*Thunk)(const FunctorBase &);
Functor0(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class Callee, class MemFunc>
class CBMemberTranslator0:public Functor0{
public:
CBMemberTranslator0(Callee &c,const MemFunc &m):
Functor0(thunk,&c,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(callee->*memFunc)();
}
};
template <class Func>
class CBFunctionTranslator0:public Functor0{
public:
CBFunctionTranslator0(Func f):Functor0(thunk,0,(PFunc)f,0,0){}
static void thunk(const FunctorBase &ftor)
{
(Func(ftor.getFunc()))();
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType>
inline CBMemberTranslator0<Callee,RT (CallType::*)()>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)())
{
typedef RT (CallType::*MemFunc)();
return CBMemberTranslator0<Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType>
inline CBMemberTranslator0<const Callee,RT (CallType::*)()const>
functor(const Callee &c,RT (CallType::* RHCB_CONST_REF f)()const)
{
typedef RT (CallType::*MemFunc)()const;
return CBMemberTranslator0<const Callee,MemFunc>(c,f);
}
template <class RT>
inline CBFunctionTranslator0<RT (*)()>
functor(RT (*f)())
{
return CBFunctionTranslator0<RT (*)()>(f);
}
/************************* no arg - with return *******************/
/*
template <class RT>
class Functor0wRet:public FunctorBase{
public:
Functor0wRet(RHCB_DUMMY_INIT = 0){}
RT operator()()const
{
return RHCB_BC4_RET_BUG(thunk(*this));
}
protected:
typedef RT (*Thunk)(const FunctorBase &);
Functor0wRet(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class RT,class Callee, class MemFunc>
class CBMemberTranslator0wRet:public Functor0wRet<RT>{
public:
CBMemberTranslator0wRet(Callee &c,const MemFunc &m):
Functor0wRet<RT>(thunk,&c,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((callee->*memFunc)());
}
};
template <class RT,class Func>
class CBFunctionTranslator0wRet:public Functor0wRet<RT>{
public:
CBFunctionTranslator0wRet(Func f):Functor0wRet<RT>(thunk,0,(PFunc)f,0,0){}
static RT thunk(const FunctorBase &ftor)
{
return (Func(ftor.getFunc()))();
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class RT,class Callee,class CallType>
inline CBMemberTranslator0wRet<RT,Callee,RT (CallType::*)()>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)())
{
typedef RT (CallType::*MemFunc)();
return CBMemberTranslator0wRet<RT,Callee,MemFunc>(c,f);
}
#endif
template <class RT,class Callee,class CallType>
inline CBMemberTranslator0wRet<RT,const Callee,RT (CallType::*)()const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)()const)
{
typedef RT (CallType::*MemFunc)()const;
return CBMemberTranslator0wRet<RT,const Callee,MemFunc>(c,f);
}
template <class RT>
inline CBFunctionTranslator0wRet<RT,RT (*)()>
functor( RT (*f)() )
{
return CBFunctionTranslator0wRet<RT,RT (*)()>(f);
}
/************************* one arg - no return *******************/
template <class P1>
class Functor1:public FunctorBase {
public:
Functor1(RHCB_DUMMY_INIT = 0){}
void operator()(P1 p1)const
{
thunk(*this,p1);
}
//for STL
typedef P1 argument_type;
typedef void result_type;
protected:
typedef void (*Thunk)(const FunctorBase &,P1);
Functor1(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class Callee, class MemFunc>
class CBMemberTranslator1:public Functor1<P1>{
public:
CBMemberTranslator1(Callee &c,const MemFunc &m):
Functor1<P1>(thunk,&c,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(callee->*memFunc)(p1);
}
};
template <class P1,class Func>
class CBFunctionTranslator1:public Functor1<P1>{
public:
CBFunctionTranslator1(Func f):Functor1<P1>(thunk,0,(PFunc)f,0,0){}
static void thunk(const FunctorBase &ftor,P1 p1)
{
(Func(ftor.getFunc()))(p1);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class Callee,class CallType>
inline CBMemberTranslator1<P1,Callee,void (CallType::*)(P1)>
functor(Callee &c,void (CallType::* RHCB_CONST_REF f)(P1))
{
typedef void (CallType::*MemFunc)(P1);
return CBMemberTranslator1<P1,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class CallType,class P1,class RT>
inline CBMemberTranslator1<P1,const Callee,RT (CallType::*)(P1)const>
functor(const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1)const)
{
typedef RT (CallType::*MemFunc)(P1)const;
return CBMemberTranslator1<P1,const Callee,MemFunc>(c,f);
}
template <class RT,class P1>
inline CBFunctionTranslator1<P1,RT (*)(P1)>
functor( RT (*f)(P1) )
{
return CBFunctionTranslator1<P1,RT (*)(P1)>(f);
}
template <class P1,class MemFunc>
class CBMemberOf1stArgTranslator1:public Functor1<P1>{
public:
CBMemberOf1stArgTranslator1(const MemFunc &m):
Functor1<P1>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(p1.*memFunc)();
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class RT,class P1,class CallType>
inline CBMemberOf1stArgTranslator1<P1,RT (CallType::*)()>
functor( RT (CallType::* RHCB_CONST_REF f)())
{
typedef RT (CallType::*MemFunc)();
return CBMemberOf1stArgTranslator1<P1,MemFunc>(f);
}
#endif
template <class RT,class P1,class CallType>
inline CBMemberOf1stArgTranslator1<P1,RT (CallType::*)()const>
functor( RT (CallType::* RHCB_CONST_REF f)()const)
{
typedef RT (CallType::*MemFunc)()const;
return CBMemberOf1stArgTranslator1<P1,MemFunc>(f);
}
/************************* one arg - with return *******************/
/*
template <class P1,class RT>
class Functor1wRet:public FunctorBase{
public:
Functor1wRet(RHCB_DUMMY_INIT = 0){}
RT operator()(P1 p1)const
{
return RHCB_BC4_RET_BUG(thunk(*this,p1));
}
//for STL
typedef P1 argument_type;
typedef RT result_type;
protected:
typedef RT (*Thunk)(const FunctorBase &,P1);
Functor1wRet(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class RT,class Callee, class MemFunc>
class CBMemberTranslator1wRet:public Functor1wRet<P1,RT>{
public:
CBMemberTranslator1wRet(Callee &c,const MemFunc &m):
Functor1wRet<P1,RT>(thunk,&c,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((callee->*memFunc)(p1));
}
};
template <class P1,class RT,class Func>
class CBFunctionTranslator1wRet:public Functor1wRet<P1,RT>{
public:
CBFunctionTranslator1wRet(Func f):
Functor1wRet<P1,RT>(thunk,0,(PFunc)f,0,0){}
static RT thunk(const FunctorBase &ftor,P1 p1)
{
return (Func(ftor.getFunc()))(p1);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType,class P1>
inline CBMemberTranslator1wRet<P1,RT,Callee,RT (CallType::*)(P1)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1))
{
typedef RT (CallType::*MemFunc)(P1);
return CBMemberTranslator1wRet<P1,RT,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1>
inline CBMemberTranslator1wRet<P1,RT,const Callee,RT (CallType::*)(P1)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1)const )
{
typedef RT (CallType::*MemFunc)(P1)const;
return CBMemberTranslator1wRet<P1,RT,const Callee,MemFunc>(c,f);
}
template <class RT,class P1>
inline CBFunctionTranslator1wRet<P1,RT,RT (*)(P1)>
functor( RT (*f)(P1) )
{
return CBFunctionTranslator1wRet<P1,RT,RT (*)(P1)>(f);
}
template <class P1,class RT,class MemFunc>
class CBMemberOf1stArgTranslator1wRet:public Functor1wRet<P1,RT>{
public:
CBMemberOf1stArgTranslator1wRet(const MemFunc &m):
Functor1wRet<P1,RT>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((p1.*memFunc)());
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class RT,class CallType>
inline CBMemberOf1stArgTranslator1wRet<P1,RT,RT (CallType::*)()>
functor( RT (CallType::* RHCB_CONST_REF f )())
{
typedef RT (CallType::*MemFunc)();
return CBMemberOf1stArgTranslator1wRet<P1,RT,MemFunc>(f);
}
#endif
template <class P1,class RT,class CallType>
inline CBMemberOf1stArgTranslator1wRet<P1,RT,RT (CallType::*)()const>
functor( RT (CallType::* RHCB_CONST_REF f)()const )
{
typedef RT (CallType::*MemFunc)()const;
return CBMemberOf1stArgTranslator1wRet<P1,RT,MemFunc>(f);
}
/************************* two args - no return *******************/
template <class P1,class P2>
class Functor2:public FunctorBase{
public:
Functor2(RHCB_DUMMY_INIT = 0){}
void operator()(P1 p1,P2 p2)const
{
thunk(*this,p1,p2);
}
//for STL
typedef P1 first_argument_type;
typedef P2 second_argument_type;
typedef void result_type;
protected:
typedef void (*Thunk)(const FunctorBase &,P1,P2);
Functor2(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class Callee, class MemFunc>
class CBMemberTranslator2:public Functor2<P1,P2>{
public:
CBMemberTranslator2(Callee &c,const MemFunc &m):
Functor2<P1,P2>(thunk,&c,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(callee->*memFunc)(p1,p2);
}
};
template <class P1,class P2,class Func>
class CBFunctionTranslator2:public Functor2<P1,P2>{
public:
CBFunctionTranslator2(Func f):Functor2<P1,P2>(thunk,0,(PFunc)f,0,0){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
(Func(ftor.getFunc()))(p1,p2);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType,class P1,class P2>
inline CBMemberTranslator2<P1,P2,Callee,RT (CallType::*)(P1,P2)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2) )
{
typedef RT (CallType::*MemFunc)(P1,P2);
return CBMemberTranslator2<P1,P2,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2>
inline CBMemberTranslator2<P1,P2,const Callee,
RT (CallType::*)(P1,P2)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2)const)
{
typedef RT (CallType::*MemFunc)(P1,P2)const;
return CBMemberTranslator2<P1,P2,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2>
inline CBFunctionTranslator2<P1,P2,RT (*)(P1,P2)>
functor( RT (*f)(P1,P2))
{
return CBFunctionTranslator2<P1,P2,RT (*)(P1,P2)>(f);
}
template <class P1,class P2,class MemFunc>
class CBMemberOf1stArgTranslator2:public Functor2<P1,P2>{
public:
CBMemberOf1stArgTranslator2(const MemFunc &m):
Functor2<P1,P2>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(p1.*memFunc)(p2);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class RT,class CallType>
inline CBMemberOf1stArgTranslator2<P1,P2,RT (CallType::*)(P1)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1))
{
typedef RT (CallType::*MemFunc)(P1);
return CBMemberOf1stArgTranslator2<P1,P2,MemFunc>(f);
}
#endif
template <class P1,class P2,class RT,class CallType>
inline CBMemberOf1stArgTranslator2<P1,P2,RT (CallType::*)(P1)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1)const)
{
typedef RT (CallType::*MemFunc)(P1)const;
return CBMemberOf1stArgTranslator2<P1,P2,MemFunc>(f);
}
/************************* two args - with return *******************/
/*
template <class P1,class P2,class RT>
class Functor2wRet:public FunctorBase{
public:
Functor2wRet(RHCB_DUMMY_INIT = 0){}
RT operator()(P1 p1,P2 p2)const
{
return RHCB_BC4_RET_BUG(thunk(*this,p1,p2));
}
//for STL
typedef P1 first_argument_type;
typedef P2 second_argument_type;
typedef RT result_type;
protected:
typedef RT (*Thunk)(const FunctorBase &,P1,P2);
Functor2wRet(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class RT,class Callee, class MemFunc>
class CBMemberTranslator2wRet:public Functor2wRet<P1,P2,RT>{
public:
CBMemberTranslator2wRet(Callee &c,const MemFunc &m):
Functor2wRet<P1,P2,RT>(thunk,&c,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((callee->*memFunc)(p1,p2));
}
};
template <class P1,class P2,class RT,class Func>
class CBFunctionTranslator2wRet:public Functor2wRet<P1,P2,RT>{
public:
CBFunctionTranslator2wRet(Func f):
Functor2wRet<P1,P2,RT>(thunk,0,(PFunc)f,0,0){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
return (Func(ftor.getFunc()))(p1,p2);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType,class P1,class P2>
inline CBMemberTranslator2wRet<P1,P2,RT,Callee,
RT (CallType::*)(P1,P2)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2))
{
typedef RT (CallType::*MemFunc)(P1,P2);
return CBMemberTranslator2wRet<P1,P2,RT,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2>
inline CBMemberTranslator2wRet<P1,P2,RT,const Callee,
RT (CallType::*)(P1,P2)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2)const)
{
typedef RT (CallType::*MemFunc)(P1,P2)const;
return CBMemberTranslator2wRet<P1,P2,RT,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2>
inline CBFunctionTranslator2wRet<P1,P2,RT,RT (*)(P1,P2)>
functor( RT (*f)(P1,P2))
{
return CBFunctionTranslator2wRet<P1,P2,RT,RT (*)(P1,P2)>(f);
}
template <class P1,class P2,class RT,class MemFunc>
class CBMemberOf1stArgTranslator2wRet:public Functor2wRet<P1,P2,RT>{
public:
CBMemberOf1stArgTranslator2wRet(const MemFunc &m):
Functor2wRet<P1,P2,RT>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((p1.*memFunc)(p2));
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class RT,class CallType>
inline CBMemberOf1stArgTranslator2wRet<P1,P2,RT,RT (CallType::*)(P1)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1))
{
typedef RT (CallType::*MemFunc)(P1);
return CBMemberOf1stArgTranslator2wRet<P1,P2,RT,MemFunc>(f);
}
#endif
template <class P1,class P2,class RT,class CallType>
inline CBMemberOf1stArgTranslator2wRet<P1,P2,RT,RT (CallType::*)(P1)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1)const)
{
typedef RT (CallType::*MemFunc)(P1)const;
return CBMemberOf1stArgTranslator2wRet<P1,P2,RT,MemFunc>(f);
}
/************************* three args - no return *******************/
template <class P1,class P2,class P3>
class Functor3:public FunctorBase{
public:
Functor3(RHCB_DUMMY_INIT = 0){}
void operator()(P1 p1,P2 p2,P3 p3)const
{
thunk(*this,p1,p2,p3);
}
protected:
typedef void (*Thunk)(const FunctorBase &,P1,P2,P3);
Functor3(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class P3,class Callee, class MemFunc>
class CBMemberTranslator3:public Functor3<P1,P2,P3>{
public:
CBMemberTranslator3(Callee &c,const MemFunc &m):
Functor3<P1,P2,P3>(thunk,&c,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(callee->*memFunc)(p1,p2,p3);
}
};
template <class P1,class P2,class P3,class Func>
class CBFunctionTranslator3:public Functor3<P1,P2,P3>{
public:
CBFunctionTranslator3(Func f):Functor3<P1,P2,P3>(thunk,0,(PFunc)f,0,0){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
(Func(ftor.getFunc()))(p1,p2,p3);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType,class P1,class P2,class P3>
inline CBMemberTranslator3<P1,P2,P3,Callee,
RT (CallType::*)(P1,P2,P3)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3);
return CBMemberTranslator3<P1,P2,P3,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2,class P3>
inline CBMemberTranslator3<P1,P2,P3,const Callee,
RT (CallType::*)(P1,P2,P3)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3)const;
return CBMemberTranslator3<P1,P2,P3,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2,class P3>
inline CBFunctionTranslator3<P1,P2,P3,RT (*)(P1,P2,P3)>
functor( RT (*f)(P1,P2,P3))
{
return CBFunctionTranslator3<P1,P2,P3,RT (*)(P1,P2,P3)>(f);
}
template <class P1,class P2,class P3,class MemFunc>
class CBMemberOf1stArgTranslator3:public Functor3<P1,P2,P3>{
public:
CBMemberOf1stArgTranslator3(const MemFunc &m):
Functor3<P1,P2,P3>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(p1.*memFunc)(p2,p3);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class RT,class CallType>
inline CBMemberOf1stArgTranslator3<P1,P2,P3,RT (CallType::*)(P1,P2)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2))
{
typedef RT (CallType::*MemFunc)(P1,P2);
return CBMemberOf1stArgTranslator3<P1,P2,P3,MemFunc>(f);
}
#endif
template <class P1,class P2,class P3,class RT,class CallType>
inline CBMemberOf1stArgTranslator3<P1,P2,P3,RT (CallType::*)(P1,P2)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2)const)
{
typedef RT (CallType::*MemFunc)(P1,P2)const;
return CBMemberOf1stArgTranslator3<P1,P2,P3,MemFunc>(f);
}
/************************* three args - with return *******************/
/*
template <class P1,class P2,class P3,class RT>
class Functor3wRet:public FunctorBase{
public:
Functor3wRet(RHCB_DUMMY_INIT = 0){}
RT operator()(P1 p1,P2 p2,P3 p3)const
{
return RHCB_BC4_RET_BUG(thunk(*this,p1,p2,p3));
}
protected:
typedef RT (*Thunk)(const FunctorBase &,P1,P2,P3);
Functor3wRet(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class P3,
class RT,class Callee, class MemFunc>
class CBMemberTranslator3wRet:public Functor3wRet<P1,P2,P3,RT>{
public:
CBMemberTranslator3wRet(Callee &c,const MemFunc &m):
Functor3wRet<P1,P2,P3,RT>(thunk,&c,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((callee->*memFunc)(p1,p2,p3));
}
};
template <class P1,class P2,class P3,class RT,class Func>
class CBFunctionTranslator3wRet:public Functor3wRet<P1,P2,P3,RT>{
public:
CBFunctionTranslator3wRet(Func f):
Functor3wRet<P1,P2,P3,RT>(thunk,0,(PFunc)f,0,0){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
return (Func(ftor.getFunc()))(p1,p2,p3);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class RT,class Callee,class CallType>
inline CBMemberTranslator3wRet<P1,P2,P3,RT,Callee,
RT (CallType::*)(P1,P2,P3)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3);
return CBMemberTranslator3wRet<P1,P2,P3,RT,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2,class P3>
inline CBMemberTranslator3wRet<P1,P2,P3,RT,const Callee,
RT (CallType::*)(P1,P2,P3)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3)const;
return CBMemberTranslator3wRet<P1,P2,P3,RT,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2,class P3>
inline CBFunctionTranslator3wRet<P1,P2,P3,RT,RT (*)(P1,P2,P3)>
functor( RT (*f)(P1,P2,P3) )
{
return CBFunctionTranslator3wRet<P1,P2,P3,RT,RT (*)(P1,P2,P3)>(f);
}
template <class P1,class P2,class P3,class RT,class MemFunc>
class CBMemberOf1stArgTranslator3wRet:public Functor3wRet<P1,P2,P3,RT>{
public:
CBMemberOf1stArgTranslator3wRet(const MemFunc &m):
Functor3wRet<P1,P2,P3,RT>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((p1.*memFunc)(p2,p3));
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class RT,class CallType>
inline CBMemberOf1stArgTranslator3wRet<P1,P2,P3,RT,RT (CallType::*)(P1,P2)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2))
{
typedef RT (CallType::*MemFunc)(P1,P2);
return CBMemberOf1stArgTranslator3wRet<P1,P2,P3,RT,MemFunc>(f);
}
#endif
template <class P1,class P2,class P3,class RT,class CallType>
inline CBMemberOf1stArgTranslator3wRet<P1,P2,P3,RT,
RT (CallType::*)(P1,P2)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2)const)
{
typedef RT (CallType::*MemFunc)(P1,P2)const;
return CBMemberOf1stArgTranslator3wRet<P1,P2,P3,RT,MemFunc>(f);
}
/************************* four args - no return *******************/
template <class P1,class P2,class P3,class P4>
class Functor4:public FunctorBase{
public:
Functor4(RHCB_DUMMY_INIT = 0){}
void operator()(P1 p1,P2 p2,P3 p3,P4 p4)const
{
thunk(*this,p1,p2,p3,p4);
}
protected:
typedef void (*Thunk)(const FunctorBase &,P1,P2,P3,P4);
Functor4(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class P3,class P4,
class Callee, class MemFunc>
class CBMemberTranslator4:public Functor4<P1,P2,P3,P4>{
public:
CBMemberTranslator4(Callee &c,const MemFunc &m):
Functor4<P1,P2,P3,P4>(thunk,&c,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(callee->*memFunc)(p1,p2,p3,p4);
}
};
template <class P1,class P2,class P3,class P4,class Func>
class CBFunctionTranslator4:public Functor4<P1,P2,P3,P4>{
public:
CBFunctionTranslator4(Func f):
Functor4<P1,P2,P3,P4>(thunk,0,(PFunc)f,0,0){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
(Func(ftor.getFunc()))(p1,p2,p3,p4);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class Callee,class RT,class CallType,class P1,class P2,class P3,class P4>
inline CBMemberTranslator4<P1,P2,P3,P4,Callee,
RT (CallType::*)(P1,P2,P3,P4)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3,P4))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3,P4);
return CBMemberTranslator4<P1,P2,P3,P4,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2,class P3,class P4>
inline CBMemberTranslator4<P1,P2,P3,P4,const Callee,
RT (CallType::*)(P1,P2,P3,P4)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3,P4)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3,P4)const;
return CBMemberTranslator4<P1,P2,P3,P4,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2,class P3,class P4>
inline CBFunctionTranslator4<P1,P2,P3,P4,RT (*)(P1,P2,P3,P4)>
functor( RT (*f)(P1,P2,P3,P4))
{
return CBFunctionTranslator4<P1,P2,P3,P4,RT (*)(P1,P2,P3,P4)>(f);
}
template <class P1,class P2,class P3,class P4,class MemFunc>
class CBMemberOf1stArgTranslator4:public Functor4<P1,P2,P3,P4>{
public:
CBMemberOf1stArgTranslator4(const MemFunc &m):
Functor4<P1,P2,P3,P4>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static void thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
(p1.*memFunc)(p2,p3,p4);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class P4,class RT,class CallType>
inline CBMemberOf1stArgTranslator4<P1,P2,P3,P4,RT (CallType::*)(P1,P2,P3)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3);
return CBMemberOf1stArgTranslator4<P1,P2,P3,P4,MemFunc>(f);
}
#endif
template <class P1,class P2,class P3,class P4,class RT,class CallType>
inline CBMemberOf1stArgTranslator4<P1,P2,P3,P4,
RT (CallType::*)(P1,P2,P3)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3)const;
return CBMemberOf1stArgTranslator4<P1,P2,P3,P4,MemFunc>(f);
}
/************************* four args - with return *******************/
/*
template <class P1,class P2,class P3,class P4,class RT>
class Functor4wRet:public FunctorBase{
public:
Functor4wRet(RHCB_DUMMY_INIT = 0){}
RT operator()(P1 p1,P2 p2,P3 p3,P4 p4)const
{
return RHCB_BC4_RET_BUG(thunk(*this,p1,p2,p3,p4));
}
protected:
typedef RT (*Thunk)(const FunctorBase &,P1,P2,P3,P4);
Functor4wRet(Thunk t,const void *c,PFunc f,const void *mf,size_t sz):
FunctorBase(c,f,mf,sz),thunk(t){}
private:
Thunk thunk;
};
template <class P1,class P2,class P3,class P4,class RT,class Callee, class MemFunc>
class CBMemberTranslator4wRet:public Functor4wRet<P1,P2,P3,P4,RT>{
public:
CBMemberTranslator4wRet(Callee &c,const MemFunc &m):
Functor4wRet<P1,P2,P3,P4,RT>(thunk,&c,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
Callee *callee = (Callee *)ftor.getCallee();
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((callee->*memFunc)(p1,p2,p3,p4));
}
};
template <class P1,class P2,class P3,class P4,class RT,class Func>
class CBFunctionTranslator4wRet:public Functor4wRet<P1,P2,P3,P4,RT>{
public:
CBFunctionTranslator4wRet(Func f):
Functor4wRet<P1,P2,P3,P4,RT>(thunk,0,(PFunc)f,0,0){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
return (Func(ftor.getFunc()))(p1,p2,p3,p4);
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class P4,class RT,class Callee,class CallType>
inline CBMemberTranslator4wRet<P1,P2,P3,P4,RT,Callee,
RT (CallType::*)(P1,P2,P3,P4)>
functor( Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3,P4))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3,P4);
return CBMemberTranslator4wRet<P1,P2,P3,P4,RT,Callee,MemFunc>(c,f);
}
#endif
template <class Callee,class RT,class CallType,class P1,class P2,class P3,class P4>
inline CBMemberTranslator4wRet<P1,P2,P3,P4,RT,const Callee,
RT (CallType::*)(P1,P2,P3,P4)const>
functor( const Callee &c,RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3,P4)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3,P4)const;
return CBMemberTranslator4wRet<P1,P2,P3,P4,RT,const Callee,MemFunc>(c,f);
}
template <class RT,class P1,class P2,class P3,class P4>
inline CBFunctionTranslator4wRet<P1,P2,P3,P4,RT,RT (*)(P1,P2,P3,P4)>
functor( RT (*f)(P1,P2,P3,P4) )
{
return CBFunctionTranslator4wRet
<P1,P2,P3,P4,RT,RT (*)(P1,P2,P3,P4)>(f);
}
template <class P1,class P2,class P3,class P4,class RT,class MemFunc>
class CBMemberOf1stArgTranslator4wRet:public Functor4wRet<P1,P2,P3,P4,RT>{
public:
CBMemberOf1stArgTranslator4wRet(const MemFunc &m):
Functor4wRet<P1,P2,P3,P4,RT>(thunk,(void *)1,0,&m,sizeof(MemFunc)){}
static RT thunk(const FunctorBase &ftor,P1 p1,P2 p2,P3 p3,P4 p4)
{
MemFunc &memFunc RHCB_CTOR_STYLE_INIT
(*(MemFunc*)(void *)(ftor.getMemFunc()));
return RHCB_BC4_RET_BUG((p1.*memFunc)(p2,p3,p4));
}
};
#if !defined(RHCB_CANT_OVERLOAD_ON_CONSTNESS)
template <class P1,class P2,class P3,class P4,class RT,class CallType>
inline CBMemberOf1stArgTranslator4wRet<P1,P2,P3,P4,RT,
RT (CallType::*)(P1,P2,P3)>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3))
{
typedef RT (CallType::*MemFunc)(P1,P2,P3);
return CBMemberOf1stArgTranslator4wRet<P1,P2,P3,P4,RT,MemFunc>(f);
}
#endif
template <class P1,class P2,class P3,class P4,class RT,class CallType>
inline CBMemberOf1stArgTranslator4wRet<P1,P2,P3,P4,RT,
RT (CallType::*)(P1,P2,P3)const>
functor( RT (CallType::* RHCB_CONST_REF f)(P1,P2,P3)const)
{
typedef RT (CallType::*MemFunc)(P1,P2,P3)const;
return CBMemberOf1stArgTranslator4wRet<P1,P2,P3,P4,RT,MemFunc>(f);
}
*/
///////////////////////////////////////////////////////////////////////////////
//
// Inlines.
//
///////////////////////////////////////////////////////////////////////////////
inline bool operator==(const FunctorBase &lhs,const FunctorBase &rhs)
{
if (!lhs.callee) {
if (rhs.callee) return false;
return lhs.func == rhs.func;
} else {
if (!rhs.callee) return false;
return lhs.callee == rhs.callee &&
!memcmp(lhs.memFunc,rhs.memFunc,FunctorBase::MEM_FUNC_SIZE);
}
}
inline bool operator!=(const FunctorBase &lhs,const FunctorBase &rhs)
{
return !(lhs == rhs);
}
inline bool operator<(const FunctorBase &lhs,const FunctorBase &rhs)
{
//must order across funcs and callee/memfuncs, funcs are first
if(!lhs.callee)
{
if(rhs.callee)
return true;
else
return lhs.func < rhs.func;
}
else
{
if(!rhs.callee)
return false;
if(lhs.callee != rhs.callee)
return lhs.callee < rhs.callee;
else
return memcmp(lhs.memFunc,rhs.memFunc,FunctorBase::MEM_FUNC_SIZE)<0;
}
}
//////////////////////////////////////////////////////////////////////////
template <class FUNCTOR>
class CFunctorsList
{
public:
// Add functor to list.
void Add( const FUNCTOR &f )
{
m_functors.push_back( f );
}
// Remvoe functor from list.
void Remove( const FUNCTOR &f )
{
Container::iterator it = std::find( m_functors.begin(),m_functors.end(),f );
if (it != m_functors.end())
{
m_functors.erase( it );
}
}
//////////////////////////////////////////////////////////////////////////
// Call all functors in this list.
// Also several template functions for multiple parameters.
//////////////////////////////////////////////////////////////////////////
void Call()
{
for (Container::iterator it = m_functors.begin(); it != m_functors.end(); ++it)
{
(*it)();
}
}
template <class T1>
void Call( const T1 &param1 )
{
for (Container::iterator it = m_functors.begin(); it != m_functors.end(); ++it)
{
(*it)( param1 );
}
}
template <class T1,class T2>
void Call( const T1 &param1,const T2 &param2 )
{
for (Container::iterator it = m_functors.begin(); it != m_functors.end(); ++it)
{
(*it)( param1,param2 );
}
}
template <class T1,class T2,class T3>
void Call( const T1 &param1,const T2 &param2,const T3 &param3 )
{
for (Container::iterator it = m_functors.begin(); it != m_functors.end(); ++it)
{
(*it)( param1,param2,param3 );
}
}
private:
typedef std::list<FUNCTOR> Container;
Container m_functors;
};
#endif // __functor_h__