[gjs/wip/carlosg/arrays-in-signals] WIP: [RFC] value: Introspect element-type in arrays
- From: Carlos Garnacho <carlosg src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gjs/wip/carlosg/arrays-in-signals] WIP: [RFC] value: Introspect element-type in arrays
- Date: Mon, 15 Feb 2021 22:58:38 +0000 (UTC)
commit 0115ee0ba8a34b0851065538a70da8a0f934d284
Author: Carlos Garnacho <carlosg gnome org>
Date: Mon Feb 15 23:51:53 2021 +0100
WIP: [RFC] value: Introspect element-type in arrays
A few things have to combine ATM here:
- Of all compound types, it has to be an array
- Of all element-types it has to be a struct/union/boxed
It will succeed then, fail otherwise.
Fixes: https://gitlab.gnome.org/GNOME/gjs/-/issues/377
gi/value.cpp | 62 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
1 file changed, 62 insertions(+)
---
diff --git a/gi/value.cpp b/gi/value.cpp
index 26a679f3..768ec127 100644
--- a/gi/value.cpp
+++ b/gi/value.cpp
@@ -11,6 +11,7 @@
#include <glib-object.h>
#include <glib.h>
+#include <js/Array.h>
#include <js/CharacterEncoding.h>
#include <js/Conversions.h>
#include <js/GCVector.h> // for RootedVector
@@ -840,6 +841,67 @@ gjs_value_from_g_value_internal(JSContext *context,
gjs_throw(context, "Failed to convert strv to array");
return false;
}
+ } else if (g_type_is_a(gtype, G_TYPE_PTR_ARRAY)) {
+ GIArgInfo *arg_info;
+ GISignalInfo *signal_info;
+ GITypeInfo type_info;
+
+ signal_info = get_signal_info_if_available(signal_query);
+ if (!signal_info) {
+ gjs_throw(context, "Unknown signal.");
+ return false;
+ }
+
+ /* Look for element-type */
+ arg_info = g_callable_info_get_arg(signal_info, arg_n - 1);
+ g_arg_info_load_type(arg_info, &type_info);
+
+ GITypeInfo *element_info = g_type_info_get_param_type (&type_info, 0);
+ GjsAutoBaseInfo interface_info =
+ g_type_info_get_interface(element_info);
+ GType gtype;
+
+ guint i;
+ JS::RootedValueVector elems(context);
+ GPtrArray *ptr_array = (GPtrArray*) g_value_get_boxed (gvalue);
+
+ for (i = 0; i < ptr_array->len; i++) {
+ if (!elems.growBy(1)) {
+ JS_ReportOutOfMemory(context);
+ return false;
+ }
+
+ switch (g_base_info_get_type(interface_info)) {
+ case GI_INFO_TYPE_STRUCT:
+ case GI_INFO_TYPE_UNION:
+ case GI_INFO_TYPE_BOXED:
+ gtype = g_registered_type_info_get_g_type(interface_info);
+ GjsAutoBaseInfo info = g_irepository_find_by_gtype(nullptr, gtype);
+ JSObject *obj;
+
+ obj = BoxedInstance::new_for_c_struct(context, info, g_ptr_array_index (ptr_array, i));
+ if (!obj)
+ return false;
+ elems[i].setObject(*obj);
+ break;
+ case GI_INFO_TYPE_ENUM:
+ case GI_INFO_TYPE_FLAGS:
+ case GI_INFO_TYPE_OBJECT:
+ case GI_INFO_TYPE_INTERFACE:
+ case GI_INFO_TYPE_VALUE:
+ default:
+ gjs_throw(context,
+ "Unable to introspect non-GTyped array");
+ return false;
+ }
+ }
+
+ JS::RootedObject obj(context, JS::NewArrayObject(context, elems));
+ if (!obj)
+ return false;
+
+ value_p.setObject(*obj);
+ return true;
} else if (g_type_is_a(gtype, G_TYPE_HASH_TABLE) ||
g_type_is_a(gtype, G_TYPE_ARRAY) ||
g_type_is_a(gtype, G_TYPE_BYTE_ARRAY) ||
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]