// 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 #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) { } ReturnType operator ()() const { return (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) { } ReturnType operator ()(Arg1 arg1) const { return (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) { } ReturnType operator ()(Arg1 arg1, Arg2 arg2) const { return (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) { } ReturnType operator ()(Arg1 arg1, Arg2 arg2, Arg3 arg3) const { return (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 ReturnType operator ()(IgnoreType1&, IgnoreType2&, IgnoreType3&) const { return BaseType::operator ()(); } template ReturnType operator ()(IgnoreType1&, IgnoreType2&) const { return BaseType::operator ()(); } template ReturnType operator ()(IgnoreType1&) const { return 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 ReturnType operator ()(Arg1 arg1, IgnoreType1&, IgnoreType2&) const { return BaseType::operator ()(arg1); } template ReturnType operator ()(Arg1 arg1, IgnoreType&) const { return 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 > ReturnType operator ()(Arg1 arg1, Arg2 arg2, IgnoreType&) const { return 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, typename ReturnType> class MemFunSelectArgImpl { public: MemFunSelectArgImpl(Functor func): m_func(func) { } template ReturnType operator ()(Type1& a, Type2& b, Type3& c) const { return m_func(STLUtil::SelectArg()(a, b, c)); } template ReturnType operator ()(Type1& a, Type2& b) const { return m_func(STLUtil::SelectArg()(a, b)); } template ReturnType operator ()(Type1& a) const { return 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, ReturnType> MemFunSelectArg0(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<0, MemFun1, ReturnType>(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, ReturnType> MemFunSelectArg1(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<1, MemFun1, ReturnType>(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, ReturnType> MemFunSelectArg2(Object& obj, ReturnType (Object:: *action)(Arg1)) { return MemFunSelectArgImpl<2, MemFun1, ReturnType>(MemFun(obj, action)); } /// Creates a functor with a bound parameter template class MemFunBind1 { public: typedef ReturnType (Object:: *Action)(Arg1); MemFunBind1(Object& obj, Action action, Arg1 arg1): m_obj(obj), m_action(action), m_arg1(arg1) { } ReturnType operator()() const { return (m_obj.*m_action)(m_arg1); } private: Object& m_obj; Action m_action; Arg1 m_arg1; }; /// Creates a functor with a bound parameter template MemFunBind1 MemFunBind( Object& obj, ReturnType (Object:: *action)(Arg1), Arg1 arg1 ) { return MemFunBind1(obj, action, arg1); } /// Creates a functor with a bound parameter template class MemFunBind2 { public: typedef ReturnType (Object:: *Action)(Arg1, Arg2); MemFunBind2(Object& obj, Action action, Arg1 arg1, Arg2 arg2): m_obj(obj), m_action(action), m_arg1(arg1), m_arg2(arg2) { } ReturnType operator()() const { return (m_obj.*m_action)(m_arg1, m_arg2); } private: Object& m_obj; Action m_action; Arg1 m_arg1; Arg2 m_arg2; }; /// Creates a functor with a bound parameter template MemFunBind2 MemFunBind( Object& obj, ReturnType (Object:: *action)(Arg1, Arg2), Arg1 arg1, Arg2 arg2 ) { return MemFunBind2(obj, action, arg1, arg2); } /// Creates a functor with a bound parameter template class MemFunBind3 { public: typedef ReturnType (Object:: *Action)(Arg1, Arg2, Arg3); MemFunBind3(Object& obj, Action action, Arg1 arg1, Arg2 arg2, Arg3 arg3): m_obj(obj), m_action(action), m_arg1(arg1), m_arg2(arg2), m_arg3(arg3) { } ReturnType operator()() const { return (m_obj.*m_action)(m_arg1, m_arg2, m_arg3); } private: Object& m_obj; Action m_action; Arg1 m_arg1; Arg2 m_arg2; Arg3 m_arg3; }; /// Creates a functor with a bound parameter template MemFunBind2 MemFunBind( Object& obj, ReturnType (Object:: *action)(Arg1, Arg2, Arg3), Arg1 arg1, Arg2 arg2, Arg3 arg3 ) { return MemFunBind3(obj, action, arg1, arg2, arg3); } } // namespace FbTk #endif // FBTK_MEM_FUN_HH