godot-cpp/include/core/Godot.hpp

491 lines
12 KiB
C++
Raw Normal View History

2017-05-13 11:55:04 +00:00
#ifndef GODOT_HPP
#define GODOT_HPP
2017-03-06 17:39:56 +00:00
#include <cstdlib>
2017-05-23 21:03:57 +00:00
#include <cstring>
2017-03-06 17:39:56 +00:00
2017-10-20 23:42:10 +00:00
#include <gdnative_api_struct.gen.h>
2017-10-03 10:37:34 +00:00
#include <nativescript/godot_nativescript.h>
2017-03-06 17:39:56 +00:00
#include "CoreTypes.hpp"
#include "Variant.hpp"
#include "Ref.hpp"
2017-03-06 17:39:56 +00:00
#include "Object.hpp"
#include "GodotGlobal.hpp"
2017-05-13 11:55:04 +00:00
2017-03-06 17:39:56 +00:00
namespace godot {
template<class T>
T *as(Object *obj)
{
return (T *) godot::nativescript_api->godot_nativescript_get_userdata(obj);
}
2017-05-13 11:55:04 +00:00
template<class T>
class GodotScript {
public:
T *owner;
2017-05-13 11:55:04 +00:00
// GodotScript() {}
void _init() {}
static const char *___get_base_type_name()
2017-05-13 11:55:04 +00:00
{
return T::___get_class_name();
}
static GodotScript<T> *___get_from_variant(Variant a)
{
return as<GodotScript<T> >((Object *) a);
}
2017-05-13 11:55:04 +00:00
static void _register_methods() {}
};
#define GODOT_CLASS(Name) \
public: inline static const char *___get_type_name() { return static_cast<const char *>(#Name); } \
2017-03-06 17:39:56 +00:00
private:
2017-04-06 03:05:53 +00:00
#define GODOT_SUBCLASS(Name, Base) \
public: inline static const char *___get_type_name() { return static_cast<const char *>(#Name); } \
inline static const char *___get_base_type_name() { return static_cast<const char *>(#Base); } \
2017-04-06 03:05:53 +00:00
private:
2017-03-06 17:39:56 +00:00
template<class T>
struct _ArgCast {
static T _arg_cast(Variant a)
{
return a;
}
};
template<class T>
struct _ArgCast<T*> {
static T *_arg_cast(Variant a)
{
return (T *) T::___get_from_variant(a);
}
};
2017-06-19 10:05:38 +00:00
template<>
struct _ArgCast<Variant> {
static Variant _arg_cast(Variant a)
{
return a;
}
};
2017-03-06 17:39:56 +00:00
// instance and destroy funcs
template<class T>
void *_godot_class_instance_func(godot_object *p, void *method_data)
2017-03-06 17:39:56 +00:00
{
2017-05-13 11:55:04 +00:00
T *d = new T();
*(godot_object **) &d->owner = p;
2017-03-06 17:39:56 +00:00
d->_init();
return d;
}
template<class T>
void _godot_class_destroy_func(godot_object *p, void *method_data, void *data)
2017-03-06 17:39:56 +00:00
{
T *d = (T *) data;
delete d;
}
template<class T>
void register_class()
{
godot_instance_create_func create = {};
create.create_func = _godot_class_instance_func<T>;
godot_instance_destroy_func destroy = {};
destroy.destroy_func = _godot_class_destroy_func<T>;
godot::nativescript_api->godot_nativescript_register_class(godot::_RegisterState::nativescript_handle, T::___get_type_name(), T::___get_base_type_name(), create, destroy);
2017-03-06 17:39:56 +00:00
T::_register_methods();
}
2017-03-18 18:01:11 +00:00
template<class T>
void register_tool_class()
{
godot_instance_create_func create = {};
create.create_func = _godot_class_instance_func<T>;
godot_instance_destroy_func destroy = {};
destroy.destroy_func = _godot_class_destroy_func<T>;
godot::nativescript_api->godot_nativescript_register_tool_class(godot::_RegisterState::nativescript_handle, T::___get_type_name(), T::___get_base_type_name(), create, destroy);
2017-03-18 18:01:11 +00:00
T::_register_methods();
}
2017-03-06 17:39:56 +00:00
// method registering
typedef godot_variant (*__godot_wrapper_method)(godot_object *, void *, void *, int, godot_variant **);
template<class T, class R, class ...args>
const char *___get_method_class_name(R (T::*p)(args... a))
{
return T::___get_type_name();
}
template<class T, class R, class ...args>
const char *___get_method_class_name(R (T::*p)(args... a) const)
{
return T::___get_type_name();
}
2017-07-23 14:13:55 +00:00
// Okay, time for some template magic.
// Many thanks to manpat from the GDL Discord Server.
2017-07-23 14:13:55 +00:00
// This is stuff that's available in C++14 I think, but whatever.
2017-07-23 14:13:55 +00:00
template<int... I>
struct __Sequence{};
2017-07-23 14:13:55 +00:00
template<int N, int... I>
struct __construct_sequence {
using type = typename __construct_sequence<N-1, N-1, I...>::type;
2017-03-06 17:39:56 +00:00
};
2017-07-23 14:13:55 +00:00
template<int... I>
struct __construct_sequence<0, I...> {
using type = __Sequence<I...>;
2017-03-06 17:39:56 +00:00
};
2017-07-23 14:13:55 +00:00
// Now the wrapping part.
template<class T, class R, class... As>
struct _WrappedMethod {
R (T::*f)(As...);
2017-07-23 14:13:55 +00:00
template<int... I>
void apply(Variant* ret, T* obj, Variant** args, __Sequence<I...>) {
*ret = (obj->*f)( _ArgCast<As>::_arg_cast(*args[I])... );
2017-03-06 17:39:56 +00:00
}
};
2017-07-23 14:13:55 +00:00
template<class T, class... As>
struct _WrappedMethod<T, void, As...> {
void (T::*f)(As...);
2017-07-23 14:13:55 +00:00
template<int... I>
void apply(Variant* ret, T* obj, Variant** args, __Sequence<I...>) {
(obj->*f)( _ArgCast<As>::_arg_cast(*args[I])... );
2017-03-06 17:39:56 +00:00
}
};
2017-07-23 14:13:55 +00:00
template<class T, class R, class... As>
godot_variant __wrapped_method(godot_object *, void *method_data, void *user_data, int num_args, godot_variant **args)
{
2017-07-23 14:13:55 +00:00
godot_variant v;
2017-10-20 23:42:10 +00:00
godot::api->godot_variant_new_nil(&v);
2017-07-23 14:13:55 +00:00
T *obj = (T *) user_data;
_WrappedMethod<T, R, As...> *method = (_WrappedMethod<T, R, As...>*) method_data;
2017-07-23 14:13:55 +00:00
Variant *var = (Variant *) &v;
Variant **arg = (Variant **) args;
2017-07-23 14:13:55 +00:00
method->apply(var, obj, arg, typename __construct_sequence<sizeof...(As)>::type {});
2017-07-23 14:13:55 +00:00
return v;
}
2017-07-23 14:13:55 +00:00
template<class T, class R, class... As>
void *___make_wrapper_function(R (T::*f)(As...))
{
2017-07-23 14:13:55 +00:00
using MethodType = _WrappedMethod<T, R, As...>;
2017-10-20 23:42:10 +00:00
MethodType *p = (MethodType *) godot::api->godot_alloc(sizeof(MethodType));
p->f = f;
return (void *) p;
}
2017-07-23 14:13:55 +00:00
template<class T, class R, class... As>
__godot_wrapper_method ___get_wrapper_function(R (T::*f)(As...))
{
2017-07-23 14:13:55 +00:00
return (__godot_wrapper_method) &__wrapped_method<T, R, As...>;
}
template<class T, class R, class ...A>
void *___make_wrapper_function(R (T::*f)(A...) const)
{
return ___make_wrapper_function((R (T::*)(A...)) f);
}
2017-03-06 17:39:56 +00:00
template<class T, class R, class ...A>
__godot_wrapper_method ___get_wrapper_function(R (T::*f)(A...) const)
{
return ___get_wrapper_function((R (T::*)(A...)) f);
}
2017-03-06 17:39:56 +00:00
template<class M>
void register_method(const char *name, M method_ptr, godot_method_rpc_mode rpc_type = GODOT_METHOD_RPC_MODE_DISABLED)
{
godot_instance_method method = {};
method.method_data = ___make_wrapper_function(method_ptr);
2017-10-20 23:42:10 +00:00
method.free_func = godot::api->godot_free;
method.method = (__godot_wrapper_method) ___get_wrapper_function(method_ptr);
godot_method_attributes attr = {};
attr.rpc_type = rpc_type;
godot::nativescript_api->godot_nativescript_register_method(godot::_RegisterState::nativescript_handle, ___get_method_class_name(method_ptr), name, attr, method);
}
2017-03-18 15:58:03 +00:00
template<class T, class P>
struct _PropertySetFunc {
void (T::*f)(P);
static void _wrapped_setter(godot_object *object, void *method_data, void *user_data, godot_variant *value)
2017-03-18 15:58:03 +00:00
{
_PropertySetFunc<T, P> *set_func = (_PropertySetFunc<T, P> *) method_data;
T *obj = (T *) user_data;
Variant *v = (Variant *) value;
2017-03-18 15:58:03 +00:00
(obj->*(set_func->f))(_ArgCast<P>::_arg_cast(*v));
2017-03-18 15:58:03 +00:00
}
};
template<class T, class P>
struct _PropertyGetFunc {
P (T::*f)();
static godot_variant _wrapped_getter(godot_object *object, void *method_data, void *user_data)
{
_PropertyGetFunc<T, P> *get_func = (_PropertyGetFunc<T, P> *) method_data;
T *obj = (T *) user_data;
godot_variant var;
2017-10-20 23:42:10 +00:00
godot::api->godot_variant_new_nil(&var);
2017-03-18 15:58:03 +00:00
Variant *v = (Variant *) &var;
*v = (obj->*(get_func->f))();
return var;
}
};
template<class T, class P>
struct _PropertyDefaultSetFunc {
P (T::*f);
static void _wrapped_setter(godot_object *object, void *method_data, void *user_data, godot_variant *value)
{
_PropertyDefaultSetFunc<T, P> *set_func = (_PropertyDefaultSetFunc<T, P> *) method_data;
T *obj = (T *) user_data;
Variant *v = (Variant *) value;
(obj->*(set_func->f)) = _ArgCast<P>::_arg_cast(*v);
}
};
template<class T, class P>
struct _PropertyDefaultGetFunc {
P (T::*f);
static godot_variant _wrapped_getter(godot_object *object, void *method_data, void *user_data)
{
_PropertyDefaultGetFunc<T, P> *get_func = (_PropertyDefaultGetFunc<T, P> *) method_data;
T *obj = (T *) user_data;
godot_variant var;
2017-10-20 23:42:10 +00:00
godot::api->godot_variant_new_nil(&var);
Variant *v = (Variant *) &var;
*v = (obj->*(get_func->f));
return var;
}
};
template<class T, class P>
void register_property(const char *name, P (T::*var), P default_value, godot_method_rpc_mode rpc_mode = GODOT_METHOD_RPC_MODE_DISABLED, godot_property_usage_flags usage = GODOT_PROPERTY_USAGE_DEFAULT, godot_property_hint hint = GODOT_PROPERTY_HINT_NONE, String hint_string = "")
{
Variant def_val = default_value;
usage = (godot_property_usage_flags) ((int) usage | GODOT_PROPERTY_USAGE_SCRIPT_VARIABLE);
if (def_val.get_type() == Variant::OBJECT) {
Object *o = def_val;
if (o && o->is_class("Resource")) {
hint = (godot_property_hint) ((int) hint | GODOT_PROPERTY_HINT_RESOURCE_TYPE);
hint_string = o->get_class();
}
}
godot_string *_hint_string = (godot_string*) &hint_string;
godot_property_attributes attr = {};
attr.type = def_val.get_type();
attr.default_value = *(godot_variant *) &def_val;
attr.hint = hint;
attr.rset_type = rpc_mode;
attr.usage = usage;
attr.hint_string = *_hint_string;
2017-10-20 23:42:10 +00:00
_PropertyDefaultSetFunc<T, P> *wrapped_set = (_PropertyDefaultSetFunc<T, P> *)godot::api->godot_alloc(sizeof(_PropertyDefaultSetFunc<T, P>));
wrapped_set->f = var;
2017-10-20 23:42:10 +00:00
_PropertyDefaultGetFunc<T, P> *wrapped_get = (_PropertyDefaultGetFunc<T, P> *) godot::api->godot_alloc(sizeof(_PropertyDefaultGetFunc<T, P>));
wrapped_get->f = var;
godot_property_set_func set_func = {};
set_func.method_data = (void *) wrapped_set;
2017-10-20 23:42:10 +00:00
set_func.free_func = godot::api->godot_free;
set_func.set_func = &_PropertyDefaultSetFunc<T, P>::_wrapped_setter;
godot_property_get_func get_func = {};
get_func.method_data = (void *) wrapped_get;
2017-10-20 23:42:10 +00:00
get_func.free_func = godot::api->godot_free;
get_func.get_func = &_PropertyDefaultGetFunc<T, P>::_wrapped_getter;
godot::nativescript_api->godot_nativescript_register_property(godot::_RegisterState::nativescript_handle, T::___get_type_name(), name, &attr, set_func, get_func);
}
2017-03-18 15:58:03 +00:00
template<class T, class P>
void register_property(const char *name, void (T::*setter)(P), P (T::*getter)(), P default_value, godot_method_rpc_mode rpc_mode = GODOT_METHOD_RPC_MODE_DISABLED, godot_property_usage_flags usage = GODOT_PROPERTY_USAGE_DEFAULT, godot_property_hint hint = GODOT_PROPERTY_HINT_NONE, String hint_string = "")
2017-03-18 15:58:03 +00:00
{
Variant def_val = default_value;
godot_property_attributes attr = {};
attr.type = def_val.get_type();
attr.default_value = *(godot_variant *) &def_val;
attr.hint = hint;
attr.rset_type = rpc_mode;
attr.usage = usage;
2017-10-20 23:42:10 +00:00
_PropertySetFunc<T, P> *wrapped_set = (_PropertySetFunc<T, P> *) godot::api->godot_alloc(sizeof(_PropertySetFunc<T, P>));
2017-03-18 15:58:03 +00:00
wrapped_set->f = setter;
2017-10-20 23:42:10 +00:00
_PropertyGetFunc<T, P> *wrapped_get = (_PropertyGetFunc<T, P> *) godot::api->godot_alloc(sizeof(_PropertyGetFunc<T, P>));
2017-03-18 15:58:03 +00:00
wrapped_get->f = getter;
godot_property_set_func set_func = {};
set_func.method_data = (void *) wrapped_set;
2017-10-20 23:42:10 +00:00
set_func.free_func = godot::api->godot_free;
2017-03-18 15:58:03 +00:00
set_func.set_func = &_PropertySetFunc<T, P>::_wrapped_setter;
godot_property_get_func get_func = {};
get_func.method_data = (void *) wrapped_get;
2017-10-20 23:42:10 +00:00
get_func.free_func = godot::api->godot_free;
2017-03-18 15:58:03 +00:00
get_func.get_func = &_PropertyGetFunc<T, P>::_wrapped_getter;
godot::nativescript_api->godot_nativescript_register_property(godot::_RegisterState::nativescript_handle, T::___get_type_name(), name, &attr, set_func, get_func);
2017-03-18 15:58:03 +00:00
}
2018-01-19 10:49:28 +00:00
template<class T, class P>
void register_property(const char *name, void (T::*setter)(P), P (T::*getter)() const, P default_value, godot_method_rpc_mode rpc_mode = GODOT_METHOD_RPC_MODE_DISABLED, godot_property_usage_flags usage = GODOT_PROPERTY_USAGE_DEFAULT, godot_property_hint hint = GODOT_PROPERTY_HINT_NONE, String hint_string = "")
{
register_property(name, setter, (P (T::*)()) getter, default_value, rpc_mode, usage, hint, hint_string);
}
template<class T>
void register_signal(String name, Dictionary args = Dictionary())
{
godot_signal signal = {};
signal.name = *(godot_string *)&name;
signal.num_args = args.size();
signal.num_default_args = 0;
2018-01-17 01:00:55 +00:00
// Need to check because malloc(0) is platform-dependent. Zero arguments will leave args to NULL.
if(signal.num_args != 0) {
signal.args = (godot_signal_argument*) godot::api->godot_alloc(sizeof(godot_signal_argument) * signal.num_args);
memset((void *) signal.args, 0, sizeof(godot_signal_argument) * signal.num_args);
}
for (int i = 0; i < signal.num_args; i++) {
// Array entry = args[i];
// String name = entry[0];
String name = args.keys()[i];
godot_string *_key = (godot_string *)&name;
2017-10-20 23:42:10 +00:00
godot::api->godot_string_new_copy(&signal.args[i].name, _key);
// if (entry.size() > 1) {
// signal.args[i].type = entry[1];
// }
signal.args[i].type = args.values()[i];
}
godot::nativescript_api->godot_nativescript_register_signal(godot::_RegisterState::nativescript_handle, T::___get_type_name(), &signal);
for (int i = 0; i < signal.num_args; i++) {
2017-10-20 23:42:10 +00:00
godot::api->godot_string_destroy(&signal.args[i].name);
}
2018-01-17 01:00:55 +00:00
if(signal.args) {
godot::api->godot_free(signal.args);
}
}
2017-03-06 17:39:56 +00:00
}
#endif // GODOT_H