Make better version of param list builder and make all function related concepts variadic templates via boost::pp

This commit is contained in:
Jason Turner
2009-05-26 19:03:09 +00:00
parent 1980ba840c
commit 8138b19390
3 changed files with 90 additions and 91 deletions

View File

@@ -1,29 +1,19 @@
#include <boost/preprocessor.hpp>
#ifndef BOOST_PP_IS_ITERATING
#ifndef __scripting_constructors_hpp__ #ifndef __scripting_constructors_hpp__
#define __scripting_constrcutors_hpp__ #define __scripting_constructors_hpp__
#include <boost/shared_ptr.hpp> #include <boost/shared_ptr.hpp>
#include <boost/bind.hpp> #include <boost/bind.hpp>
#include <boost/function.hpp> #include <boost/function.hpp>
//Constructors, to be made variadic
template<typename Class> template<typename Class>
boost::shared_ptr<Class> constructor() boost::shared_ptr<Class> constructor()
{ {
return boost::shared_ptr<Class>(new Class()); return boost::shared_ptr<Class>(new Class());
} }
template<typename Class, typename Param1>
boost::shared_ptr<Class> constructor(Param1 p1)
{
return boost::shared_ptr<Class>(new Class(p1));
}
template<typename Class, typename Param1, typename Param2>
boost::shared_ptr<Class> constructor(Param1 p1, Param2 p2)
{
return boost::shared_ptr<Class>(new Class(p1, p2));
}
template<typename Class> template<typename Class>
boost::function<boost::shared_ptr<Class> ()> build_constructor() boost::function<boost::shared_ptr<Class> ()> build_constructor()
{ {
@@ -31,20 +21,26 @@ boost::function<boost::shared_ptr<Class> ()> build_constructor()
return boost::function<boost::shared_ptr<Class> ()>(func(&(constructor<Class>))); return boost::function<boost::shared_ptr<Class> ()>(func(&(constructor<Class>)));
} }
template<typename Class, typename Param1> #define BOOST_PP_ITERATION_LIMITS ( 1, 10 )
boost::function<boost::shared_ptr<Class> (Param1)> build_constructor() #define BOOST_PP_FILENAME_1 "scripting_constructors.hpp"
#include BOOST_PP_ITERATE()
# endif
#else
# define n BOOST_PP_ITERATION()
template<typename Class, BOOST_PP_ENUM_PARAMS(n, typename Param) >
boost::shared_ptr<Class> constructor( BOOST_PP_ENUM_BINARY_PARAMS(n, Param, p) )
{ {
typedef boost::shared_ptr<Class> (*func)(Param1); return boost::shared_ptr<Class>(new Class( BOOST_PP_ENUM_PARAMS(n, p) ));
return boost::function<boost::shared_ptr<Class> (Param1)>(func(&(constructor<Class, Param1>)));
} }
template<typename Class, typename Param1, typename Param2> template<typename Class, BOOST_PP_ENUM_PARAMS(n, typename Param) >
boost::function<boost::shared_ptr<Class> (Param1, Param2)> build_constructor() boost::function<boost::shared_ptr<Class> (BOOST_PP_ENUM_PARAMS(n, Param))> build_constructor()
{ {
typedef boost::shared_ptr<Class> (*func)(Param1, Param2); typedef boost::shared_ptr<Class> (*func)(BOOST_PP_ENUM_PARAMS(n, Param));
return boost::function<boost::shared_ptr<Class> (Param1, Param2)>(func(&(constructor<Class, Param1, Param2>))); return boost::function<boost::shared_ptr<Class> (BOOST_PP_ENUM_PARAMS(n, Param))>(func(&(constructor<Class, BOOST_PP_ENUM_PARAMS(n, Param)>)));
} }
#endif #endif

View File

@@ -1,6 +1,12 @@
#ifndef __scripting_function_hpp__ #include <boost/preprocessor.hpp>
#define __scripting_function_hpp__
#define gettypeinfo(z,n,text) ti.push_back(Get_Type_Info<Param ## n>()());
#define casthelper(z,n,text) ,Cast_Helper<Param ## n>()(params[n])
#ifndef BOOST_PP_IS_ITERATING
#ifndef __scripting_functions_hpp__
#define __scripting_functions_hpp__
#include "scripting_object.hpp" #include "scripting_object.hpp"
#include "scripting_type_info.hpp" #include "scripting_type_info.hpp"
@@ -10,29 +16,6 @@
#include <stdexcept> #include <stdexcept>
#include <vector> #include <vector>
template<typename Ret>
std::vector<Type_Info> build_param_type_list(const boost::function<Ret ()> &f)
{
return std::vector<Type_Info>();
}
template<typename Ret, typename Param1>
std::vector<Type_Info> build_param_type_list(const boost::function<Ret (Param1)> &f)
{
std::vector<Type_Info> ti;
ti.push_back(Get_Type_Info<Param1>()());
return ti;
}
template<typename Ret, typename Param1, typename Param2>
std::vector<Type_Info> build_param_type_list(const boost::function<Ret (Param1, Param2)> &f)
{
std::vector<Type_Info> ti;
ti.push_back(Get_Type_Info<Param1>()());
ti.push_back(Get_Type_Info<Param2>()());
return ti;
}
// handle_return implementations // handle_return implementations
template<typename Ret> template<typename Ret>
struct Handle_Return struct Handle_Return
@@ -63,31 +46,16 @@ struct Handle_Return<void>
}; };
// Build param type list (variadic)
// call_func implementations todo: handle reference return types template<typename Ret>
// to be made variadic std::vector<Type_Info> build_param_type_list(const boost::function<Ret ()> &f)
template<typename Ret, typename Param1, typename Param2>
Scripting_Object call_func(const boost::function<Ret (Param1, Param2)> &f, const std::vector<Scripting_Object> &params)
{ {
if (params.size() != 2) std::vector<Type_Info> ti;
{ ti.push_back(Get_Type_Info<Ret>()());
throw std::range_error("Incorrect number of parameters"); return ti;
} else {
return Handle_Return<Ret>()(boost::bind(f, Cast_Helper<Param1>()(params[0]), Cast_Helper<Param2>()(params[1])));
}
}
template<typename Ret, typename Param1>
Scripting_Object call_func(const boost::function<Ret (Param1)> &f, const std::vector<Scripting_Object> &params)
{
if (params.size() != 1)
{
throw std::range_error("Incorrect number of parameters");
} else {
return Handle_Return<Ret>()(boost::bind(f, Cast_Helper<Param1>()(params[0])));
}
} }
// call_func implementations (variadic)
template<typename Ret> template<typename Ret>
Scripting_Object call_func(const boost::function<Ret ()> &f, const std::vector<Scripting_Object> &params) Scripting_Object call_func(const boost::function<Ret ()> &f, const std::vector<Scripting_Object> &params)
{ {
@@ -100,6 +68,34 @@ Scripting_Object call_func(const boost::function<Ret ()> &f, const std::vector<S
} }
struct Param_List_Builder
{
Param_List_Builder &operator<<(const Scripting_Object &so)
{
objects.push_back(so);
return *this;
}
template<typename T>
Param_List_Builder &operator<<(T t)
{
objects.push_back(Scripting_Object(t));
return *this;
}
operator const std::vector<Scripting_Object> &() const
{
return objects;
}
std::vector<Scripting_Object> objects;
};
#define BOOST_PP_ITERATION_LIMITS ( 1, 10 )
#define BOOST_PP_FILENAME_1 "scripting_functions.hpp"
#include BOOST_PP_ITERATE()
class Function_Handler class Function_Handler
{ {
@@ -132,26 +128,33 @@ class Function_Handler_Impl : public Function_Handler
Func m_f; Func m_f;
}; };
std::vector<Scripting_Object> build_param_list(const Scripting_Object &so)
# endif
#else
# define n BOOST_PP_ITERATION()
template<typename Ret, BOOST_PP_ENUM_PARAMS(n, typename Param) >
std::vector<Type_Info> build_param_type_list(const boost::function<Ret (BOOST_PP_ENUM_PARAMS(n, Param))> &f)
{ {
std::vector<Scripting_Object> sos; std::vector<Type_Info> ti;
sos.push_back(so); ti.push_back(Get_Type_Info<Ret>()());
return sos;
BOOST_PP_REPEAT(n, gettypeinfo, ~)
return ti;
} }
std::vector<Scripting_Object> build_param_list(const Scripting_Object &so1, const Scripting_Object &so2)
template<typename Ret, BOOST_PP_ENUM_PARAMS(n, typename Param)>
Scripting_Object call_func(const boost::function<Ret (BOOST_PP_ENUM_PARAMS(n, Param))> &f,
const std::vector<Scripting_Object> &params)
{ {
std::vector<Scripting_Object> sos; if (params.size() != n)
sos.push_back(so1); {
sos.push_back(so2); throw std::range_error("Incorrect number of parameters");
return sos; } else {
return Handle_Return<Ret>()(boost::bind(f BOOST_PP_REPEAT(n, casthelper, ~)));
} }
std::vector<Scripting_Object> build_param_list(const Scripting_Object &so1, const Scripting_Object &so2, const Scripting_Object &so3)
{
std::vector<Scripting_Object> sos;
sos.push_back(so1);
sos.push_back(so2);
sos.push_back(so3);
return sos;
} }
#endif #endif

View File

@@ -78,10 +78,10 @@ int main()
(*method1)(sos); (*method1)(sos);
Scripting_Object o = (*method1)(sos); Scripting_Object o = (*method1)(sos);
(*ss.get_function("print"))(build_param_list(ss.get_object("str"))); (*ss.get_function("print"))(Param_List_Builder() << ss.get_object("str"));
//Add new dynamically created object from registered "Test" constructor //Add new dynamically created object from registered "Test" constructor
ss.add_object("testobj2", (*ss.get_function("Test"))(build_param_list(Scripting_Object(std::string("Yo"))))); ss.add_object("testobj2", (*ss.get_function("Test"))(Param_List_Builder() << std::string("Yo")));
std::cout << Cast_Helper<int>()(o) << std::endl; std::cout << Cast_Helper<int>()(o) << std::endl;