Add a test for g_signal_get_invocation_hint
[platform/upstream/glib.git] / gobject / tests / binding.c
index 628ef5e..5f87e84 100644 (file)
@@ -7,7 +7,9 @@ typedef struct _BindingSource
   GObject parent_instance;
 
   gint foo;
+  gint bar;
   gdouble value;
+  gboolean toggle;
 } BindingSource;
 
 typedef struct _BindingSourceClass
@@ -20,10 +22,12 @@ enum
   PROP_SOURCE_0,
 
   PROP_SOURCE_FOO,
-
-  PROP_SOURCE_VALUE
+  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
@@ -40,10 +44,18 @@ binding_source_set_property (GObject      *gobject,
       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);
     }
@@ -63,10 +75,18 @@ binding_source_get_property (GObject    *gobject,
       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);
     }
@@ -85,11 +105,20 @@ binding_source_class_init (BindingSourceClass *klass)
                                                      -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
@@ -103,6 +132,7 @@ typedef struct _BindingTarget
 
   gint bar;
   gdouble value;
+  gboolean toggle;
 } BindingTarget;
 
 typedef struct _BindingTargetClass
@@ -115,10 +145,11 @@ enum
   PROP_TARGET_0,
 
   PROP_TARGET_BAR,
-
-  PROP_TARGET_VALUE
+  PROP_TARGET_VALUE,
+  PROP_TARGET_TOGGLE
 };
 
+static GType binding_target_get_type (void);
 G_DEFINE_TYPE (BindingTarget, binding_target, G_TYPE_OBJECT);
 
 static void
@@ -139,6 +170,10 @@ binding_target_set_property (GObject      *gobject,
       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);
     }
@@ -162,6 +197,10 @@ binding_target_get_property (GObject    *gobject,
       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);
     }
@@ -180,11 +219,15 @@ binding_target_class_init (BindingTargetClass *klass)
                                                      -1, 100,
                                                      0,
                                                      G_PARAM_READWRITE));
-  g_object_class_install_property (gobject_class, PROP_SOURCE_VALUE,
+  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
@@ -194,44 +237,44 @@ binding_target_init (BindingTarget *self)
 
 static gboolean
 celsius_to_fahrenheit (GBinding     *binding,
-                       const GValue *source_value,
-                       GValue       *target_value,
+                       const GValue *from_value,
+                       GValue       *to_value,
                        gpointer      user_data G_GNUC_UNUSED)
 {
   gdouble celsius, fahrenheit;
 
-  g_assert (G_VALUE_HOLDS (source_value, G_TYPE_DOUBLE));
-  g_assert (G_VALUE_HOLDS (target_value, G_TYPE_DOUBLE));
+  g_assert (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
+  g_assert (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
 
-  celsius = g_value_get_double (source_value);
+  celsius = g_value_get_double (from_value);
   fahrenheit = (9 * celsius / 5) + 32.0;
 
   if (g_test_verbose ())
     g_print ("Converting %.2fC to %.2fF\n", celsius, fahrenheit);
 
-  g_value_set_double (target_value, fahrenheit);
+  g_value_set_double (to_value, fahrenheit);
 
   return TRUE;
 }
 
 static gboolean
 fahrenheit_to_celsius (GBinding     *binding,
-                       const GValue *source_value,
-                       GValue       *target_value,
+                       const GValue *from_value,
+                       GValue       *to_value,
                        gpointer      user_data G_GNUC_UNUSED)
 {
   gdouble celsius, fahrenheit;
 
-  g_assert (G_VALUE_HOLDS (source_value, G_TYPE_DOUBLE));
-  g_assert (G_VALUE_HOLDS (target_value, G_TYPE_DOUBLE));
+  g_assert (G_VALUE_HOLDS (from_value, G_TYPE_DOUBLE));
+  g_assert (G_VALUE_HOLDS (to_value, G_TYPE_DOUBLE));
 
-  fahrenheit = g_value_get_double (source_value);
+  fahrenheit = g_value_get_double (from_value);
   celsius = 5 * (fahrenheit - 32.0) / 9;
 
   if (g_test_verbose ())
     g_print ("Converting %.2fF to %.2fC\n", fahrenheit, celsius);
 
-  g_value_set_double (target_value, celsius);
+  g_value_set_double (to_value, celsius);
 
   return TRUE;
 }
@@ -247,8 +290,9 @@ binding_default (void)
                                     target, "bar",
                                     G_BINDING_DEFAULT);
 
-  g_assert (g_binding_get_source (binding) == source);
-  g_assert (g_binding_get_target (binding) == target);
+  g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
+  g_assert ((BindingSource *) g_binding_get_source (binding) == source);
+  g_assert ((BindingTarget *) g_binding_get_target (binding) == target);
   g_assert_cmpstr (g_binding_get_source_property (binding), ==, "foo");
   g_assert_cmpstr (g_binding_get_target_property (binding), ==, "bar");
   g_assert_cmpint (g_binding_get_flags (binding), ==, G_BINDING_DEFAULT);
@@ -266,6 +310,7 @@ binding_default (void)
 
   g_object_unref (source);
   g_object_unref (target);
+  g_assert (binding == NULL);
 }
 
 static void
@@ -278,6 +323,7 @@ binding_bidirectional (void)
   binding = g_object_bind_property (source, "foo",
                                     target, "bar",
                                     G_BINDING_BIDIRECTIONAL);
+  g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
 
   g_object_set (source, "foo", 42, NULL);
   g_assert_cmpint (source->foo, ==, target->bar);
@@ -292,6 +338,7 @@ binding_bidirectional (void)
 
   g_object_unref (source);
   g_object_unref (target);
+  g_assert (binding == NULL);
 }
 
 static void
@@ -303,11 +350,55 @@ data_free (gpointer data)
 }
 
 static void
-binding_transform (void)
+binding_transform_default (void)
 {
   BindingSource *source = g_object_new (binding_source_get_type (), NULL);
   BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
   GBinding *binding;
+  gpointer src, trg;
+  gchar *src_prop, *trg_prop;
+  GBindingFlags flags;
+
+  binding = g_object_bind_property (source, "foo",
+                                    target, "value",
+                                    G_BINDING_BIDIRECTIONAL);
+
+  g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
+
+  g_object_get (binding,
+                "source", &src,
+                "source-property", &src_prop,
+                "target", &trg,
+                "target-property", &trg_prop,
+                "flags", &flags,
+                NULL);
+  g_assert (src == source);
+  g_assert (trg == target);
+  g_assert_cmpstr (src_prop, ==, "foo");
+  g_assert_cmpstr (trg_prop, ==, "value");
+  g_assert_cmpint (flags, ==, G_BINDING_BIDIRECTIONAL);
+  g_object_unref (src);
+  g_object_unref (trg);
+  g_free (src_prop);
+  g_free (trg_prop);
+
+  g_object_set (source, "foo", 24, NULL);
+  g_assert_cmpfloat (target->value, ==, 24.0);
+
+  g_object_set (target, "value", 69.0, NULL);
+  g_assert_cmpint (source->foo, ==, 69);
+
+  g_object_unref (target);
+  g_object_unref (source);
+  g_assert (binding == NULL);
+}
+
+static void
+binding_transform (void)
+{
+  BindingSource *source = g_object_new (binding_source_get_type (), NULL);
+  BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
+  GBinding *binding G_GNUC_UNUSED;
   gboolean unused_data = FALSE;
 
   binding = g_object_bind_property_full (source, "value",
@@ -330,6 +421,38 @@ binding_transform (void)
 }
 
 static void
+binding_transform_closure (void)
+{
+  BindingSource *source = g_object_new (binding_source_get_type (), NULL);
+  BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
+  GBinding *binding G_GNUC_UNUSED;
+  gboolean unused_data_1 = FALSE, unused_data_2 = FALSE;
+  GClosure *c2f_clos, *f2c_clos;
+
+  c2f_clos = g_cclosure_new (G_CALLBACK (celsius_to_fahrenheit), &unused_data_1, (GClosureNotify) data_free);
+
+  f2c_clos = g_cclosure_new (G_CALLBACK (fahrenheit_to_celsius), &unused_data_2, (GClosureNotify) data_free);
+
+  binding = g_object_bind_property_with_closures (source, "value",
+                                                  target, "value",
+                                                  G_BINDING_BIDIRECTIONAL,
+                                                  c2f_clos,
+                                                  f2c_clos);
+
+  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));
+
+  g_object_unref (source);
+  g_object_unref (target);
+
+  g_assert (unused_data_1);
+  g_assert (unused_data_2);
+}
+
+static void
 binding_chain (void)
 {
   BindingSource *a = g_object_new (binding_source_get_type (), NULL);
@@ -337,9 +460,14 @@ binding_chain (void)
   BindingSource *c = g_object_new (binding_source_get_type (), NULL);
   GBinding *binding_1, *binding_2;
 
+  g_test_bug ("621782");
+
   /* A -> B, B -> C */
   binding_1 = g_object_bind_property (a, "foo", b, "foo", G_BINDING_BIDIRECTIONAL);
+  g_object_add_weak_pointer (G_OBJECT (binding_1), (gpointer *) &binding_1);
+
   binding_2 = g_object_bind_property (b, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
+  g_object_add_weak_pointer (G_OBJECT (binding_2), (gpointer *) &binding_2);
 
   /* verify the chain */
   g_object_set (a, "foo", 42, NULL);
@@ -348,7 +476,9 @@ binding_chain (void)
 
   /* unbind A -> B and B -> C */
   g_object_unref (binding_1);
+  g_assert (binding_1 == NULL);
   g_object_unref (binding_2);
+  g_assert (binding_2 == NULL);
 
   /* bind A -> C directly */
   binding_2 = g_object_bind_property (a, "foo", c, "foo", G_BINDING_BIDIRECTIONAL);
@@ -398,17 +528,107 @@ binding_sync_create (void)
   g_object_unref (target);
 }
 
+static void
+binding_invert_boolean (void)
+{
+  BindingSource *source = g_object_new (binding_source_get_type (),
+                                        "toggle", TRUE,
+                                        NULL);
+  BindingTarget *target = g_object_new (binding_target_get_type (),
+                                        "toggle", FALSE,
+                                        NULL);
+  GBinding *binding;
+
+  binding = g_object_bind_property (source, "toggle",
+                                    target, "toggle",
+                                    G_BINDING_BIDIRECTIONAL | G_BINDING_INVERT_BOOLEAN);
+
+  g_assert (source->toggle);
+  g_assert (!target->toggle);
+
+  g_object_set (source, "toggle", FALSE, NULL);
+  g_assert (!source->toggle);
+  g_assert (target->toggle);
+
+  g_object_set (target, "toggle", FALSE, NULL);
+  g_assert (source->toggle);
+  g_assert (!target->toggle);
+
+  g_object_unref (binding);
+  g_object_unref (source);
+  g_object_unref (target);
+}
+
+static void
+binding_same_object (void)
+{
+  BindingSource *source = g_object_new (binding_source_get_type (),
+                                        "foo", 100,
+                                        "bar", 50,
+                                        NULL);
+  GBinding *binding;
+
+  binding = g_object_bind_property (source, "foo",
+                                    source, "bar",
+                                    G_BINDING_BIDIRECTIONAL);
+  g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
+
+  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);
+
+  g_object_unref (source);
+  g_assert (binding == NULL);
+}
+
+static void
+binding_unbind (void)
+{
+  BindingSource *source = g_object_new (binding_source_get_type (), NULL);
+  BindingTarget *target = g_object_new (binding_target_get_type (), NULL);
+  GBinding *binding;
+
+  binding = g_object_bind_property (source, "foo",
+                                    target, "bar",
+                                    G_BINDING_DEFAULT);
+  g_object_add_weak_pointer (G_OBJECT (binding), (gpointer *) &binding);
+
+  g_object_set (source, "foo", 42, NULL);
+  g_assert_cmpint (source->foo, ==, target->bar);
+
+  g_object_set (target, "bar", 47, NULL);
+  g_assert_cmpint (source->foo, !=, target->bar);
+
+  g_binding_unbind (binding);
+  g_assert (binding == NULL);
+
+  g_object_set (source, "foo", 0, NULL);
+  g_assert_cmpint (source->foo, !=, target->bar);
+
+  g_object_unref (source);
+  g_object_unref (target);
+}
+
 int
 main (int argc, char *argv[])
 {
-  g_type_init ();
   g_test_init (&argc, &argv, NULL);
 
+  g_test_bug_base ("http://bugzilla.gnome.org/");
+
   g_test_add_func ("/binding/default", binding_default);
   g_test_add_func ("/binding/bidirectional", binding_bidirectional);
   g_test_add_func ("/binding/transform", binding_transform);
+  g_test_add_func ("/binding/transform-default", binding_transform_default);
+  g_test_add_func ("/binding/transform-closure", binding_transform_closure);
   g_test_add_func ("/binding/chain", binding_chain);
   g_test_add_func ("/binding/sync-create", binding_sync_create);
+  g_test_add_func ("/binding/invert-boolean", binding_invert_boolean);
+  g_test_add_func ("/binding/same-object", binding_same_object);
+  g_test_add_func ("/binding/unbind", binding_unbind);
 
   return g_test_run ();
 }