3 #include <glib-object.h>
5 typedef struct _BindingSource
7 GObject parent_instance;
15 typedef struct _BindingSourceClass
17 GObjectClass parent_class;
26 PROP_SOURCE_DOUBLE_VALUE,
30 static GType binding_source_get_type (void);
31 G_DEFINE_TYPE (BindingSource, binding_source, G_TYPE_OBJECT)
34 binding_source_set_property (GObject *gobject,
39 BindingSource *source = (BindingSource *) gobject;
44 source->foo = g_value_get_int (value);
48 source->bar = g_value_get_int (value);
51 case PROP_SOURCE_DOUBLE_VALUE:
52 source->double_value = g_value_get_double (value);
55 case PROP_SOURCE_TOGGLE:
56 source->toggle = g_value_get_boolean (value);
60 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
65 binding_source_get_property (GObject *gobject,
70 BindingSource *source = (BindingSource *) gobject;
75 g_value_set_int (value, source->foo);
79 g_value_set_int (value, source->bar);
82 case PROP_SOURCE_DOUBLE_VALUE:
83 g_value_set_double (value, source->double_value);
86 case PROP_SOURCE_TOGGLE:
87 g_value_set_boolean (value, source->toggle);
91 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
96 binding_source_class_init (BindingSourceClass *klass)
98 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
100 gobject_class->set_property = binding_source_set_property;
101 gobject_class->get_property = binding_source_get_property;
103 g_object_class_install_property (gobject_class, PROP_SOURCE_FOO,
104 g_param_spec_int ("foo", "Foo", "Foo",
108 g_object_class_install_property (gobject_class, PROP_SOURCE_BAR,
109 g_param_spec_int ("bar", "Bar", "Bar",
113 g_object_class_install_property (gobject_class, PROP_SOURCE_DOUBLE_VALUE,
114 g_param_spec_double ("double-value", "Value", "Value",
118 g_object_class_install_property (gobject_class, PROP_SOURCE_TOGGLE,
119 g_param_spec_boolean ("toggle", "Toggle", "Toggle",
125 binding_source_init (BindingSource *self)
129 typedef struct _BindingTarget
131 GObject parent_instance;
134 gdouble double_value;
138 typedef struct _BindingTargetClass
140 GObjectClass parent_class;
141 } BindingTargetClass;
148 PROP_TARGET_DOUBLE_VALUE,
152 static GType binding_target_get_type (void);
153 G_DEFINE_TYPE (BindingTarget, binding_target, G_TYPE_OBJECT)
156 binding_target_set_property (GObject *gobject,
161 BindingTarget *target = (BindingTarget *) gobject;
165 case PROP_TARGET_BAR:
166 target->bar = g_value_get_int (value);
169 case PROP_TARGET_DOUBLE_VALUE:
170 target->double_value = g_value_get_double (value);
173 case PROP_TARGET_TOGGLE:
174 target->toggle = g_value_get_boolean (value);
178 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
183 binding_target_get_property (GObject *gobject,
188 BindingTarget *target = (BindingTarget *) gobject;
192 case PROP_TARGET_BAR:
193 g_value_set_int (value, target->bar);
196 case PROP_TARGET_DOUBLE_VALUE:
197 g_value_set_double (value, target->double_value);
200 case PROP_TARGET_TOGGLE:
201 g_value_set_boolean (value, target->toggle);
205 G_OBJECT_WARN_INVALID_PROPERTY_ID (gobject, prop_id, pspec);
210 binding_target_class_init (BindingTargetClass *klass)
212 GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
214 gobject_class->set_property = binding_target_set_property;
215 gobject_class->get_property = binding_target_get_property;
217 g_object_class_install_property (gobject_class, PROP_TARGET_BAR,
218 g_param_spec_int ("bar", "Bar", "Bar",
222 g_object_class_install_property (gobject_class, PROP_TARGET_DOUBLE_VALUE,
223 g_param_spec_double ("double-value", "Value", "Value",
227 g_object_class_install_property (gobject_class, PROP_TARGET_TOGGLE,
228 g_param_spec_boolean ("toggle", "Toggle", "Toggle",
234 binding_target_init (BindingTarget *self)
239 celsius_to_fahrenheit (GBinding *binding,
240 const GValue *from_value,
242 gpointer user_data G_GNUC_UNUSED)
244 gdouble celsius, fahrenheit;
246 g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
247 g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
249 celsius = g_value_get_double (from_value);
250 fahrenheit = (9 * celsius / 5) + 32.0;
252 if (g_test_verbose ())
253 g_printerr ("Converting %.2fC to %.2fF\n", celsius, fahrenheit);
255 g_value_set_double (to_value, fahrenheit);
261 fahrenheit_to_celsius (GBinding *binding,
262 const GValue *from_value,
264 gpointer user_data G_GNUC_UNUSED)
266 gdouble celsius, fahrenheit;
268 g_assert_true (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
269 g_assert_true (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
271 fahrenheit = g_value_get_double (from_value);
272 celsius = 5 * (fahrenheit - 32.0) / 9;
274 if (g_test_verbose ())
275 g_printerr ("Converting %.2fF to %.2fC\n", fahrenheit, celsius);
277 g_value_set_double (to_value, celsius);
283 binding_default (void)
285 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
286 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
289 binding = g_object_bind_property (source, "foo",
293 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
294 g_assert_true ((BindingSource *) g_binding_get_source (binding) == source);
295 g_assert_true ((BindingTarget *) g_binding_get_target (binding) == target);
296 g_assert_cmpstr (g_binding_get_source_property (binding), ==, "foo");
297 g_assert_cmpstr (g_binding_get_target_property (binding), ==, "bar");
298 g_assert_cmpint (g_binding_get_flags (binding), ==, G_BINDING_DEFAULT);
300 g_object_set (source, "foo", 42, NULL);
301 g_assert_cmpint (source->foo, ==, target->bar);
303 g_object_set (target, "bar", 47, NULL);
304 g_assert_cmpint (source->foo, !=, target->bar);
306 g_object_unref (binding);
308 g_object_set (source, "foo", 0, NULL);
309 g_assert_cmpint (source->foo, !=, target->bar);
311 g_object_unref (source);
312 g_object_unref (target);
313 g_assert_null (binding);
317 binding_canonicalisation (void)
319 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
320 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
323 g_test_summary ("Test that bindings set up with non-canonical property names work");
325 binding = g_object_bind_property (source, "double_value",
326 target, "double_value",
329 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
330 g_assert_true ((BindingSource *) g_binding_get_source (binding) == source);
331 g_assert_true ((BindingTarget *) g_binding_get_target (binding) == target);
332 g_assert_cmpstr (g_binding_get_source_property (binding), ==, "double-value");
333 g_assert_cmpstr (g_binding_get_target_property (binding), ==, "double-value");
334 g_assert_cmpint (g_binding_get_flags (binding), ==, G_BINDING_DEFAULT);
336 g_object_set (source, "double-value", 24.0, NULL);
337 g_assert_cmpfloat (target->double_value, ==, source->double_value);
339 g_object_set (target, "double-value", 69.0, NULL);
340 g_assert_cmpfloat (source->double_value, !=, target->double_value);
342 g_object_unref (target);
343 g_object_unref (source);
344 g_assert_null (binding);
348 binding_bidirectional (void)
350 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
351 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
354 binding = g_object_bind_property (source, "foo",
356 G_BINDING_BIDIRECTIONAL);
357 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
359 g_object_set (source, "foo", 42, NULL);
360 g_assert_cmpint (source->foo, ==, target->bar);
362 g_object_set (target, "bar", 47, NULL);
363 g_assert_cmpint (source->foo, ==, target->bar);
365 g_object_unref (binding);
367 g_object_set (source, "foo", 0, NULL);
368 g_assert_cmpint (source->foo, !=, target->bar);
370 g_object_unref (source);
371 g_object_unref (target);
372 g_assert_null (binding);
376 data_free (gpointer data)
384 binding_transform_default (void)
386 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
387 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
390 gchar *src_prop, *trg_prop;
393 binding = g_object_bind_property (source, "foo",
394 target, "double-value",
395 G_BINDING_BIDIRECTIONAL);
397 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
399 g_object_get (binding,
401 "source-property", &src_prop,
403 "target-property", &trg_prop,
406 g_assert_true (src == source);
407 g_assert_true (trg == target);
408 g_assert_cmpstr (src_prop, ==, "foo");
409 g_assert_cmpstr (trg_prop, ==, "double-value");
410 g_assert_cmpint (flags, ==, G_BINDING_BIDIRECTIONAL);
411 g_object_unref (src);
412 g_object_unref (trg);
416 g_object_set (source, "foo", 24, NULL);
417 g_assert_cmpfloat (target->double_value, ==, 24.0);
419 g_object_set (target, "double-value", 69.0, NULL);
420 g_assert_cmpint (source->foo, ==, 69);
422 g_object_unref (target);
423 g_object_unref (source);
424 g_assert_null (binding);
428 binding_transform (void)
430 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
431 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
432 GBinding *binding G_GNUC_UNUSED;
433 gboolean unused_data = FALSE;
435 binding = g_object_bind_property_full (source, "double-value",
436 target, "double-value",
437 G_BINDING_BIDIRECTIONAL,
438 celsius_to_fahrenheit,
439 fahrenheit_to_celsius,
440 &unused_data, data_free);
442 g_object_set (source, "double-value", 24.0, NULL);
443 g_assert_cmpfloat (target->double_value, ==, ((9 * 24.0 / 5) + 32.0));
445 g_object_set (target, "double-value", 69.0, NULL);
446 g_assert_cmpfloat (source->double_value, ==, (5 * (69.0 - 32.0) / 9));
448 g_object_unref (source);
449 g_object_unref (target);
451 g_assert_true (unused_data);
455 binding_transform_closure (void)
457 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
458 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
459 GBinding *binding G_GNUC_UNUSED;
460 gboolean unused_data_1 = FALSE, unused_data_2 = FALSE;
461 GClosure *c2f_clos, *f2c_clos;
463 c2f_clos = g_cclosure_new (G_CALLBACK (celsius_to_fahrenheit), &unused_data_1, (GClosureNotify) data_free);
465 f2c_clos = g_cclosure_new (G_CALLBACK (fahrenheit_to_celsius), &unused_data_2, (GClosureNotify) data_free);
467 binding = g_object_bind_property_with_closures (source, "double-value",
468 target, "double-value",
469 G_BINDING_BIDIRECTIONAL,
473 g_object_set (source, "double-value", 24.0, NULL);
474 g_assert_cmpfloat (target->double_value, ==, ((9 * 24.0 / 5) + 32.0));
476 g_object_set (target, "double-value", 69.0, NULL);
477 g_assert_cmpfloat (source->double_value, ==, (5 * (69.0 - 32.0) / 9));
479 g_object_unref (source);
480 g_object_unref (target);
482 g_assert_true (unused_data_1);
483 g_assert_true (unused_data_2);
489 BindingSource *a = g_object_new (binding_source_get_type (), NULL);
490 BindingSource *b = g_object_new (binding_source_get_type (), NULL);
491 BindingSource *c = g_object_new (binding_source_get_type (), NULL);
492 GBinding *binding_1, *binding_2;
494 g_test_bug_base ("http://bugzilla.gnome.org/");
495 g_test_bug ("621782");
498 binding_1 = g_object_bind_property (a, "foo", b, "foo", G_BINDING_BIDIRECTIONAL);
499 g_object_add_weak_pointer (G_OBJECT (binding_1), (gpointer *) &binding_1);
501 binding_2 = g_object_bind_property (b, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
502 g_object_add_weak_pointer (G_OBJECT (binding_2), (gpointer *) &binding_2);
504 /* verify the chain */
505 g_object_set (a, "foo", 42, NULL);
506 g_assert_cmpint (a->foo, ==, b->foo);
507 g_assert_cmpint (b->foo, ==, c->foo);
509 /* unbind A -> B and B -> C */
510 g_object_unref (binding_1);
511 g_assert_null (binding_1);
512 g_object_unref (binding_2);
513 g_assert_null (binding_2);
515 /* bind A -> C directly */
516 binding_2 = g_object_bind_property (a, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
518 /* verify the chain is broken */
519 g_object_set (a, "foo", 47, NULL);
520 g_assert_cmpint (a->foo, !=, b->foo);
521 g_assert_cmpint (a->foo, ==, c->foo);
529 binding_sync_create (void)
531 BindingSource *source = g_object_new (binding_source_get_type (),
534 BindingTarget *target = g_object_new (binding_target_get_type (),
539 binding = g_object_bind_property (source, "foo",
541 G_BINDING_DEFAULT | G_BINDING_SYNC_CREATE);
543 g_assert_cmpint (source->foo, ==, 42);
544 g_assert_cmpint (target->bar, ==, 42);
546 g_object_set (source, "foo", 47, NULL);
547 g_assert_cmpint (source->foo, ==, target->bar);
549 g_object_unref (binding);
551 g_object_set (target, "bar", 49, NULL);
553 binding = g_object_bind_property (source, "foo",
555 G_BINDING_BIDIRECTIONAL | G_BINDING_SYNC_CREATE);
556 g_assert_cmpint (source->foo, ==, 47);
557 g_assert_cmpint (target->bar, ==, 47);
559 g_object_unref (source);
560 g_object_unref (target);
564 binding_invert_boolean (void)
566 BindingSource *source = g_object_new (binding_source_get_type (),
569 BindingTarget *target = g_object_new (binding_target_get_type (),
574 binding = g_object_bind_property (source, "toggle",
576 G_BINDING_BIDIRECTIONAL | G_BINDING_INVERT_BOOLEAN);
578 g_assert_true (source->toggle);
579 g_assert_false (target->toggle);
581 g_object_set (source, "toggle", FALSE, NULL);
582 g_assert_false (source->toggle);
583 g_assert_true (target->toggle);
585 g_object_set (target, "toggle", FALSE, NULL);
586 g_assert_true (source->toggle);
587 g_assert_false (target->toggle);
589 g_object_unref (binding);
590 g_object_unref (source);
591 g_object_unref (target);
595 binding_same_object (void)
597 BindingSource *source = g_object_new (binding_source_get_type (),
603 binding = g_object_bind_property (source, "foo",
605 G_BINDING_BIDIRECTIONAL);
606 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
608 g_object_set (source, "foo", 10, NULL);
609 g_assert_cmpint (source->foo, ==, 10);
610 g_assert_cmpint (source->bar, ==, 10);
611 g_object_set (source, "bar", 30, NULL);
612 g_assert_cmpint (source->foo, ==, 30);
613 g_assert_cmpint (source->bar, ==, 30);
615 g_object_unref (source);
616 g_assert_null (binding);
620 binding_unbind (void)
622 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
623 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
626 binding = g_object_bind_property (source, "foo",
629 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
631 g_object_set (source, "foo", 42, NULL);
632 g_assert_cmpint (source->foo, ==, target->bar);
634 g_object_set (target, "bar", 47, NULL);
635 g_assert_cmpint (source->foo, !=, target->bar);
637 g_binding_unbind (binding);
638 g_assert_null (binding);
640 g_object_set (source, "foo", 0, NULL);
641 g_assert_cmpint (source->foo, !=, target->bar);
643 g_object_unref (source);
644 g_object_unref (target);
647 /* g_binding_unbind() has a special case for this */
648 source = g_object_new (binding_source_get_type (), NULL);
649 binding = g_object_bind_property (source, "foo",
652 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
654 g_binding_unbind (binding);
655 g_assert_null (binding);
657 g_object_unref (source);
660 /* When source or target die, so does the binding if there is no other ref */
662 binding_unbind_weak (void)
665 BindingSource *source;
666 BindingTarget *target;
668 /* first source, then target */
669 source = g_object_new (binding_source_get_type (), NULL);
670 target = g_object_new (binding_target_get_type (), NULL);
671 binding = g_object_bind_property (source, "foo",
674 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
675 g_assert_nonnull (binding);
676 g_object_unref (source);
677 g_assert_null (binding);
678 g_object_unref (target);
679 g_assert_null (binding);
681 /* first target, then source */
682 source = g_object_new (binding_source_get_type (), NULL);
683 target = g_object_new (binding_target_get_type (), NULL);
684 binding = g_object_bind_property (source, "foo",
687 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
688 g_assert_nonnull (binding);
689 g_object_unref (target);
690 g_assert_null (binding);
691 g_object_unref (source);
692 g_assert_null (binding);
694 /* target and source are the same */
695 source = g_object_new (binding_source_get_type (), NULL);
696 binding = g_object_bind_property (source, "foo",
699 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
700 g_assert_nonnull (binding);
701 g_object_unref (source);
702 g_assert_null (binding);
705 /* Test that every call to unbind() after the first is a noop */
707 binding_unbind_multiple (void)
709 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
710 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
716 binding = g_object_bind_property (source, "foo",
719 g_object_ref (binding);
720 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
721 g_assert_nonnull (binding);
723 /* this shouldn't crash */
724 for (i = 0; i < 50; i++)
726 g_binding_unbind (binding);
727 g_assert_nonnull (binding);
730 g_object_unref (binding);
731 g_assert_null (binding);
733 g_object_unref (source);
734 g_object_unref (target);
740 BindingSource *source = g_object_new (binding_source_get_type (), NULL);
741 BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
744 /* double -> boolean is not supported */
745 binding = g_object_bind_property (source, "double-value",
748 g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
750 g_test_expect_message ("GLib-GObject", G_LOG_LEVEL_WARNING,
751 "*Unable to convert*double*boolean*");
752 g_object_set (source, "double-value", 1.0, NULL);
753 g_test_assert_expected_messages ();
755 g_object_unref (source);
756 g_object_unref (target);
757 g_assert_null (binding);
761 main (int argc, char *argv[])
763 g_test_init (&argc, &argv, NULL);
765 g_test_bug_base ("https://gitlab.gnome.org/GNOME/glib/issues/");
767 g_test_add_func ("/binding/default", binding_default);
768 g_test_add_func ("/binding/canonicalisation", binding_canonicalisation);
769 g_test_add_func ("/binding/bidirectional", binding_bidirectional);
770 g_test_add_func ("/binding/transform", binding_transform);
771 g_test_add_func ("/binding/transform-default", binding_transform_default);
772 g_test_add_func ("/binding/transform-closure", binding_transform_closure);
773 g_test_add_func ("/binding/chain", binding_chain);
774 g_test_add_func ("/binding/sync-create", binding_sync_create);
775 g_test_add_func ("/binding/invert-boolean", binding_invert_boolean);
776 g_test_add_func ("/binding/same-object", binding_same_object);
777 g_test_add_func ("/binding/unbind", binding_unbind);
778 g_test_add_func ("/binding/unbind-weak", binding_unbind_weak);
779 g_test_add_func ("/binding/unbind-multiple", binding_unbind_multiple);
780 g_test_add_func ("/binding/fail", binding_fail);
782 return g_test_run ();