efl/src/bindings/js/eo_js/eo_js_constructor.hh

137 lines
4.2 KiB
C++

#ifndef EFL_EO_JS_CONSTRUCTOR_HH
#define EFL_EO_JS_CONSTRUCTOR_HH
#include <Eina.hh>
#include <eina_tuple.hh>
#include <eina_function.hh>
#include <Eo.h>
#include <cstdlib>
#include <functional>
#include <iostream>
#include <cassert>
#include <vector>
namespace efl { namespace eo { namespace js {
inline eina::js::compatibility_return_type constructor(eina::js::compatibility_callback_info_type args)
{
if(args.IsConstructCall())
{
void* data = v8::External::Cast(*args.Data())->Value();
std::function<eina::js::compatibility_return_type(eina::js::compatibility_callback_info_type)>*
f = static_cast<std::function<eina::js::compatibility_return_type(eina::js::compatibility_callback_info_type)>*>
(data);
return (*f)(args);
}
else
{
std::size_t argc = args.Length();
std::vector<v8::Local<v8::Value> > argv (argc ? argc : 1 );
for(int i = 0; i != args.Length(); ++i)
argv[i] = args[i];
args.Callee()->NewInstance(argc, &argv[0]);
return eina::js::compatibility_return();
}
}
template <typename...F>
struct constructor_caller
{
struct call
{
template <typename T>
void operator()(T function) const
{
int const parameters
= std::tuple_size<typename eina::_mpl::function_params<T>::type>::value;
if(*current + parameters <= args->Length())
{
aux(function, eina::make_index_sequence<parameters>());
*current += parameters;
}
else
{
eina::js::compatibility_throw
(v8::Exception::TypeError
(eina::js::compatibility_new<v8::String>(args->GetIsolate(), "Expected more arguments for this call")));
throw std::logic_error("");
}
}
template <typename U, std::size_t I>
static
typename std::tuple_element<I, typename eina::_mpl::function_params<U>::type>::type
get_value(v8::Local<v8::Value> v, v8::Isolate* isolate)
{
typename std::tuple_element<I, typename eina::_mpl::function_params<U>::type>::type
tmp =
eina::js::get_value_from_javascript
(v, isolate, ""
, eina::js::value_tag<typename std::tuple_element
<I, typename eina::_mpl::function_params<U>::type>::type>());
return tmp;
}
template <typename T, std::size_t... I>
void aux(T function, eina::index_sequence<I...>) const
{
function(get_value<T, I>((*args)[I + *current], args->GetIsolate())...);
}
int* current;
eina::js::compatibility_callback_info_pointer args;
};
eina::js::compatibility_return_type operator()(eina::js::compatibility_callback_info_type args) const
{
int current_index = 1;
if(args.Length() != 0)
{
try
{
Eo* parent = eina::js::get_value_from_javascript
(args[0], args.GetIsolate(), "", eina::js::value_tag<Eo*>());
Eo* eo = eo_add
(klass
, parent
, eina::_mpl::for_each(constructors, call{&current_index, &args})
);
assert(eo != 0);
v8::Local<v8::Object> self = args.This();
self->SetInternalField(0, eina::js::compatibility_new<v8::External>(args.GetIsolate(), eo));
efl::eina::js::make_weak(args.GetIsolate(), self
, [eo]
{
eo_unref(eo);
});
}
catch(std::logic_error const&) {}
}
else
{
eina::js::compatibility_throw
(v8::Exception::TypeError
(eina::js::compatibility_new<v8::String>(args.GetIsolate(), "Expected at least one argument for this call")));
}
return eina::js::compatibility_return();
}
Eo_Class const* klass;
std::tuple<F...> constructors;
};
template <typename... F>
v8::Handle<v8::Value> constructor_data(v8::Isolate* isolate, Eo_Class const* klass, F... f)
{
return eina::js::compatibility_new<v8::External>
(isolate
, new std::function<eina::js::compatibility_return_type(eina::js::compatibility_callback_info_type)>
(constructor_caller<F...>{klass, std::tuple<F...>{f...}}));
}
} } }
#endif