129 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			129 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is distributed under the BSD License.
 | |
| // See "license.txt" for details.
 | |
| // Copyright 2009-2012, Jonathan Turner (jonathan@emptycrate.com)
 | |
| // Copyright 2009-2014, Jason Turner (jason@emptycrate.com)
 | |
| // http://www.chaiscript.com
 | |
| 
 | |
| #ifndef CHAISCRIPT_FUNCTION_CALL_HPP_
 | |
| #define CHAISCRIPT_FUNCTION_CALL_HPP_
 | |
| 
 | |
| #include <functional>
 | |
| #include <iostream>
 | |
| #include <string>
 | |
| #include <vector>
 | |
| 
 | |
| #include "boxed_cast.hpp"
 | |
| #include "function_call_detail.hpp"
 | |
| #include "proxy_functions.hpp"
 | |
| 
 | |
| namespace chaiscript {
 | |
| class Boxed_Value;
 | |
| class Type_Conversions;
 | |
| namespace detail {
 | |
| template <typename T> struct Cast_Helper;
 | |
| }  // namespace detail
 | |
| }  // namespace chaiscript
 | |
| 
 | |
| namespace chaiscript
 | |
| {
 | |
|   namespace dispatch
 | |
|   {
 | |
|     /// Build a function caller that knows how to dispatch on a set of functions
 | |
|     /// example:
 | |
|     /// std::function<void (int)> f =
 | |
|     ///      build_function_caller(dispatchkit.get_function("print"));
 | |
|     /// \returns A std::function object for dispatching
 | |
|     /// \param[in] funcs the set of functions to dispatch on.
 | |
|     template<typename FunctionType>
 | |
|       std::function<FunctionType>
 | |
|       functor(const std::vector<Const_Proxy_Function> &funcs, const Type_Conversions *t_conversions)
 | |
|       {
 | |
|         FunctionType *p=nullptr;
 | |
|         return detail::build_function_caller_helper(p, funcs, t_conversions);
 | |
|       }
 | |
| 
 | |
|     /// Build a function caller for a particular Proxy_Function object
 | |
|     /// useful in the case that a function is being pass out from scripting back
 | |
|     /// into code
 | |
|     /// example:
 | |
|     /// void my_function(Proxy_Function f)
 | |
|     /// {
 | |
|     ///   std::function<void (int)> local_f =
 | |
|     ///      build_function_caller(f);
 | |
|     /// }
 | |
|     /// \returns A std::function object for dispatching
 | |
|     /// \param[in] func A function to execute.
 | |
|     template<typename FunctionType>
 | |
|       std::function<FunctionType>
 | |
|       functor(Const_Proxy_Function func, const Type_Conversions *t_conversions)
 | |
|       {
 | |
|         return functor<FunctionType>(std::vector<Const_Proxy_Function>({func}), t_conversions);
 | |
|       }
 | |
| 
 | |
|     /// Helper for automatically unboxing a Boxed_Value that contains a function object
 | |
|     /// and creating a typesafe C++ function caller from it.
 | |
|     template<typename FunctionType>
 | |
|       std::function<FunctionType>
 | |
|       functor(const Boxed_Value &bv, const Type_Conversions *t_conversions)
 | |
|       {
 | |
|         return functor<FunctionType>(boxed_cast<Const_Proxy_Function >(bv, t_conversions), t_conversions);
 | |
|       }
 | |
|   }
 | |
| 
 | |
|   namespace detail{
 | |
|     /// Cast helper to handle automatic casting to const std::function &
 | |
|     template<typename Signature>
 | |
|       struct Cast_Helper<const std::function<Signature> &>
 | |
|       {
 | |
|         typedef std::function<Signature> Result_Type;
 | |
| 
 | |
|         static Result_Type cast(const Boxed_Value &ob, const Type_Conversions *t_conversions)
 | |
|         {
 | |
|           if (ob.get_type_info().bare_equal(user_type<Const_Proxy_Function>()))
 | |
|           {
 | |
|             return dispatch::functor<Signature>(ob, t_conversions);
 | |
|           } else {
 | |
|             return Cast_Helper_Inner<const std::function<Signature> &>::cast(ob, t_conversions);
 | |
|           }
 | |
|         }
 | |
|       };
 | |
| 
 | |
|     /// Cast helper to handle automatic casting to std::function
 | |
|     template<typename Signature>
 | |
|       struct Cast_Helper<std::function<Signature> >
 | |
|       {
 | |
|         typedef std::function<Signature> Result_Type;
 | |
| 
 | |
|         static Result_Type cast(const Boxed_Value &ob, const Type_Conversions *t_conversions)
 | |
|         {
 | |
|           if (ob.get_type_info().bare_equal(user_type<Const_Proxy_Function>()))
 | |
|           {
 | |
|             return dispatch::functor<Signature>(ob, t_conversions);
 | |
|           } else {
 | |
|             return Cast_Helper_Inner<std::function<Signature> >::cast(ob, t_conversions);
 | |
|           }
 | |
|         }
 | |
|       };
 | |
| 
 | |
|     /// Cast helper to handle automatic casting to const std::function
 | |
|     template<typename Signature>
 | |
|       struct Cast_Helper<const std::function<Signature> >
 | |
|       {
 | |
|         typedef std::function<Signature> Result_Type;
 | |
| 
 | |
|         static Result_Type cast(const Boxed_Value &ob, const Type_Conversions *t_conversions)
 | |
|         {
 | |
|           if (ob.get_type_info().bare_equal(user_type<Const_Proxy_Function>()))
 | |
|           {
 | |
|             return dispatch::functor<Signature>(ob, t_conversions);
 | |
|           } else {
 | |
|             return Cast_Helper_Inner<const std::function<Signature> >::cast(ob, t_conversions);
 | |
|           }
 | |
|         }
 | |
|       };
 | |
|   }
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | 
