/* GObject - GLib Type, Object, Parameter and Signal Library * * Copyright (C) 2015-2022 Christian Hergert * Copyright (C) 2015 Garrett Regier * * SPDX-License-Identifier: LGPL-2.1-or-later * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General * Public License along with this library; if not, see . * * SPDX-License-Identifier: LGPL-2.1-or-later */ #include /* Copied from glib */ typedef struct _BindingSource { GObject parent_instance; gint foo; gint bar; gdouble value; gboolean toggle; } BindingSource; typedef struct _BindingSourceClass { GObjectClass parent_class; } BindingSourceClass; enum { PROP_SOURCE_FOO = 1, PROP_SOURCE_BAR, PROP_SOURCE_VALUE, PROP_SOURCE_TOGGLE }; static GType binding_source_get_type (void); G_DEFINE_TYPE (BindingSource, binding_source, G_TYPE_OBJECT); static void binding_source_set_property (GObject *gobject, guint prop_id, const GValue *value, GParamSpec *pspec) { BindingSource *source = (BindingSource *) gobject; switch (prop_id) { case PROP_SOURCE_FOO: source->foo = g_value_get_int (value); break; case PROP_SOURCE_BAR: source->bar = g_value_get_int (value); break; case PROP_SOURCE_VALUE: source->value = g_value_get_double (value); break; case PROP_SOURCE_TOGGLE: source->toggle = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); } } static void binding_source_get_property (GObject *gobject, guint prop_id, GValue *value, GParamSpec *pspec) { BindingSource *source = (BindingSource *) gobject; switch (prop_id) { case PROP_SOURCE_FOO: g_value_set_int (value, source->foo); break; case PROP_SOURCE_BAR: g_value_set_int (value, source->bar); break; case PROP_SOURCE_VALUE: g_value_set_double (value, source->value); break; case PROP_SOURCE_TOGGLE: g_value_set_boolean (value, source->toggle); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); } } static void binding_source_class_init (BindingSourceClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->set_property = binding_source_set_property; gobject_class->get_property = binding_source_get_property; g_object_class_install_property (gobject_class, PROP_SOURCE_FOO, g_param_spec_int ("foo", "Foo", "Foo", -1, 100, 0, G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, PROP_SOURCE_BAR, g_param_spec_int ("bar", "Bar", "Bar", -1, 100, 0, G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, PROP_SOURCE_VALUE, g_param_spec_double ("value", "Value", "Value", -100.0, 200.0, 0.0, G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, PROP_SOURCE_TOGGLE, g_param_spec_boolean ("toggle", "Toggle", "Toggle", FALSE, G_PARAM_READWRITE)); } static void binding_source_init (BindingSource *self) { } typedef struct _BindingTarget { GObject parent_instance; gint bar; gdouble value; gboolean toggle; } BindingTarget; typedef struct _BindingTargetClass { GObjectClass parent_class; } BindingTargetClass; enum { PROP_TARGET_BAR = 1, PROP_TARGET_VALUE, PROP_TARGET_TOGGLE }; static GType binding_target_get_type (void); G_DEFINE_TYPE (BindingTarget, binding_target, G_TYPE_OBJECT); static void binding_target_set_property (GObject *gobject, guint prop_id, const GValue *value, GParamSpec *pspec) { BindingTarget *target = (BindingTarget *) gobject; switch (prop_id) { case PROP_TARGET_BAR: target->bar = g_value_get_int (value); break; case PROP_TARGET_VALUE: target->value = g_value_get_double (value); break; case PROP_TARGET_TOGGLE: target->toggle = g_value_get_boolean (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); } } static void binding_target_get_property (GObject *gobject, guint prop_id, GValue *value, GParamSpec *pspec) { BindingTarget *target = (BindingTarget *) gobject; switch (prop_id) { case PROP_TARGET_BAR: g_value_set_int (value, target->bar); break; case PROP_TARGET_VALUE: g_value_set_double (value, target->value); break; case PROP_TARGET_TOGGLE: g_value_set_boolean (value, target->toggle); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec); } } static void binding_target_class_init (BindingTargetClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); gobject_class->set_property = binding_target_set_property; gobject_class->get_property = binding_target_get_property; g_object_class_install_property (gobject_class, PROP_TARGET_BAR, g_param_spec_int ("bar", "Bar", "Bar", -1, 100, 0, G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, PROP_TARGET_VALUE, g_param_spec_double ("value", "Value", "Value", -100.0, 200.0, 0.0, G_PARAM_READWRITE)); g_object_class_install_property (gobject_class, PROP_TARGET_TOGGLE, g_param_spec_boolean ("toggle", "Toggle", "Toggle", FALSE, G_PARAM_READWRITE)); } static void binding_target_init (BindingTarget *self) { } static gboolean celsius_to_fahrenheit (GBinding *binding, const GValue *from_value, GValue *to_value, gpointer user_data G_GNUC_UNUSED) { gdouble celsius, fahrenheit; g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE)); g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE)); celsius = g_value_get_double (from_value); fahrenheit = (9 * celsius / 5) + 32.0; if (g_test_verbose ()) g_printerr ("Converting %.2fC to %.2fF\n", celsius, fahrenheit); g_value_set_double (to_value, fahrenheit); return TRUE; } static gboolean fahrenheit_to_celsius (GBinding *binding, const GValue *from_value, GValue *to_value, gpointer user_data G_GNUC_UNUSED) { gdouble celsius, fahrenheit; g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE)); g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE)); fahrenheit = g_value_get_double (from_value); celsius = 5 * (fahrenheit - 32.0) / 9; if (g_test_verbose ()) g_printerr ("Converting %.2fF to %.2fC\n", fahrenheit, celsius); g_value_set_double (to_value, celsius); return TRUE; } static void test_binding_group_invalid (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); /* Invalid Target Property */ g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL, "*find_property*target_property*!=*NULL*"); g_binding_group_bind (group, "value", target, "does-not-exist", G_BINDING_DEFAULT); g_test_assert_expected_messages (); g_binding_group_set_source (group, NULL); /* Invalid Source Property */ g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL, "*find_property*source_property*!=*NULL*"); g_binding_group_set_source (group, source); g_binding_group_bind (group, "does-not-exist", target, "value", G_BINDING_DEFAULT); g_test_assert_expected_messages (); g_binding_group_set_source (group, NULL); /* Invalid Source */ g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL, "*find_property*->source_property*!=*NULL*"); g_binding_group_bind (group, "does-not-exist", target, "value", G_BINDING_DEFAULT); g_binding_group_set_source (group, source); g_test_assert_expected_messages (); g_object_unref (target); g_object_unref (source); g_object_unref (group); } static void test_binding_group_default (void) { gsize i, j; GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *targets[5]; BindingSource *readback; for (i = 0; i < G_N_ELEMENTS (targets); ++i) { targets[i] = g_object_new (binding_target_get_type (), NULL); g_binding_group_bind (group, "foo", targets[i], "bar", G_BINDING_DEFAULT); } g_assert_null (g_binding_group_dup_source (group)); g_binding_group_set_source (group, source); readback = g_binding_group_dup_source (group); g_assert_true (readback == source); g_object_unref (readback); for (i = 0; i < 2; ++i) { g_object_set (source, "foo", 42, NULL); for (j = 0; j < G_N_ELEMENTS (targets); ++j) g_assert_cmpint (source->foo, ==, targets[j]->bar); g_object_set (targets[0], "bar", 47, NULL); g_assert_cmpint (source->foo, !=, targets[0]->bar); /* Check that we transition the source correctly */ g_binding_group_set_source (group, NULL); g_assert_null (g_binding_group_dup_source (group)); g_binding_group_set_source (group, source); readback = g_binding_group_dup_source (group); g_assert_true (readback == source); g_object_unref (readback); } g_object_unref (group); g_object_set (source, "foo", 0, NULL); for (i = 0; i < G_N_ELEMENTS (targets); ++i) g_assert_cmpint (source->foo, !=, targets[i]->bar); g_object_unref (source); for (i = 0; i < G_N_ELEMENTS (targets); ++i) g_object_unref (targets[i]); } static void test_binding_group_bidirectional (void) { gsize i, j; GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *targets[5]; BindingSource *readback; for (i = 0; i < G_N_ELEMENTS (targets); ++i) { targets[i] = g_object_new (binding_target_get_type (), NULL); g_binding_group_bind (group, "value", targets[i], "value", G_BINDING_BIDIRECTIONAL); } g_assert_null (g_binding_group_dup_source (group)); g_binding_group_set_source (group, source); readback = g_binding_group_dup_source (group); g_assert_true (readback == source); g_object_unref (readback); for (i = 0; i < 2; ++i) { g_object_set (source, "value", 42.0, NULL); for (j = 0; j < G_N_ELEMENTS (targets); ++j) g_assert_cmpfloat (source->value, ==, targets[j]->value); g_object_set (targets[0], "value", 47.0, NULL); g_assert_cmpfloat (source->value, ==, targets[0]->value); /* Check that we transition the source correctly */ g_binding_group_set_source (group, NULL); g_assert_null (g_binding_group_dup_source (group)); g_binding_group_set_source (group, source); readback = g_binding_group_dup_source (group); g_assert_true (readback == source); g_object_unref (readback); } g_object_unref (group); g_object_set (targets[0], "value", 0.0, NULL); g_assert_cmpfloat (source->value, !=, targets[0]->value); g_object_unref (source); for (i = 0; i < G_N_ELEMENTS (targets); ++i) g_object_unref (targets[i]); } static void transform_destroy_notify (gpointer data) { gboolean *transform_destroy_called = data; *transform_destroy_called = TRUE; } static void test_binding_group_transform (void) { gboolean transform_destroy_called = FALSE; GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); g_binding_group_set_source (group, source); g_binding_group_bind_full (group, "value", target, "value", G_BINDING_BIDIRECTIONAL, celsius_to_fahrenheit, fahrenheit_to_celsius, &transform_destroy_called, transform_destroy_notify); g_object_set (source, "value", 24.0, NULL); g_assert_cmpfloat (target->value, ==, ((9 * 24.0 / 5) + 32.0)); g_object_set (target, "value", 69.0, NULL); g_assert_cmpfloat (source->value, ==, (5 * (69.0 - 32.0) / 9)); /* The GDestroyNotify should only be called when the * set is freed, not when the various GBindings are freed */ g_binding_group_set_source (group, NULL); g_assert_false (transform_destroy_called); g_object_unref (group); g_assert_true (transform_destroy_called); g_object_unref (source); g_object_unref (target); } static void test_binding_group_transform_closures (void) { gboolean transform_destroy_called_1 = FALSE; gboolean transform_destroy_called_2 = FALSE; GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); GClosure *c2f_closure, *f2c_closure; c2f_closure = g_cclosure_new (G_CALLBACK (celsius_to_fahrenheit), &transform_destroy_called_1, (GClosureNotify) transform_destroy_notify); f2c_closure = g_cclosure_new (G_CALLBACK (fahrenheit_to_celsius), &transform_destroy_called_2, (GClosureNotify) transform_destroy_notify); g_binding_group_set_source (group, source); g_binding_group_bind_with_closures (group, "value", target, "value", G_BINDING_BIDIRECTIONAL, c2f_closure, f2c_closure); g_object_set (source, "value", 24.0, NULL); g_assert_cmpfloat (target->value, ==, ((9 * 24.0 / 5) + 32.0)); g_object_set (target, "value", 69.0, NULL); g_assert_cmpfloat (source->value, ==, (5 * (69.0 - 32.0) / 9)); /* The GClsoureNotify should only be called when the * set is freed, not when the various GBindings are freed */ g_binding_group_set_source (group, NULL); g_assert_false (transform_destroy_called_1); g_assert_false (transform_destroy_called_2); g_object_unref (group); g_assert_true (transform_destroy_called_1); g_assert_true (transform_destroy_called_2); g_object_unref (source); g_object_unref (target); } static void test_binding_group_same_object (void) { gsize i; GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), "foo", 100, "bar", 50, NULL); g_binding_group_set_source (group, source); g_binding_group_bind (group, "foo", source, "bar", G_BINDING_BIDIRECTIONAL); for (i = 0; i < 2; ++i) { g_object_set (source, "foo", 10, NULL); g_assert_cmpint (source->foo, ==, 10); g_assert_cmpint (source->bar, ==, 10); g_object_set (source, "bar", 30, NULL); g_assert_cmpint (source->foo, ==, 30); g_assert_cmpint (source->bar, ==, 30); /* Check that it is possible both when initially * adding the binding and when changing the source */ g_binding_group_set_source (group, NULL); g_binding_group_set_source (group, source); } g_object_unref (source); g_object_unref (group); } static void test_binding_group_weak_ref_source (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); BindingSource *readback; g_binding_group_set_source (group, source); g_binding_group_bind (group, "value", target, "value", G_BINDING_BIDIRECTIONAL); g_object_add_weak_pointer (G_OBJECT (source), (gpointer)&source); readback = g_binding_group_dup_source (group); g_assert_true (readback == source); g_object_unref (readback); g_object_unref (source); g_assert_null (source); g_assert_null (g_binding_group_dup_source (group)); /* Hopefully this would explode if the binding was still alive */ g_object_set (target, "value", 0.0, NULL); g_object_unref (target); g_object_unref (group); } static void test_binding_group_weak_ref_target (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); g_binding_group_set_source (group, source); g_binding_group_bind (group, "value", target, "value", G_BINDING_BIDIRECTIONAL); g_object_set (source, "value", 47.0, NULL); g_assert_cmpfloat (target->value, ==, 47.0); g_object_add_weak_pointer (G_OBJECT (target), (gpointer)&target); g_object_unref (target); g_assert_null (target); /* Hopefully this would explode if the binding was still alive */ g_object_set (source, "value", 0.0, NULL); g_object_unref (source); g_object_unref (group); } static void test_binding_group_properties (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); BindingSource *other; g_binding_group_set_source (group, source); g_binding_group_bind (group, "value", target, "value", G_BINDING_BIDIRECTIONAL); g_object_get (group, "source", &other, NULL); g_assert_true (other == source); g_object_unref (other); g_object_set (group, "source", NULL, NULL); g_object_get (group, "source", &other, NULL); g_assert_null (other); g_object_add_weak_pointer (G_OBJECT (target), (gpointer)&target); g_object_unref (target); g_assert_null (target); g_object_unref (source); g_object_unref (group); } static void test_binding_group_weak_notify_no_bindings (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); g_binding_group_set_source (group, source); g_assert_finalize_object (source); g_assert_finalize_object (group); } static void test_binding_group_empty_closures (void) { GBindingGroup *group = g_binding_group_new (); BindingSource *source = g_object_new (binding_source_get_type (), NULL); BindingTarget *target = g_object_new (binding_target_get_type (), NULL); g_binding_group_bind_full (group, "value", target, "value", 0, NULL, NULL, NULL, NULL); g_assert_finalize_object (group); g_assert_finalize_object (target); g_assert_finalize_object (source); } gint main (gint argc, gchar *argv[]) { g_test_init (&argc, &argv, NULL); g_test_add_func ("/GObject/BindingGroup/invalid", test_binding_group_invalid); g_test_add_func ("/GObject/BindingGroup/default", test_binding_group_default); g_test_add_func ("/GObject/BindingGroup/bidirectional", test_binding_group_bidirectional); g_test_add_func ("/GObject/BindingGroup/transform", test_binding_group_transform); g_test_add_func ("/GObject/BindingGroup/transform-closures", test_binding_group_transform_closures); g_test_add_func ("/GObject/BindingGroup/same-object", test_binding_group_same_object); g_test_add_func ("/GObject/BindingGroup/weak-ref-source", test_binding_group_weak_ref_source); g_test_add_func ("/GObject/BindingGroup/weak-ref-target", test_binding_group_weak_ref_target); g_test_add_func ("/GObject/BindingGroup/properties", test_binding_group_properties); g_test_add_func ("/GObject/BindingGroup/weak-notify-no-bindings", test_binding_group_weak_notify_no_bindings); g_test_add_func ("/GObject/BindingGroup/empty-closures", test_binding_group_empty_closures); return g_test_run (); }