// MemFun.hh for FbTk, Fluxbox Toolkit // Copyright (c) 2008 Henrik Kinnunen (fluxgen at fluxbox dot org) // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. #ifndef FBTK_MEM_FUN_HH #define FBTK_MEM_FUN_HH #include "SelectArg.hh" namespace FbTk { /// No argument functor template class MemFun0 { public: typedef ReturnType (Object:: *Action)(); MemFun0(Object& obj, Action action): m_obj(obj), m_action(action) { } void operator ()() { (m_obj.*m_action)(); } private: Object& m_obj; Action m_action; }; template MemFun0 MemFun( Object& obj, ReturnType (Object:: *action)() ) { return MemFun0(obj, action); } /// One argument functor template class MemFun1 { public: typedef ReturnType (Object:: *Action)(Arg1); MemFun1(Object& obj, Action action): m_obj(obj), m_action(action) { } void operator ()(Arg1 arg1) { (m_obj.*m_action)(arg1); } private: Object& m_obj; Action m_action; }; /// One argument functor helper function template MemFun1 MemFun( Object& obj, ReturnType (Object:: *action)(Arg1) ) { return MemFun1(obj, action); } /// Two argument functor template class MemFun2 { public: typedef ReturnType (Object:: *Action)(Arg1,Arg2); MemFun2(Object& obj, Action action): m_obj(obj), m_action(action) { } void operator ()(Arg1 arg1, Arg2 arg2) { (m_obj.*m_action)(arg1, arg2); } private: Object& m_obj; Action m_action; }; /// Two argument functor helper function template MemFun2 MemFun( Object& obj, ReturnType (Object:: *action)(Arg1,Arg2) ) { return MemFun2(obj, action); } /// Three argument functor template class MemFun3 { public: typedef ReturnType (Object:: *Action)(Arg1,Arg2,Arg3); MemFun3(Object& obj, Action action): m_obj(obj), m_action(action) { } void operator ()(Arg1 arg1, Arg2 arg2, Arg3 arg3) { (m_obj.*m_action)(arg1, arg2, arg3); } private: Object& m_obj; Action m_action; }; /// Three argument functor helper template MemFun3 MemFun( Object& obj, ReturnType (Object:: *action)(Arg1, Arg2, Arg3) ) { return MemFun3(obj, action); } /// Ignores all arguments template class MemFun0IgnoreArgs: public MemFun0 { public: typedef MemFun0 BaseType; MemFun0IgnoreArgs(Object& obj, typename BaseType::Action action): BaseType(obj, action) { } template void operator ()(IgnoreType1&, IgnoreType2&, IgnoreType3&) { BaseType::operator ()(); } template void operator ()(IgnoreType1&, IgnoreType2&) { BaseType::operator ()(); } template void operator ()(IgnoreType1&) { BaseType::operator ()(); } }; /// Ignores second and third argument template class MemFun1IgnoreArgs: public MemFun1 { public: typedef MemFun1 BaseType; MemFun1IgnoreArgs(Object& obj, typename BaseType::Action& action): BaseType(obj, action) { } template void operator ()(Arg1 arg1, IgnoreType1&, IgnoreType2&) { BaseType::operator ()(arg1); } template void operator ()(Arg1 arg1, IgnoreType&) { BaseType::operator ()(arg1); } }; /// Takes two arguments but ignores the third template class MemFun2IgnoreArgs: public MemFun2 { public: typedef MemFun2 BaseType; MemFun2IgnoreArgs(Object& obj, typename BaseType::Action& action): BaseType(obj, action) { } template < typename IgnoreType > void operator ()(Arg1 arg1, Arg2 arg2, IgnoreType&) { BaseType::operator ()(arg1, arg2); } }; /// Creates functor that ignores all arguments. template MemFun0IgnoreArgs MemFunIgnoreArgs( Object& obj, ReturnType (Object:: *action)() ) { return MemFun0IgnoreArgs(obj, action); } /// Creates functor that ignores second and third argument. template MemFun1IgnoreArgs MemFunIgnoreArgs( Object& obj, ReturnType (Object:: *action)(Arg1) ) { return MemFun1IgnoreArgs(obj, action); } /// Creates functor that ignores third argument. template MemFun2IgnoreArgs MemFunIgnoreArgs( Object& obj, ReturnType (Object:: *action)(Arg1,Arg2) ) { return MemFun2IgnoreArgs(obj, action); } /** * Creates a functor that selects a specific argument of three possible ones * and uses it for the single argument operator. */ template < int ArgNum, typename Functor> class MemFunSelectArgImpl { public: MemFunSelectArgImpl(Functor func): m_func(func) { } template void operator ()(Type1& a, Type2& b, Type3& c) { m_func(STLUtil::SelectArg()(a, b, c)); } template void operator ()(Type1& a, Type2& b) { m_func(STLUtil::SelectArg()(a, b)); } template void operator ()(Type1& a) { m_func(a); } private: Functor m_func; }; /// Creates a functor that selects the first argument of three possible ones /// and uses it for the single argument operator. template MemFunSelectArgImpl<0, MemFun1 > MemFunSelectArg0(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<0, MemFun1 >(MemFun(obj, action)); } /// Creates a functor that selects the second argument (or first if there is /// only one) of three possible onesand uses it for the single argument operator. template MemFunSelectArgImpl<1, MemFun1 > MemFunSelectArg1(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<1, MemFun1 >(MemFun(obj, action)); } /// Creates a functor that selects the third argument (or the last argument if there is /// less than three arguments) of three possible onesand uses it for the single argument operator. template MemFunSelectArgImpl<2, MemFun1 > MemFunSelectArg2(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<2, MemFun1 >(MemFun(obj, action)); } } // namespace FbTk #endif // FBTK_MEM_FUN_HH