mirror of
				https://gitlab.gnome.org/GNOME/glib.git
				synced 2025-10-31 08:22:16 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			3475 lines
		
	
	
		
			105 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			3475 lines
		
	
	
		
			105 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "config.h"
 | ||
| 
 | ||
| #include <stdlib.h>
 | ||
| #include <locale.h>
 | ||
| #include <libintl.h>
 | ||
| #include <unistd.h>
 | ||
| #include <sys/types.h>
 | ||
| #include <gio/gio.h>
 | ||
| #include <gstdio.h>
 | ||
| #define G_SETTINGS_ENABLE_BACKEND
 | ||
| #include <gio/gsettingsbackend.h>
 | ||
| 
 | ||
| #include "testenum.h"
 | ||
| 
 | ||
| #ifdef HAVE_XLOCALE_H
 | ||
| /* Needed on macOS and FreeBSD for uselocale() */
 | ||
| #include <xlocale.h>
 | ||
| #endif
 | ||
| 
 | ||
| static const gchar *locale_dir = ".";
 | ||
| 
 | ||
| static gboolean backend_set;
 | ||
| 
 | ||
| /* These tests rely on the schemas in org.gtk.test.gschema.xml
 | ||
|  * to be compiled and installed in the same directory.
 | ||
|  */
 | ||
| 
 | ||
| typedef struct
 | ||
| {
 | ||
|   gchar *tmp_dir;
 | ||
| } Fixture;
 | ||
| 
 | ||
| static void
 | ||
| setup (Fixture       *fixture,
 | ||
|        gconstpointer  user_data)
 | ||
| {
 | ||
|   GError *error = NULL;
 | ||
| 
 | ||
|   fixture->tmp_dir = g_dir_make_tmp ("gio-test-gsettings_XXXXXX", &error);
 | ||
|   g_assert_no_error (error);
 | ||
| 
 | ||
|   g_test_message ("Using temporary directory: %s", fixture->tmp_dir);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| teardown (Fixture       *fixture,
 | ||
|           gconstpointer  user_data)
 | ||
| {
 | ||
|   g_assert_no_errno (g_rmdir (fixture->tmp_dir));
 | ||
|   g_clear_pointer (&fixture->tmp_dir, g_free);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| check_and_free (GVariant    *value,
 | ||
|                 const gchar *expected)
 | ||
| {
 | ||
|   gchar *printed;
 | ||
| 
 | ||
|   printed = g_variant_print (value, TRUE);
 | ||
|   g_assert_cmpstr (printed, ==, expected);
 | ||
|   g_free (printed);
 | ||
| 
 | ||
|   g_variant_unref (value);
 | ||
| }
 | ||
| 
 | ||
| /* Wrapper around g_assert_cmpstr() which gets a setting from a #GSettings
 | ||
|  * using g_settings_get(). */
 | ||
| #define settings_assert_cmpstr(settings, key, op, expected_value) G_STMT_START { \
 | ||
|   gchar *__str; \
 | ||
|   g_settings_get ((settings), (key), "s", &__str); \
 | ||
|   g_assert_cmpstr (__str, op, (expected_value)); \
 | ||
|   g_free (__str); \
 | ||
| } G_STMT_END
 | ||
| 
 | ||
| 
 | ||
| /* Just to get warmed up: Read and set a string, and
 | ||
|  * verify that can read the changed string back
 | ||
|  */
 | ||
| static void
 | ||
| test_basic (void)
 | ||
| {
 | ||
|   gchar *str = NULL;
 | ||
|   GObject *b;
 | ||
|   gchar *path;
 | ||
|   gboolean has_unapplied;
 | ||
|   gboolean delay_apply;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_object_get (settings,
 | ||
|                 "schema-id", &str,
 | ||
|                 "backend", &b,
 | ||
|                 "path", &path,
 | ||
|                 "has-unapplied", &has_unapplied,
 | ||
|                 "delay-apply", &delay_apply,
 | ||
|                 NULL);
 | ||
|   g_assert_cmpstr (str, ==, "org.gtk.test");
 | ||
|   g_assert_nonnull (b);
 | ||
|   g_assert_cmpstr (path, ==, "/tests/");
 | ||
|   g_assert_false (has_unapplied);
 | ||
|   g_assert_false (delay_apply);
 | ||
|   g_free (str);
 | ||
|   g_object_unref (b);
 | ||
|   g_free (path);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "goodbye world");
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "goodbye world");
 | ||
| 
 | ||
|   if (!backend_set && g_test_undefined ())
 | ||
|     {
 | ||
|       GSettings *tmp_settings = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|       g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
 | ||
|                              "*g_settings_set_value*expects type*");
 | ||
|       g_settings_set (tmp_settings, "greeting", "i", 555);
 | ||
|       g_test_assert_expected_messages ();
 | ||
| 
 | ||
|       g_object_unref (tmp_settings);
 | ||
|     }
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "goodbye world");
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
|   str = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (str, ==, "Hello, earthlings");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "this is the end");
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Check that we get an error when getting a key
 | ||
|  * that is not in the schema
 | ||
|  */
 | ||
| static void
 | ||
| test_unknown_key (void)
 | ||
| {
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_test_subprocess ())
 | ||
|     {
 | ||
|       GSettings *settings;
 | ||
|       GVariant *value;
 | ||
| 
 | ||
|       settings = g_settings_new ("org.gtk.test");
 | ||
|       value = g_settings_get_value (settings, "no_such_key");
 | ||
| 
 | ||
|       g_assert_null (value);
 | ||
| 
 | ||
|       g_object_unref (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   g_test_trap_subprocess (NULL, 0, G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_failed ();
 | ||
|   g_test_trap_assert_stderr ("*does not contain*");
 | ||
| }
 | ||
| 
 | ||
| /* Check that we get an error when the schema
 | ||
|  * has not been installed
 | ||
|  */
 | ||
| static void
 | ||
| test_no_schema (void)
 | ||
| {
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_test_subprocess ())
 | ||
|     {
 | ||
|       GSettings *settings;
 | ||
| 
 | ||
|       settings = g_settings_new ("no.such.schema");
 | ||
| 
 | ||
|       g_assert_null (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   g_test_trap_subprocess (NULL, 0, G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_failed ();
 | ||
|   g_test_trap_assert_stderr ("*Settings schema 'no.such.schema' is not installed*");
 | ||
| }
 | ||
| 
 | ||
| /* Check that we get an error when passing a type string
 | ||
|  * that does not match the schema
 | ||
|  */
 | ||
| static void
 | ||
| test_wrong_type (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   gchar *str = NULL;
 | ||
| 
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_test_expect_message ("GLib", G_LOG_LEVEL_CRITICAL,
 | ||
|                          "*given value has a type of*");
 | ||
|   g_test_expect_message ("GLib", G_LOG_LEVEL_CRITICAL,
 | ||
|                          "*valid_format_string*");
 | ||
|   g_settings_get (settings, "greeting", "o", &str);
 | ||
|   g_test_assert_expected_messages ();
 | ||
| 
 | ||
|   g_assert_null (str);
 | ||
| 
 | ||
|   g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
 | ||
|                          "*expects type 's'*");
 | ||
|   g_settings_set (settings, "greeting", "o", "/a/path");
 | ||
|   g_test_assert_expected_messages ();
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Check errors with explicit paths */
 | ||
| static void
 | ||
| test_wrong_path (void)
 | ||
| {
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_test_subprocess ())
 | ||
|     {
 | ||
|       GSettings *settings G_GNUC_UNUSED;
 | ||
| 
 | ||
|       settings = g_settings_new_with_path ("org.gtk.test", "/wrong-path/");
 | ||
|       return;
 | ||
|     }
 | ||
|   g_test_trap_subprocess (NULL, 0, G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_failed ();
 | ||
|   g_test_trap_assert_stderr ("*but path * specified by schema*");
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_no_path (void)
 | ||
| {
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_test_subprocess ())
 | ||
|     {
 | ||
|       GSettings *settings G_GNUC_UNUSED;
 | ||
| 
 | ||
|       settings = g_settings_new ("org.gtk.test.no-path");
 | ||
|       return;
 | ||
|     }
 | ||
|   g_test_trap_subprocess (NULL, 0, G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_failed ();
 | ||
|   g_test_trap_assert_stderr ("*attempting to create schema * without a path**");
 | ||
| }
 | ||
| 
 | ||
| 
 | ||
| /* Check that we can successfully read and set the full
 | ||
|  * range of all basic types
 | ||
|  */
 | ||
| static void
 | ||
| test_basic_types (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   gboolean b;
 | ||
|   guint8 byte;
 | ||
|   gint16 i16;
 | ||
|   guint16 u16;
 | ||
|   gint32 i32;
 | ||
|   guint32 u32;
 | ||
|   gint64 i64;
 | ||
|   guint64 u64;
 | ||
|   gdouble d;
 | ||
|   gchar *str;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.basic-types");
 | ||
| 
 | ||
|   g_settings_get (settings, "test-boolean", "b", &b);
 | ||
|   g_assert_cmpint (b, ==, 1);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-boolean", "b", 0);
 | ||
|   g_settings_get (settings, "test-boolean", "b", &b);
 | ||
|   g_assert_cmpint (b, ==, 0);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpint (byte, ==, 25);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-byte", "y", G_MAXUINT8);
 | ||
|   g_settings_get (settings, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpint (byte, ==, G_MAXUINT8);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-int16", "n", &i16);
 | ||
|   g_assert_cmpint (i16, ==, -1234);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int16", "n", G_MININT16);
 | ||
|   g_settings_get (settings, "test-int16", "n", &i16);
 | ||
|   g_assert_cmpint (i16, ==, G_MININT16);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int16", "n", G_MAXINT16);
 | ||
|   g_settings_get (settings, "test-int16", "n", &i16);
 | ||
|   g_assert_cmpint (i16, ==, G_MAXINT16);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-uint16", "q", &u16);
 | ||
|   g_assert_cmpuint (u16, ==, 1234);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-uint16", "q", G_MAXUINT16);
 | ||
|   g_settings_get (settings, "test-uint16", "q", &u16);
 | ||
|   g_assert_cmpuint (u16, ==, G_MAXUINT16);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-int32", "i", &i32);
 | ||
|   g_assert_cmpint (i32, ==, -123456);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int32", "i", G_MININT32);
 | ||
|   g_settings_get (settings, "test-int32", "i", &i32);
 | ||
|   g_assert_cmpint (i32, ==, G_MININT32);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int32", "i", G_MAXINT32);
 | ||
|   g_settings_get (settings, "test-int32", "i", &i32);
 | ||
|   g_assert_cmpint (i32, ==, G_MAXINT32);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-uint32", "u", &u32);
 | ||
|   g_assert_cmpuint (u32, ==, 123456);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-uint32", "u", G_MAXUINT32);
 | ||
|   g_settings_get (settings, "test-uint32", "u", &u32);
 | ||
|   g_assert_cmpuint (u32, ==, G_MAXUINT32);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-int64", "x", &i64);
 | ||
|   g_assert_cmpuint (i64, ==, -123456789);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int64", "x", G_MININT64);
 | ||
|   g_settings_get (settings, "test-int64", "x", &i64);
 | ||
|   g_assert_cmpuint (i64, ==, G_MININT64);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-int64", "x", G_MAXINT64);
 | ||
|   g_settings_get (settings, "test-int64", "x", &i64);
 | ||
|   g_assert_cmpuint (i64, ==, G_MAXINT64);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-uint64", "t", &u64);
 | ||
|   g_assert_cmpuint (u64, ==, 123456789);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-uint64", "t", G_MAXUINT64);
 | ||
|   g_settings_get (settings, "test-uint64", "t", &u64);
 | ||
|   g_assert_cmpuint (u64, ==, G_MAXUINT64);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-double", "d", &d);
 | ||
|   g_assert_cmpfloat (d, ==, 123.456);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-double", "d", G_MINDOUBLE);
 | ||
|   g_settings_get (settings, "test-double", "d", &d);
 | ||
|   g_assert_cmpfloat (d, ==, G_MINDOUBLE);
 | ||
| 
 | ||
|   g_settings_set (settings, "test-double", "d", G_MAXDOUBLE);
 | ||
|   g_settings_get (settings, "test-double", "d", &d);
 | ||
|   g_assert_cmpfloat (d, ==, G_MAXDOUBLE);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "test-string", ==, "a string, it seems");
 | ||
| 
 | ||
|   g_settings_get (settings, "test-objectpath", "o", &str);
 | ||
|   g_assert_cmpstr (str, ==, "/a/object/path");
 | ||
|   g_object_unref (settings);
 | ||
|   g_free (str);
 | ||
|   str = NULL;
 | ||
| }
 | ||
| 
 | ||
| /* Check that we can read an set complex types like
 | ||
|  * tuples, arrays and dictionaries
 | ||
|  */
 | ||
| static void
 | ||
| test_complex_types (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   gchar *s;
 | ||
|   gint i1, i2;
 | ||
|   GVariantIter *iter = NULL;
 | ||
|   GVariant *v = NULL;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.complex-types");
 | ||
| 
 | ||
|   g_settings_get (settings, "test-tuple", "(s(ii))", &s, &i1, &i2);
 | ||
|   g_assert_cmpstr (s, ==, "one");
 | ||
|   g_assert_cmpint (i1,==, 2);
 | ||
|   g_assert_cmpint (i2,==, 3);
 | ||
|   g_free (s) ;
 | ||
|   s = NULL;
 | ||
| 
 | ||
|   g_settings_set (settings, "test-tuple", "(s(ii))", "none", 0, 0);
 | ||
|   g_settings_get (settings, "test-tuple", "(s(ii))", &s, &i1, &i2);
 | ||
|   g_assert_cmpstr (s, ==, "none");
 | ||
|   g_assert_cmpint (i1,==, 0);
 | ||
|   g_assert_cmpint (i2,==, 0);
 | ||
|   g_free (s);
 | ||
|   s = NULL;
 | ||
| 
 | ||
|   g_settings_get (settings, "test-array", "ai", &iter);
 | ||
|   g_assert_cmpint (g_variant_iter_n_children (iter), ==, 6);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 0);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 1);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 2);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 3);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 4);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_assert_cmpint (i1, ==, 5);
 | ||
|   g_assert_false (g_variant_iter_next (iter, "i", &i1));
 | ||
|   g_variant_iter_free (iter);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-dict", "a{sau}", &iter);
 | ||
|   g_assert_cmpint (g_variant_iter_n_children (iter), ==, 2);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "{&s@au}", &s, &v));
 | ||
|   g_assert_cmpstr (s, ==, "AC");
 | ||
|   g_assert_cmpstr ((char *)g_variant_get_type (v), ==, "au");
 | ||
|   g_variant_unref (v);
 | ||
|   g_assert_true (g_variant_iter_next (iter, "{&s@au}", &s, &v));
 | ||
|   g_assert_cmpstr (s, ==, "IV");
 | ||
|   g_assert_cmpstr ((char *)g_variant_get_type (v), ==, "au");
 | ||
|   g_variant_unref (v);
 | ||
|   g_variant_iter_free (iter);
 | ||
| 
 | ||
|   v = g_settings_get_value (settings, "test-dict");
 | ||
|   g_assert_cmpstr ((char *)g_variant_get_type (v), ==, "a{sau}");
 | ||
|   g_variant_unref (v);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static gboolean changed_cb_called;
 | ||
| 
 | ||
| static void
 | ||
| changed_cb (GSettings   *settings,
 | ||
|             const gchar *key,
 | ||
|             gpointer     data)
 | ||
| {
 | ||
|   changed_cb_called = TRUE;
 | ||
| 
 | ||
|   g_assert_cmpstr (key, ==, data);
 | ||
| }
 | ||
| 
 | ||
| /* Test that basic change notification with the changed signal works.
 | ||
|  */
 | ||
| static void
 | ||
| test_changes (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettings *settings2;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_signal_connect (settings, "changed",
 | ||
|                     G_CALLBACK (changed_cb), "greeting");
 | ||
| 
 | ||
|   changed_cb_called = FALSE;
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "new greeting");
 | ||
|   g_assert_true (changed_cb_called);
 | ||
| 
 | ||
|   settings2 = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   changed_cb_called = FALSE;
 | ||
| 
 | ||
|   g_settings_set (settings2, "greeting", "s", "hi");
 | ||
|   g_assert_true (changed_cb_called);
 | ||
| 
 | ||
|   g_object_unref (settings2);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static gboolean changed_cb_called2;
 | ||
| 
 | ||
| static void
 | ||
| changed_cb2 (GSettings   *settings,
 | ||
|              const gchar *key,
 | ||
|              gpointer     data)
 | ||
| {
 | ||
|   gboolean *p = data;
 | ||
| 
 | ||
|   *p = TRUE;
 | ||
| }
 | ||
| 
 | ||
| /* Test that changes done to a delay-mode instance
 | ||
|  * don't appear to the outside world until apply. Also
 | ||
|  * check that we get change notification when they are
 | ||
|  * applied.
 | ||
|  * Also test that the has-unapplied property is properly
 | ||
|  * maintained.
 | ||
|  */
 | ||
| static void
 | ||
| test_delay_apply (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettings *settings2;
 | ||
|   gboolean writable;
 | ||
|   GVariant *v;
 | ||
|   const gchar *s;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   settings2 = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_settings_set (settings2, "greeting", "s", "top o' the morning");
 | ||
| 
 | ||
|   changed_cb_called = FALSE;
 | ||
|   changed_cb_called2 = FALSE;
 | ||
| 
 | ||
|   g_signal_connect (settings, "changed",
 | ||
|                     G_CALLBACK (changed_cb2), &changed_cb_called);
 | ||
|   g_signal_connect (settings2, "changed",
 | ||
|                     G_CALLBACK (changed_cb2), &changed_cb_called2);
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "greetings from test_delay_apply");
 | ||
| 
 | ||
|   g_assert_true (changed_cb_called);
 | ||
|   g_assert_false (changed_cb_called2);
 | ||
| 
 | ||
|   /* Try resetting the key and ensure a notification is emitted on the delayed #GSettings object. */
 | ||
|   changed_cb_called = FALSE;
 | ||
|   changed_cb_called2 = FALSE;
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
| 
 | ||
|   g_assert_true (changed_cb_called);
 | ||
|   g_assert_false (changed_cb_called2);
 | ||
| 
 | ||
|   /* Locally change the greeting again. */
 | ||
|   changed_cb_called = FALSE;
 | ||
|   changed_cb_called2 = FALSE;
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "greetings from test_delay_apply");
 | ||
| 
 | ||
|   g_assert_true (changed_cb_called);
 | ||
|   g_assert_false (changed_cb_called2);
 | ||
| 
 | ||
|   writable = g_settings_is_writable (settings, "greeting");
 | ||
|   g_assert_true (writable);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "greetings from test_delay_apply");
 | ||
| 
 | ||
|   v = g_settings_get_user_value (settings, "greeting");
 | ||
|   s = g_variant_get_string (v, NULL);
 | ||
|   g_assert_cmpstr (s, ==, "greetings from test_delay_apply");
 | ||
|   g_variant_unref (v);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings2, "greeting", ==, "top o' the morning");
 | ||
| 
 | ||
|   g_assert_true (g_settings_get_has_unapplied (settings));
 | ||
|   g_assert_false (g_settings_get_has_unapplied (settings2));
 | ||
| 
 | ||
|   changed_cb_called = FALSE;
 | ||
|   changed_cb_called2 = FALSE;
 | ||
| 
 | ||
|   g_settings_apply (settings);
 | ||
| 
 | ||
|   g_assert_false (changed_cb_called);
 | ||
|   g_assert_true (changed_cb_called2);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "greetings from test_delay_apply");
 | ||
|   settings_assert_cmpstr (settings2, "greeting", ==, "greetings from test_delay_apply");
 | ||
| 
 | ||
|   g_assert_false (g_settings_get_has_unapplied (settings));
 | ||
|   g_assert_false (g_settings_get_has_unapplied (settings2));
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
|   g_settings_apply (settings);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   g_object_unref (settings2);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Test that reverting unapplied changes in a delay-apply
 | ||
|  * settings instance works.
 | ||
|  */
 | ||
| static void
 | ||
| test_delay_revert (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettings *settings2;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   settings2 = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_settings_set (settings2, "greeting", "s", "top o' the morning");
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "top o' the morning");
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "greetings from test_delay_revert");
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "greetings from test_delay_revert");
 | ||
|   settings_assert_cmpstr (settings2, "greeting", ==, "top o' the morning");
 | ||
| 
 | ||
|   g_assert_true (g_settings_get_has_unapplied (settings));
 | ||
| 
 | ||
|   g_settings_revert (settings);
 | ||
| 
 | ||
|   g_assert_false (g_settings_get_has_unapplied (settings));
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "top o' the morning");
 | ||
|   settings_assert_cmpstr (settings2, "greeting", ==, "top o' the morning");
 | ||
| 
 | ||
|   g_object_unref (settings2);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_delay_child (void)
 | ||
| {
 | ||
|   GSettings *base;
 | ||
|   GSettings *settings;
 | ||
|   GSettings *child;
 | ||
|   guint8 byte;
 | ||
|   gboolean delay;
 | ||
| 
 | ||
|   base = g_settings_new ("org.gtk.test.basic-types");
 | ||
|   g_settings_set (base, "test-byte", "y", 36);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   g_settings_delay (settings);
 | ||
|   g_object_get (settings, "delay-apply", &delay, NULL);
 | ||
|   g_assert_true (delay);
 | ||
| 
 | ||
|   child = g_settings_get_child (settings, "basic-types");
 | ||
|   g_assert_nonnull (child);
 | ||
| 
 | ||
|   g_object_get (child, "delay-apply", &delay, NULL);
 | ||
|   g_assert_true (delay);
 | ||
| 
 | ||
|   g_settings_get (child, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpuint (byte, ==, 36);
 | ||
| 
 | ||
|   g_settings_set (child, "test-byte", "y", 42);
 | ||
| 
 | ||
|   /* make sure the child was delayed too */
 | ||
|   g_settings_get (base, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpuint (byte, ==, 36);
 | ||
| 
 | ||
|   /* apply the child and the changes should be saved */
 | ||
|   g_settings_apply (child);
 | ||
|   g_settings_get (base, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpuint (byte, ==, 42);
 | ||
| 
 | ||
|   g_object_unref (child);
 | ||
|   g_object_unref (settings);
 | ||
|   g_object_unref (base);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_delay_reset_key (void)
 | ||
| {
 | ||
|   GSettings *direct_settings = NULL, *delayed_settings = NULL;
 | ||
| 
 | ||
|   g_test_summary ("Test that resetting a key on a delayed settings instance works");
 | ||
| 
 | ||
|   delayed_settings = g_settings_new ("org.gtk.test");
 | ||
|   direct_settings = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_settings_set (direct_settings, "greeting", "s", "ey up");
 | ||
| 
 | ||
|   settings_assert_cmpstr (delayed_settings, "greeting", ==, "ey up");
 | ||
| 
 | ||
|   /* Set up a delayed settings backend. */
 | ||
|   g_settings_delay (delayed_settings);
 | ||
| 
 | ||
|   g_settings_set (delayed_settings, "greeting", "s", "how do");
 | ||
| 
 | ||
|   settings_assert_cmpstr (delayed_settings, "greeting", ==, "how do");
 | ||
|   settings_assert_cmpstr (direct_settings, "greeting", ==, "ey up");
 | ||
| 
 | ||
|   g_assert_true (g_settings_get_has_unapplied (delayed_settings));
 | ||
| 
 | ||
|   g_settings_reset (delayed_settings, "greeting");
 | ||
| 
 | ||
|   /* There are still unapplied settings, because the reset is resetting to the
 | ||
|    * value from the schema, not the value from @direct_settings. */
 | ||
|   g_assert_true (g_settings_get_has_unapplied (delayed_settings));
 | ||
| 
 | ||
|   settings_assert_cmpstr (delayed_settings, "greeting", ==, "Hello, earthlings");
 | ||
|   settings_assert_cmpstr (direct_settings, "greeting", ==, "ey up");
 | ||
| 
 | ||
|   /* Apply the settings changes (i.e. the reset). */
 | ||
|   g_settings_apply (delayed_settings);
 | ||
| 
 | ||
|   g_assert_false (g_settings_get_has_unapplied (delayed_settings));
 | ||
| 
 | ||
|   settings_assert_cmpstr (delayed_settings, "greeting", ==, "Hello, earthlings");
 | ||
|   settings_assert_cmpstr (direct_settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   g_object_unref (direct_settings);
 | ||
|   g_object_unref (delayed_settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| keys_changed_cb (GSettings    *settings,
 | ||
|                  const GQuark *keys,
 | ||
|                  gint          n_keys)
 | ||
| {
 | ||
|   g_assert_cmpint (n_keys, ==, 2);
 | ||
| 
 | ||
|   g_assert_true ((keys[0] == g_quark_from_static_string ("greeting") &&
 | ||
|                   keys[1] == g_quark_from_static_string ("farewell")) ||
 | ||
|                  (keys[1] == g_quark_from_static_string ("greeting") &&
 | ||
|                   keys[0] == g_quark_from_static_string ("farewell")));
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "greetings from test_atomic");
 | ||
|   settings_assert_cmpstr (settings, "farewell", ==, "atomic bye-bye");
 | ||
| }
 | ||
| 
 | ||
| /* Check that delay-applied changes appear atomically.
 | ||
|  * More specifically, verify that all changed keys appear
 | ||
|  * with their new value while handling the change-event signal.
 | ||
|  */
 | ||
| static void
 | ||
| test_atomic (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettings *settings2;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   settings2 = g_settings_new ("org.gtk.test");
 | ||
| 
 | ||
|   g_settings_set (settings2, "greeting", "s", "top o' the morning");
 | ||
| 
 | ||
|   changed_cb_called = FALSE;
 | ||
|   changed_cb_called2 = FALSE;
 | ||
| 
 | ||
|   g_signal_connect (settings2, "change-event",
 | ||
|                     G_CALLBACK (keys_changed_cb), NULL);
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "greetings from test_atomic");
 | ||
|   g_settings_set (settings, "farewell", "s", "atomic bye-bye");
 | ||
| 
 | ||
|   g_settings_apply (settings);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "greetings from test_atomic");
 | ||
|   settings_assert_cmpstr (settings, "farewell", ==, "atomic bye-bye");
 | ||
|   settings_assert_cmpstr (settings2, "greeting", ==, "greetings from test_atomic");
 | ||
|   settings_assert_cmpstr (settings2, "farewell", ==, "atomic bye-bye");
 | ||
| 
 | ||
|   g_object_unref (settings2);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* On Windows the interaction between the C library locale and libintl
 | ||
|  * (from GNU gettext) is not like on POSIX, so just skip these tests
 | ||
|  * for now.
 | ||
|  *
 | ||
|  * There are several issues:
 | ||
|  *
 | ||
|  * 1) The C library doesn't use LC_MESSAGES, that is implemented only
 | ||
|  * in libintl (defined in its <libintl.h>).
 | ||
|  *
 | ||
|  * 2) The locale names that uselocale() accepts and returns aren't in
 | ||
|  * the "de_DE" style, but like "German_Germany".
 | ||
|  *
 | ||
|  * 3) libintl looks at the Win32 thread locale and not the C library
 | ||
|  * locale. (And even if libintl would use the C library's locale, as
 | ||
|  * there are several alternative C library DLLs, libintl might be
 | ||
|  * linked to a different one than the application code, so they
 | ||
|  * wouldn't have the same C library locale anyway.)
 | ||
|  */
 | ||
| 
 | ||
| /* Test that translations work for schema defaults.
 | ||
|  *
 | ||
|  * This test relies on the de.po file in the same directory
 | ||
|  * to be compiled into ./de/LC_MESSAGES/test.mo
 | ||
|  */
 | ||
| static void
 | ||
| test_l10n (void)
 | ||
| {
 | ||
| #ifndef HAVE_USELOCALE
 | ||
|   g_test_skip ("Unsafe to change locale because platform does not support uselocale()");
 | ||
| #else
 | ||
|   GSettings *settings;
 | ||
|   gchar *str;
 | ||
|   locale_t original_locale;
 | ||
|   locale_t new_locale;
 | ||
|   locale_t result;
 | ||
| 
 | ||
|   bindtextdomain ("test", locale_dir);
 | ||
|   bind_textdomain_codeset ("test", "UTF-8");
 | ||
| 
 | ||
|   original_locale = uselocale ((locale_t) 0);
 | ||
|   g_assert_true (original_locale != (locale_t) 0);
 | ||
|   new_locale = newlocale (LC_MESSAGES_MASK, "C", (locale_t) 0);
 | ||
|   g_assert_true (new_locale != (locale_t) 0);
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result == original_locale);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.localized");
 | ||
|   str = g_settings_get_string (settings, "error-message");
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
|   freelocale (new_locale);
 | ||
| 
 | ||
|   g_assert_cmpstr (str, ==, "Unnamed");
 | ||
|   g_free (str);
 | ||
|   str = NULL;
 | ||
| 
 | ||
|   new_locale = newlocale (LC_MESSAGES_MASK, "de_DE.UTF-8", (locale_t) 0);
 | ||
|   if (new_locale == (locale_t) 0)
 | ||
|     {
 | ||
|       g_test_skip ("Cannot run test because de_DE.UTF-8 locale is not available");
 | ||
|       g_object_unref (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result == original_locale);
 | ||
| 
 | ||
|   /* Only do the test if translation is actually working... */
 | ||
|   if (g_str_equal (dgettext ("test", "\"Unnamed\""), "\"Unbenannt\""))
 | ||
|     {
 | ||
|       str = g_settings_get_string (settings, "error-message");
 | ||
| 
 | ||
|       g_assert_cmpstr (str, ==, "Unbenannt");
 | ||
|       g_free (str);
 | ||
|       str = NULL;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       g_test_skip ("translation is not working");
 | ||
|     }
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
|   freelocale (new_locale);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| #endif
 | ||
| }
 | ||
| 
 | ||
| /* Test that message context works as expected with translated
 | ||
|  * schema defaults. Also, verify that non-ASCII UTF-8 content
 | ||
|  * works.
 | ||
|  *
 | ||
|  * This test relies on the de.po file in the same directory
 | ||
|  * to be compiled into ./de/LC_MESSAGES/test.mo
 | ||
|  */
 | ||
| static void
 | ||
| test_l10n_context (void)
 | ||
| {
 | ||
| #ifndef HAVE_USELOCALE
 | ||
|   g_test_skip ("Unsafe to change locale because platform does not support uselocale()");
 | ||
| #else
 | ||
|   GSettings *settings;
 | ||
|   gchar *str;
 | ||
|   locale_t original_locale;
 | ||
|   locale_t new_locale;
 | ||
|   locale_t result;
 | ||
| 
 | ||
|   bindtextdomain ("test", locale_dir);
 | ||
|   bind_textdomain_codeset ("test", "UTF-8");
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.localized");
 | ||
| 
 | ||
|   original_locale = uselocale ((locale_t) 0);
 | ||
|   g_assert_true (original_locale != (locale_t) 0);
 | ||
|   new_locale = newlocale (LC_MESSAGES_MASK, "C", (locale_t) 0);
 | ||
|   g_assert_true (new_locale != (locale_t) 0);
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result == original_locale);
 | ||
| 
 | ||
|   g_settings_get (settings, "backspace", "s", &str);
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
|   freelocale (new_locale);
 | ||
| 
 | ||
|   g_assert_cmpstr (str, ==, "BackSpace");
 | ||
|   g_free (str);
 | ||
|   str = NULL;
 | ||
| 
 | ||
|   new_locale = newlocale (LC_MESSAGES_MASK, "de_DE.UTF-8", (locale_t) 0);
 | ||
|   if (new_locale == (locale_t) 0)
 | ||
|     {
 | ||
|       g_test_skip ("Cannot run test because de_DE.UTF-8 locale is not available");
 | ||
|       g_object_unref (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result == original_locale);
 | ||
| 
 | ||
|   /* Only do the test if translation is actually working... */
 | ||
|   if (g_str_equal (dgettext ("test", "\"Unnamed\""), "\"Unbenannt\""))
 | ||
|     settings_assert_cmpstr (settings, "backspace", ==, "Löschen");
 | ||
|   else
 | ||
|     g_test_skip ("translation is not working");
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
|   freelocale (new_locale);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| #endif
 | ||
| }
 | ||
| 
 | ||
| /* Test use of l10n="time" and LC_TIME. */
 | ||
| static void
 | ||
| test_l10n_time (void)
 | ||
| {
 | ||
| #ifndef HAVE_USELOCALE
 | ||
|   g_test_skip ("Unsafe to change locale because platform does not support uselocale()");
 | ||
| #else
 | ||
|   GSettings *settings;
 | ||
|   gchar *str;
 | ||
|   locale_t original_locale;
 | ||
|   locale_t new_locale;
 | ||
|   locale_t result;
 | ||
| 
 | ||
|   g_test_summary ("Test that l10n='time' attribute uses the correct category for translations");
 | ||
|   g_test_bug ("https://gitlab.gnome.org/GNOME/glib/-/issues/2575");
 | ||
| 
 | ||
|   bindtextdomain ("test", locale_dir);
 | ||
|   bind_textdomain_codeset ("test", "UTF-8");
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.localized");
 | ||
| 
 | ||
|   original_locale = uselocale ((locale_t) 0);
 | ||
|   g_assert_true (original_locale != (locale_t) 0);
 | ||
|   new_locale = duplocale (original_locale);
 | ||
|   g_assert_true (new_locale != (locale_t) 0);
 | ||
|   g_clear_pointer (&new_locale, freelocale);
 | ||
| 
 | ||
|   new_locale = newlocale (LC_TIME_MASK, "C", new_locale);
 | ||
|   g_assert_true (new_locale != (locale_t) 0);
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result == original_locale);
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "midnight");
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
| 
 | ||
|   g_assert_cmpstr (str, ==, "12:00 AM");
 | ||
|   g_free (str);
 | ||
|   g_clear_pointer (&new_locale, freelocale);
 | ||
|   str = NULL;
 | ||
| 
 | ||
|   new_locale = newlocale (LC_TIME_MASK, "de_DE.UTF-8", new_locale);
 | ||
|   if (new_locale == (locale_t) 0)
 | ||
|     {
 | ||
|       g_test_skip ("Cannot run test because de_DE.UTF-8 locale is not available");
 | ||
|       g_object_unref (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   result = uselocale (new_locale);
 | ||
|   g_assert_true (result != (locale_t) 0);
 | ||
| 
 | ||
|   /* Only do the test if translation is actually working... */
 | ||
|   if (g_str_equal (dgettext ("test", "\"12:00 AM\""), "\"00:00\""))
 | ||
|     {
 | ||
|       str = g_settings_get_string (settings, "midnight");
 | ||
| 
 | ||
|       g_assert_cmpstr (str, ==, "00:00");
 | ||
|       g_free (str);
 | ||
|       str = NULL;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       g_test_skip ("translation is not working");
 | ||
|     }
 | ||
| 
 | ||
|   result = uselocale (original_locale);
 | ||
|   g_assert_true (result == new_locale);
 | ||
|   g_clear_pointer (&new_locale, freelocale);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| #endif
 | ||
| }
 | ||
| 
 | ||
| enum
 | ||
| {
 | ||
|   PROP_0,
 | ||
|   PROP_BOOL,
 | ||
|   PROP_ANTI_BOOL,
 | ||
|   PROP_BYTE,
 | ||
|   PROP_INT16,
 | ||
|   PROP_UINT16,
 | ||
|   PROP_INT,
 | ||
|   PROP_UINT,
 | ||
|   PROP_INT64,
 | ||
|   PROP_UINT64,
 | ||
|   PROP_DOUBLE,
 | ||
|   PROP_STRING,
 | ||
|   PROP_NO_READ,
 | ||
|   PROP_NO_WRITE,
 | ||
|   PROP_STRV,
 | ||
|   PROP_ENUM,
 | ||
|   PROP_FLAGS
 | ||
| };
 | ||
| 
 | ||
| typedef struct
 | ||
| {
 | ||
|   GObject parent_instance;
 | ||
| 
 | ||
|   gboolean bool_prop;
 | ||
|   gboolean anti_bool_prop;
 | ||
|   gint8 byte_prop;
 | ||
|   gint int16_prop;
 | ||
|   guint16 uint16_prop;
 | ||
|   gint int_prop;
 | ||
|   guint uint_prop;
 | ||
|   gint64 int64_prop;
 | ||
|   guint64 uint64_prop;
 | ||
|   gdouble double_prop;
 | ||
|   gchar *string_prop;
 | ||
|   gchar *no_read_prop;
 | ||
|   gchar *no_write_prop;
 | ||
|   gchar **strv_prop;
 | ||
|   guint enum_prop;
 | ||
|   guint flags_prop;
 | ||
| } TestObject;
 | ||
| 
 | ||
| typedef struct
 | ||
| {
 | ||
|   GObjectClass parent_class;
 | ||
| } TestObjectClass;
 | ||
| 
 | ||
| static GType test_object_get_type (void);
 | ||
| G_DEFINE_TYPE (TestObject, test_object, G_TYPE_OBJECT)
 | ||
| 
 | ||
| static void
 | ||
| test_object_init (TestObject *object)
 | ||
| {
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_object_finalize (GObject *object)
 | ||
| {
 | ||
|   TestObject *testo = (TestObject*)object;
 | ||
|   g_strfreev (testo->strv_prop);
 | ||
|   g_free (testo->string_prop);
 | ||
|   G_OBJECT_CLASS (test_object_parent_class)->finalize (object);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_object_get_property (GObject    *object,
 | ||
|                           guint       prop_id,
 | ||
|                           GValue     *value,
 | ||
|                           GParamSpec *pspec)
 | ||
| {
 | ||
|   TestObject *test_object = (TestObject *)object;
 | ||
| 
 | ||
|   switch (prop_id)
 | ||
|     {
 | ||
|     case PROP_BOOL:
 | ||
|       g_value_set_boolean (value, test_object->bool_prop);
 | ||
|       break;
 | ||
|     case PROP_ANTI_BOOL:
 | ||
|       g_value_set_boolean (value, test_object->anti_bool_prop);
 | ||
|       break;
 | ||
|     case PROP_BYTE:
 | ||
|       g_value_set_schar (value, test_object->byte_prop);
 | ||
|       break;
 | ||
|     case PROP_UINT16:
 | ||
|       g_value_set_uint (value, test_object->uint16_prop);
 | ||
|       break;
 | ||
|     case PROP_INT16:
 | ||
|       g_value_set_int (value, test_object->int16_prop);
 | ||
|       break;
 | ||
|     case PROP_INT:
 | ||
|       g_value_set_int (value, test_object->int_prop);
 | ||
|       break;
 | ||
|     case PROP_UINT:
 | ||
|       g_value_set_uint (value, test_object->uint_prop);
 | ||
|       break;
 | ||
|     case PROP_INT64:
 | ||
|       g_value_set_int64 (value, test_object->int64_prop);
 | ||
|       break;
 | ||
|     case PROP_UINT64:
 | ||
|       g_value_set_uint64 (value, test_object->uint64_prop);
 | ||
|       break;
 | ||
|     case PROP_DOUBLE:
 | ||
|       g_value_set_double (value, test_object->double_prop);
 | ||
|       break;
 | ||
|     case PROP_STRING:
 | ||
|       g_value_set_string (value, test_object->string_prop);
 | ||
|       break;
 | ||
|     case PROP_NO_WRITE:
 | ||
|       g_value_set_string (value, test_object->no_write_prop);
 | ||
|       break;
 | ||
|     case PROP_STRV:
 | ||
|       g_value_set_boxed (value, test_object->strv_prop);
 | ||
|       break;
 | ||
|     case PROP_ENUM:
 | ||
|       g_value_set_enum (value, test_object->enum_prop);
 | ||
|       break;
 | ||
|     case PROP_FLAGS:
 | ||
|       g_value_set_flags (value, test_object->flags_prop);
 | ||
|       break;
 | ||
|     default:
 | ||
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | ||
|       break;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_object_set_property (GObject      *object,
 | ||
|                           guint         prop_id,
 | ||
|                           const GValue *value,
 | ||
|                           GParamSpec   *pspec)
 | ||
| {
 | ||
|   TestObject *test_object = (TestObject *)object;
 | ||
| 
 | ||
|   switch (prop_id)
 | ||
|     {
 | ||
|     case PROP_BOOL:
 | ||
|       test_object->bool_prop = g_value_get_boolean (value);
 | ||
|       break;
 | ||
|     case PROP_ANTI_BOOL:
 | ||
|       test_object->anti_bool_prop = g_value_get_boolean (value);
 | ||
|       break;
 | ||
|     case PROP_BYTE:
 | ||
|       test_object->byte_prop = g_value_get_schar (value);
 | ||
|       break;
 | ||
|     case PROP_INT16:
 | ||
|       test_object->int16_prop = g_value_get_int (value);
 | ||
|       break;
 | ||
|     case PROP_UINT16:
 | ||
|       test_object->uint16_prop = g_value_get_uint (value);
 | ||
|       break;
 | ||
|     case PROP_INT:
 | ||
|       test_object->int_prop = g_value_get_int (value);
 | ||
|       break;
 | ||
|     case PROP_UINT:
 | ||
|       test_object->uint_prop = g_value_get_uint (value);
 | ||
|       break;
 | ||
|     case PROP_INT64:
 | ||
|       test_object->int64_prop = g_value_get_int64 (value);
 | ||
|       break;
 | ||
|     case PROP_UINT64:
 | ||
|       test_object->uint64_prop = g_value_get_uint64 (value);
 | ||
|       break;
 | ||
|     case PROP_DOUBLE:
 | ||
|       test_object->double_prop = g_value_get_double (value);
 | ||
|       break;
 | ||
|     case PROP_STRING:
 | ||
|       g_free (test_object->string_prop);
 | ||
|       test_object->string_prop = g_value_dup_string (value);
 | ||
|       break;
 | ||
|     case PROP_NO_READ:
 | ||
|       g_free (test_object->no_read_prop);
 | ||
|       test_object->no_read_prop = g_value_dup_string (value);
 | ||
|       break;
 | ||
|     case PROP_STRV:
 | ||
|       g_strfreev (test_object->strv_prop);
 | ||
|       test_object->strv_prop = g_value_dup_boxed (value);
 | ||
|       break;
 | ||
|     case PROP_ENUM:
 | ||
|       test_object->enum_prop = g_value_get_enum (value);
 | ||
|       break;
 | ||
|     case PROP_FLAGS:
 | ||
|       test_object->flags_prop = g_value_get_flags (value);
 | ||
|       break;
 | ||
|     default:
 | ||
|       G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
 | ||
|       break;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static GType
 | ||
| test_enum_get_type (void)
 | ||
| {
 | ||
|   static GType define_type_id = 0;
 | ||
| 
 | ||
|   if (g_once_init_enter_pointer (&define_type_id))
 | ||
|     {
 | ||
|       static const GEnumValue values[] = {
 | ||
|         { TEST_ENUM_FOO, "TEST_ENUM_FOO", "foo" },
 | ||
|         { TEST_ENUM_BAR, "TEST_ENUM_BAR", "bar" },
 | ||
|         { TEST_ENUM_BAZ, "TEST_ENUM_BAZ", "baz" },
 | ||
|         { TEST_ENUM_QUUX, "TEST_ENUM_QUUX", "quux" },
 | ||
|         { 0, NULL, NULL }
 | ||
|       };
 | ||
| 
 | ||
|       GType type_id = g_enum_register_static ("TestEnum", values);
 | ||
|       g_once_init_leave_pointer (&define_type_id, type_id);
 | ||
|     }
 | ||
| 
 | ||
|   return define_type_id;
 | ||
| }
 | ||
| 
 | ||
| static GType
 | ||
| test_flags_get_type (void)
 | ||
| {
 | ||
|   static GType define_type_id = 0;
 | ||
| 
 | ||
|   if (g_once_init_enter_pointer (&define_type_id))
 | ||
|     {
 | ||
|       static const GFlagsValue values[] = {
 | ||
|         { TEST_FLAGS_NONE, "TEST_FLAGS_NONE", "none" },
 | ||
|         { TEST_FLAGS_MOURNING, "TEST_FLAGS_MOURNING", "mourning" },
 | ||
|         { TEST_FLAGS_LAUGHING, "TEST_FLAGS_LAUGHING", "laughing" },
 | ||
|         { TEST_FLAGS_WALKING, "TEST_FLAGS_WALKING", "walking" },
 | ||
|         { 0, NULL, NULL }
 | ||
|       };
 | ||
| 
 | ||
|       GType type_id = g_flags_register_static ("TestFlags", values);
 | ||
|       g_once_init_leave_pointer (&define_type_id, type_id);
 | ||
|     }
 | ||
| 
 | ||
|   return define_type_id;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_object_class_init (TestObjectClass *class)
 | ||
| {
 | ||
|   GObjectClass *gobject_class = G_OBJECT_CLASS (class);
 | ||
| 
 | ||
|   gobject_class->get_property = test_object_get_property;
 | ||
|   gobject_class->set_property = test_object_set_property;
 | ||
|   gobject_class->finalize = test_object_finalize;
 | ||
| 
 | ||
|   g_object_class_install_property (gobject_class, PROP_BOOL,
 | ||
|     g_param_spec_boolean ("bool", "", "", FALSE, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_ANTI_BOOL,
 | ||
|     g_param_spec_boolean ("anti-bool", "", "", FALSE, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_BYTE,
 | ||
|     g_param_spec_char ("byte", "", "", G_MININT8, G_MAXINT8, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_INT16,
 | ||
|     g_param_spec_int ("int16", "", "", -G_MAXINT16, G_MAXINT16, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_UINT16,
 | ||
|     g_param_spec_uint ("uint16", "", "", 0, G_MAXUINT16, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_INT,
 | ||
|     g_param_spec_int ("int", "", "", G_MININT, G_MAXINT, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_UINT,
 | ||
|     g_param_spec_uint ("uint", "", "", 0, G_MAXUINT, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_INT64,
 | ||
|     g_param_spec_int64 ("int64", "", "", G_MININT64, G_MAXINT64, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_UINT64,
 | ||
|     g_param_spec_uint64 ("uint64", "", "", 0, G_MAXUINT64, 0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_DOUBLE,
 | ||
|     g_param_spec_double ("double", "", "", -G_MAXDOUBLE, G_MAXDOUBLE, 0.0, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_STRING,
 | ||
|     g_param_spec_string ("string", "", "", NULL, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_NO_WRITE,
 | ||
|     g_param_spec_string ("no-write", "", "", NULL, G_PARAM_READABLE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_NO_READ,
 | ||
|     g_param_spec_string ("no-read", "", "", NULL, G_PARAM_WRITABLE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_STRV,
 | ||
|     g_param_spec_boxed ("strv", "", "", G_TYPE_STRV, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_ENUM,
 | ||
|     g_param_spec_enum ("enum", "", "", test_enum_get_type (), TEST_ENUM_FOO, G_PARAM_READWRITE));
 | ||
|   g_object_class_install_property (gobject_class, PROP_FLAGS,
 | ||
|     g_param_spec_flags ("flags", "", "", test_flags_get_type (), TEST_FLAGS_NONE, G_PARAM_READWRITE));
 | ||
| }
 | ||
| 
 | ||
| static TestObject *
 | ||
| test_object_new (void)
 | ||
| {
 | ||
|   return (TestObject*)g_object_new (test_object_get_type (), NULL);
 | ||
| }
 | ||
| 
 | ||
| /* Test basic binding functionality for simple types.
 | ||
|  * Verify that with bidirectional bindings, changes on either side
 | ||
|  * are notified on the other end.
 | ||
|  */
 | ||
| static void
 | ||
| test_simple_binding (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   gboolean b;
 | ||
|   gchar y;
 | ||
|   gint i;
 | ||
|   guint u;
 | ||
|   gint16 n;
 | ||
|   guint16 q;
 | ||
|   gint n2;
 | ||
|   guint q2;
 | ||
|   gint64 i64;
 | ||
|   guint64 u64;
 | ||
|   gdouble d;
 | ||
|   gchar *s;
 | ||
|   GVariant *value;
 | ||
|   gchar **strv;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "bool", obj, "bool", G_SETTINGS_BIND_DEFAULT);
 | ||
|   g_object_set (obj, "bool", TRUE, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_boolean (settings, "bool"), ==, TRUE);
 | ||
| 
 | ||
|   g_settings_set_boolean (settings, "bool", FALSE);
 | ||
|   b = TRUE;
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_settings_bind (settings, "anti-bool", obj, "anti-bool",
 | ||
|                    G_SETTINGS_BIND_INVERT_BOOLEAN);
 | ||
|   g_object_set (obj, "anti-bool", FALSE, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_boolean (settings, "anti-bool"), ==, TRUE);
 | ||
| 
 | ||
|   g_settings_set_boolean (settings, "anti-bool", FALSE);
 | ||
|   b = FALSE;
 | ||
|   g_object_get (obj, "anti-bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, TRUE);
 | ||
| 
 | ||
|   g_settings_bind (settings, "byte", obj, "byte", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "byte", 123, NULL);
 | ||
|   y = 'c';
 | ||
|   g_settings_get (settings, "byte", "y", &y);
 | ||
|   g_assert_cmpint (y, ==, 123);
 | ||
| 
 | ||
|   g_settings_set (settings, "byte", "y", 54);
 | ||
|   y = 'c';
 | ||
|   g_object_get (obj, "byte", &y, NULL);
 | ||
|   g_assert_cmpint (y, ==, 54);
 | ||
| 
 | ||
|   g_settings_bind (settings, "int16", obj, "int16", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "int16", 1234, NULL);
 | ||
|   n = 4321;
 | ||
|   g_settings_get (settings, "int16", "n", &n);
 | ||
|   g_assert_cmpint (n, ==, 1234);
 | ||
| 
 | ||
|   g_settings_set (settings, "int16", "n", 4321);
 | ||
|   n2 = 1111;
 | ||
|   g_object_get (obj, "int16", &n2, NULL);
 | ||
|   g_assert_cmpint (n2, ==, 4321);
 | ||
| 
 | ||
|   g_settings_bind (settings, "uint16", obj, "uint16", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "uint16", (guint16) G_MAXUINT16, NULL);
 | ||
|   q = 1111;
 | ||
|   g_settings_get (settings, "uint16", "q", &q);
 | ||
|   g_assert_cmpuint (q, ==, G_MAXUINT16);
 | ||
| 
 | ||
|   g_settings_set (settings, "uint16", "q", (guint16) G_MAXINT16);
 | ||
|   q2 = 1111;
 | ||
|   g_object_get (obj, "uint16", &q2, NULL);
 | ||
|   g_assert_cmpuint (q2, ==, (guint16) G_MAXINT16);
 | ||
| 
 | ||
|   g_settings_bind (settings, "int", obj, "int", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "int", 12345, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "int"), ==, 12345);
 | ||
| 
 | ||
|   g_settings_set_int (settings, "int", 54321);
 | ||
|   i = 1111;
 | ||
|   g_object_get (obj, "int", &i, NULL);
 | ||
|   g_assert_cmpint (i, ==, 54321);
 | ||
| 
 | ||
|   g_settings_bind (settings, "uint", obj, "uint", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "uint", 12345, NULL);
 | ||
|   g_assert_cmpuint (g_settings_get_uint (settings, "uint"), ==, 12345);
 | ||
| 
 | ||
|   g_settings_set_uint (settings, "uint", 54321);
 | ||
|   u = 1111;
 | ||
|   g_object_get (obj, "uint", &u, NULL);
 | ||
|   g_assert_cmpuint (u, ==, 54321);
 | ||
| 
 | ||
|   g_settings_bind (settings, "uint64", obj, "uint64", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "uint64", (guint64) 12345, NULL);
 | ||
|   g_assert_cmpuint (g_settings_get_uint64 (settings, "uint64"), ==, 12345);
 | ||
| 
 | ||
|   g_settings_set_uint64 (settings, "uint64", 54321);
 | ||
|   u64 = 1111;
 | ||
|   g_object_get (obj, "uint64", &u64, NULL);
 | ||
|   g_assert_cmpuint (u64, ==, 54321);
 | ||
| 
 | ||
|   g_settings_bind (settings, "int64", obj, "int64", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "int64", (gint64) G_MAXINT64, NULL);
 | ||
|   i64 = 1111;
 | ||
|   g_settings_get (settings, "int64", "x", &i64);
 | ||
|   g_assert_cmpint (i64, ==, G_MAXINT64);
 | ||
| 
 | ||
|   g_settings_set (settings, "int64", "x", (gint64) G_MININT64);
 | ||
|   i64 = 1111;
 | ||
|   g_object_get (obj, "int64", &i64, NULL);
 | ||
|   g_assert_cmpint (i64, ==, G_MININT64);
 | ||
| 
 | ||
|   g_settings_bind (settings, "uint64", obj, "uint64", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "uint64", (guint64) G_MAXUINT64, NULL);
 | ||
|   u64 = 1111;
 | ||
|   g_settings_get (settings, "uint64", "t", &u64);
 | ||
|   g_assert_cmpuint (u64, ==, G_MAXUINT64);
 | ||
| 
 | ||
|   g_settings_set (settings, "uint64", "t", (guint64) G_MAXINT64);
 | ||
|   u64 = 1111;
 | ||
|   g_object_get (obj, "uint64", &u64, NULL);
 | ||
|   g_assert_cmpuint (u64, ==, (guint64) G_MAXINT64);
 | ||
| 
 | ||
|   g_settings_bind (settings, "string", obj, "string", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "string", "bu ba", NULL);
 | ||
|   s = g_settings_get_string (settings, "string");
 | ||
|   g_assert_cmpstr (s, ==, "bu ba");
 | ||
|   g_free (s);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "bla bla");
 | ||
|   g_object_get (obj, "string", &s, NULL);
 | ||
|   g_assert_cmpstr (s, ==, "bla bla");
 | ||
|   g_free (s);
 | ||
| 
 | ||
|   g_settings_bind (settings, "chararray", obj, "string", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "string", "non-unicode:\315", NULL);
 | ||
|   value = g_settings_get_value (settings, "chararray");
 | ||
|   g_assert_cmpstr (g_variant_get_bytestring (value), ==, "non-unicode:\315");
 | ||
|   g_variant_unref (value);
 | ||
| 
 | ||
|   g_settings_bind (settings, "double", obj, "double", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "double", G_MAXFLOAT, NULL);
 | ||
|   g_assert_cmpfloat (g_settings_get_double (settings, "double"), ==, G_MAXFLOAT);
 | ||
| 
 | ||
|   g_settings_set_double (settings, "double", G_MINFLOAT);
 | ||
|   d = 1.0;
 | ||
|   g_object_get (obj, "double", &d, NULL);
 | ||
|   g_assert_cmpfloat (d, ==, G_MINFLOAT);
 | ||
| 
 | ||
|   g_object_set (obj, "double", G_MAXDOUBLE, NULL);
 | ||
|   g_assert_cmpfloat (g_settings_get_double (settings, "double"), ==, G_MAXDOUBLE);
 | ||
| 
 | ||
|   g_settings_set_double (settings, "double", -G_MINDOUBLE);
 | ||
|   d = 1.0;
 | ||
|   g_object_get (obj, "double", &d, NULL);
 | ||
|   g_assert_cmpfloat (d, ==, -G_MINDOUBLE);
 | ||
| 
 | ||
|   strv = g_strsplit ("plastic bag,middle class,polyethylene", ",", 0);
 | ||
|   g_settings_bind (settings, "strv", obj, "strv", G_SETTINGS_BIND_DEFAULT);
 | ||
|   g_object_set (obj, "strv", strv, NULL);
 | ||
|   g_strfreev (strv);
 | ||
|   strv = g_settings_get_strv (settings, "strv");
 | ||
|   s = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (s, ==, "plastic bag,middle class,polyethylene");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (s);
 | ||
|   strv = g_strsplit ("decaffeinate,unleaded,keep all surfaces clean", ",", 0);
 | ||
|   g_settings_set_strv (settings, "strv", (const gchar **) strv);
 | ||
|   g_strfreev (strv);
 | ||
|   g_object_get (obj, "strv", &strv, NULL);
 | ||
|   s = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (s, ==, "decaffeinate,unleaded,keep all surfaces clean");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (s);
 | ||
|   g_settings_set_strv (settings, "strv", NULL);
 | ||
|   g_object_get (obj, "strv", &strv, NULL);
 | ||
|   g_assert_nonnull (strv);
 | ||
|   g_assert_cmpint (g_strv_length (strv), ==, 0);
 | ||
|   g_strfreev (strv);
 | ||
| 
 | ||
|   g_settings_bind (settings, "enum", obj, "enum", G_SETTINGS_BIND_DEFAULT);
 | ||
|   g_object_set (obj, "enum", TEST_ENUM_BAZ, NULL);
 | ||
|   s = g_settings_get_string (settings, "enum");
 | ||
|   g_assert_cmpstr (s, ==, "baz");
 | ||
|   g_free (s);
 | ||
|   g_assert_cmpint (g_settings_get_enum (settings, "enum"), ==, TEST_ENUM_BAZ);
 | ||
| 
 | ||
|   g_settings_set_enum (settings, "enum", TEST_ENUM_QUUX);
 | ||
|   i = 230;
 | ||
|   g_object_get (obj, "enum", &i, NULL);
 | ||
|   g_assert_cmpint (i, ==, TEST_ENUM_QUUX);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "enum", "baz");
 | ||
|   i = 230;
 | ||
|   g_object_get (obj, "enum", &i, NULL);
 | ||
|   g_assert_cmpint (i, ==, TEST_ENUM_BAZ);
 | ||
| 
 | ||
|   g_settings_bind (settings, "flags", obj, "flags", G_SETTINGS_BIND_DEFAULT);
 | ||
|   g_object_set (obj, "flags", TEST_FLAGS_MOURNING, NULL);
 | ||
|   strv = g_settings_get_strv (settings, "flags");
 | ||
|   g_assert_cmpint (g_strv_length (strv), ==, 1);
 | ||
|   g_assert_cmpstr (strv[0], ==, "mourning");
 | ||
|   g_strfreev (strv);
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_flags (settings, "flags"), ==, TEST_FLAGS_MOURNING);
 | ||
| 
 | ||
|   g_settings_set_flags (settings, "flags", TEST_FLAGS_MOURNING | TEST_FLAGS_WALKING);
 | ||
|   i = 230;
 | ||
|   g_object_get (obj, "flags", &i, NULL);
 | ||
|   g_assert_cmpint (i, ==, TEST_FLAGS_MOURNING | TEST_FLAGS_WALKING);
 | ||
| 
 | ||
|   g_settings_bind (settings, "uint", obj, "uint", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "uint", 12345, NULL);
 | ||
|   g_assert_cmpuint (g_settings_get_uint (settings, "uint"), ==, 12345);
 | ||
| 
 | ||
|   g_settings_set_uint (settings, "uint", 54321);
 | ||
|   u = 1111;
 | ||
|   g_object_get (obj, "uint", &u, NULL);
 | ||
|   g_assert_cmpuint (u, ==, 54321);
 | ||
| 
 | ||
|   g_settings_bind (settings, "range", obj, "uint", G_SETTINGS_BIND_DEFAULT);
 | ||
|   g_object_set (obj, "uint", 22, NULL);
 | ||
|   u = 1111;
 | ||
|   g_assert_cmpuint (g_settings_get_uint (settings, "range"), ==, 22);
 | ||
|   g_object_get (obj, "uint", &u, NULL);
 | ||
|   g_assert_cmpuint (u, ==, 22);
 | ||
| 
 | ||
|   g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
 | ||
|                          "* is out of schema-specified range for*");
 | ||
|   g_object_set (obj, "uint", 45, NULL);
 | ||
|   g_test_assert_expected_messages ();
 | ||
|   u = 1111;
 | ||
|   g_object_get (obj, "uint", &u, NULL);
 | ||
|   g_assert_cmpuint (g_settings_get_uint (settings, "range"), ==, 22);
 | ||
|   /* The value of the object is currently not reset back to its initial value
 | ||
|   g_assert_cmpuint (u, ==, 22); */
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_unbind (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "int", obj, "int", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_set (obj, "int", 12345, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "int"), ==, 12345);
 | ||
| 
 | ||
|   g_settings_unbind (obj, "int");
 | ||
| 
 | ||
|   g_object_set (obj, "int", 54321, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "int"), ==, 12345);
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_bind_writable (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   gboolean b;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
| 
 | ||
|   g_settings_bind_writable (settings, "int", obj, "bool", FALSE);
 | ||
| 
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_true (b);
 | ||
| 
 | ||
|   g_settings_unbind (obj, "bool");
 | ||
| 
 | ||
|   g_settings_bind_writable (settings, "int", obj, "bool", TRUE);
 | ||
| 
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_false (b);
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Test one-way bindings.
 | ||
|  * Verify that changes on one side show up on the other,
 | ||
|  * but not vice versa
 | ||
|  */
 | ||
| static void
 | ||
| test_directional_binding (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   gboolean b;
 | ||
|   gint i;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
|   g_settings_set_boolean (settings, "bool", FALSE);
 | ||
| 
 | ||
|   g_settings_bind (settings, "bool", obj, "bool", G_SETTINGS_BIND_GET);
 | ||
| 
 | ||
|   g_settings_set_boolean (settings, "bool", TRUE);
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, TRUE);
 | ||
| 
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_boolean (settings, "bool"), ==, TRUE);
 | ||
| 
 | ||
|   g_object_set (obj, "int", 20, NULL);
 | ||
|   g_settings_set_int (settings, "int", 20);
 | ||
| 
 | ||
|   g_settings_bind (settings, "int", obj, "int", G_SETTINGS_BIND_SET);
 | ||
| 
 | ||
|   g_object_set (obj, "int", 32, NULL);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "int"), ==, 32);
 | ||
| 
 | ||
|   g_settings_set_int (settings, "int", 20);
 | ||
|   g_object_get (obj, "int", &i, NULL);
 | ||
|   g_assert_cmpint (i, ==, 32);
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Test that type mismatch is caught when creating a binding */
 | ||
| static void
 | ||
| test_typesafe_binding (void)
 | ||
| {
 | ||
|   if (!g_test_undefined ())
 | ||
|     return;
 | ||
| 
 | ||
|   if (g_test_subprocess ())
 | ||
|     {
 | ||
|       TestObject *obj;
 | ||
|       GSettings *settings;
 | ||
| 
 | ||
|       settings = g_settings_new ("org.gtk.test.binding");
 | ||
|       obj = test_object_new ();
 | ||
| 
 | ||
|       g_settings_bind (settings, "string", obj, "int", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|       g_object_unref (obj);
 | ||
|       g_object_unref (settings);
 | ||
|       return;
 | ||
|     }
 | ||
|   g_test_trap_subprocess (NULL, 0, G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_failed ();
 | ||
|   g_test_trap_assert_stderr ("*not compatible*");
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| string_to_bool (GValue   *value,
 | ||
|                 GVariant *variant,
 | ||
|                 gpointer  user_data)
 | ||
| {
 | ||
|   const gchar *s;
 | ||
| 
 | ||
|   s = g_variant_get_string (variant, NULL);
 | ||
|   g_value_set_boolean (value, g_strcmp0 (s, "true") == 0);
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| static GVariant *
 | ||
| bool_to_string (const GValue       *value,
 | ||
|                 const GVariantType *expected_type,
 | ||
|                 gpointer            user_data)
 | ||
| {
 | ||
|   if (g_value_get_boolean (value))
 | ||
|     return g_variant_new_string ("true");
 | ||
|   else
 | ||
|     return g_variant_new_string ("false");
 | ||
| }
 | ||
| 
 | ||
| static GVariant *
 | ||
| bool_to_bool (const GValue       *value,
 | ||
|               const GVariantType *expected_type,
 | ||
|               gpointer            user_data)
 | ||
| {
 | ||
|   return g_variant_new_boolean (g_value_get_boolean (value));
 | ||
| }
 | ||
| 
 | ||
| /* Test custom bindings.
 | ||
|  * Translate strings to booleans and back
 | ||
|  */
 | ||
| static void
 | ||
| test_custom_binding (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   gchar *s;
 | ||
|   gboolean b;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "true");
 | ||
| 
 | ||
|   g_settings_bind_with_mapping (settings, "string",
 | ||
|                                 obj, "bool",
 | ||
|                                 G_SETTINGS_BIND_DEFAULT,
 | ||
|                                 string_to_bool,
 | ||
|                                 bool_to_string,
 | ||
|                                 NULL, NULL);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "false");
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "not true");
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_object_set (obj, "bool", TRUE, NULL);
 | ||
|   s = g_settings_get_string (settings, "string");
 | ||
|   g_assert_cmpstr (s, ==, "true");
 | ||
|   g_free (s);
 | ||
| 
 | ||
|   g_settings_bind_with_mapping (settings, "string",
 | ||
|                                 obj, "bool",
 | ||
|                                 G_SETTINGS_BIND_DEFAULT,
 | ||
|                                 string_to_bool, bool_to_bool,
 | ||
|                                 NULL, NULL);
 | ||
|   g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
 | ||
|                          "*binding mapping function for key 'string' returned"
 | ||
|                          " GVariant of type 'b' when type 's' was requested*");
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
|   g_test_assert_expected_messages ();
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Same test as above, but with closures
 | ||
|  */
 | ||
| static void
 | ||
| test_bind_with_mapping_closures (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   char *s;
 | ||
|   gboolean b;
 | ||
|   GClosure *get;
 | ||
|   GClosure *set;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "true");
 | ||
| 
 | ||
|   get = g_cclosure_new (G_CALLBACK (string_to_bool), NULL, NULL);
 | ||
|   set = g_cclosure_new (G_CALLBACK (bool_to_string), NULL, NULL);
 | ||
| 
 | ||
|   g_settings_bind_with_mapping_closures (settings, "string",
 | ||
|                                          G_OBJECT (obj), "bool",
 | ||
|                                          G_SETTINGS_BIND_DEFAULT, get, set);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "false");
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "not true");
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_object_set (obj, "bool", TRUE, NULL);
 | ||
|   s = g_settings_get_string (settings, "string");
 | ||
|   g_assert_cmpstr (s, ==, "true");
 | ||
|   g_free (s);
 | ||
| 
 | ||
|   set = g_cclosure_new (G_CALLBACK (bool_to_bool), NULL, NULL);
 | ||
| 
 | ||
|   g_settings_bind_with_mapping_closures (settings, "string",
 | ||
|                                          G_OBJECT (obj), "bool",
 | ||
|                                          G_SETTINGS_BIND_DEFAULT, get, set);
 | ||
|   g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_CRITICAL,
 | ||
|                          "*binding mapping function for key 'string' returned"
 | ||
|                          " GVariant of type 'b' when type 's' was requested*");
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
|   g_test_assert_expected_messages ();
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| typedef struct
 | ||
| {
 | ||
|   gboolean get_called;
 | ||
|   gboolean set_called;
 | ||
|   gboolean get_freed;
 | ||
|   gboolean set_freed;
 | ||
| } BindWithMappingData;
 | ||
| 
 | ||
| static gboolean
 | ||
| get_callback (GValue *value,
 | ||
|               GVariant *variant,
 | ||
|               void *user_data)
 | ||
| {
 | ||
|   BindWithMappingData *data = (BindWithMappingData *) user_data;
 | ||
|   data->get_called = TRUE;
 | ||
| 
 | ||
|   g_assert_true (G_VALUE_HOLDS_BOOLEAN (value));
 | ||
|   g_assert_true (g_variant_is_of_type (variant, G_VARIANT_TYPE_STRING));
 | ||
| 
 | ||
|   return string_to_bool (value, variant, NULL);
 | ||
| }
 | ||
| 
 | ||
| static GVariant *
 | ||
| set_callback (const GValue *value,
 | ||
|               const GVariantType *expected_type,
 | ||
|               void *user_data)
 | ||
| {
 | ||
|   BindWithMappingData *data = (BindWithMappingData *) user_data;
 | ||
|   data->set_called = TRUE;
 | ||
| 
 | ||
|   g_assert_true (G_VALUE_HOLDS_BOOLEAN (value));
 | ||
|   g_assert_true (g_variant_type_equal (expected_type, G_VARIANT_TYPE_STRING));
 | ||
| 
 | ||
|   return bool_to_string (value, expected_type, NULL);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| teardown_get (void *user_data, GClosure *closure)
 | ||
| {
 | ||
|   BindWithMappingData *data = (BindWithMappingData *) user_data;
 | ||
|   data->get_freed = TRUE;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| teardown_set (void *user_data, GClosure *closure)
 | ||
| {
 | ||
|   BindWithMappingData *data = (BindWithMappingData *) user_data;
 | ||
|   data->set_freed = TRUE;
 | ||
| }
 | ||
| 
 | ||
| /* Tests the types of GValue and GVariant passed to the closures */
 | ||
| static void
 | ||
| test_bind_with_mapping_closures_parameters (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   GClosure *get;
 | ||
|   GClosure *set;
 | ||
|   gboolean val;
 | ||
|   BindWithMappingData data = { FALSE, FALSE, FALSE, FALSE };
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_set_string (settings, "string", "true");
 | ||
| 
 | ||
|   get = g_cclosure_new (G_CALLBACK (get_callback), &data, teardown_get);
 | ||
|   set = g_cclosure_new (G_CALLBACK (set_callback), &data, teardown_set);
 | ||
| 
 | ||
|   g_settings_bind_with_mapping_closures (settings, "string",
 | ||
|                                          G_OBJECT (obj), "bool",
 | ||
|                                          G_SETTINGS_BIND_DEFAULT, get, set);
 | ||
| 
 | ||
|   g_assert_true (data.get_called);
 | ||
|   g_assert_false (data.set_called);
 | ||
|   g_object_get (obj, "bool", &val, NULL);
 | ||
|   g_assert_true (val);
 | ||
| 
 | ||
|   data.get_called = FALSE;
 | ||
|   g_object_set (obj, "bool", FALSE, NULL);
 | ||
|   g_assert_true (data.set_called);
 | ||
|   g_assert_false (data.get_called);
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
| 
 | ||
|   g_assert_true (data.get_freed);
 | ||
|   g_assert_true (data.set_freed);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Test that with G_SETTINGS_BIND_NO_CHANGES, the
 | ||
|  * initial settings value is transported to the object
 | ||
|  * side, but later settings changes do not affect the
 | ||
|  * object
 | ||
|  */
 | ||
| static void
 | ||
| test_no_change_binding (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
|   gboolean b;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_object_set (obj, "bool", TRUE, NULL);
 | ||
|   g_settings_set_boolean (settings, "bool", FALSE);
 | ||
| 
 | ||
|   g_settings_bind (settings, "bool", obj, "bool", G_SETTINGS_BIND_GET_NO_CHANGES);
 | ||
| 
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_settings_set_boolean (settings, "bool", TRUE);
 | ||
|   g_object_get (obj, "bool", &b, NULL);
 | ||
|   g_assert_cmpint (b, ==, FALSE);
 | ||
| 
 | ||
|   g_settings_set_boolean (settings, "bool", FALSE);
 | ||
|   g_object_set (obj, "bool", TRUE, NULL);
 | ||
|   b = g_settings_get_boolean (settings, "bool");
 | ||
|   g_assert_cmpint (b, ==, TRUE);
 | ||
| 
 | ||
|   g_object_unref (obj);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| /* Test that binding a non-readable property only
 | ||
|  * works in 'GET' mode.
 | ||
|  */
 | ||
| static void
 | ||
| test_no_read_binding_fail (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "string", obj, "no-read", 0);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_no_read_binding_pass (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "string", obj, "no-read", G_SETTINGS_BIND_GET);
 | ||
| 
 | ||
|   exit (0);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_no_read_binding (void)
 | ||
| {
 | ||
|   if (g_test_undefined ())
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/no-read-binding/subprocess/fail", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*property*is not readable*");
 | ||
|     }
 | ||
| 
 | ||
|   g_test_trap_subprocess ("/gsettings/no-read-binding/subprocess/pass", 0,
 | ||
|                           G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_passed ();
 | ||
| }
 | ||
| 
 | ||
| /* Test that binding a non-writable property only
 | ||
|  * works in 'SET' mode.
 | ||
|  */
 | ||
| static void
 | ||
| test_no_write_binding_fail (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "string", obj, "no-write", 0);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_no_write_binding_pass (void)
 | ||
| {
 | ||
|   TestObject *obj;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.binding");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   g_settings_bind (settings, "string", obj, "no-write", G_SETTINGS_BIND_SET);
 | ||
| 
 | ||
|   exit (0);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_no_write_binding (void)
 | ||
| {
 | ||
|   if (g_test_undefined ())
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/no-write-binding/subprocess/fail", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*property*is not writable*");
 | ||
|     }
 | ||
| 
 | ||
|   g_test_trap_subprocess ("/gsettings/no-write-binding/subprocess/pass", 0,
 | ||
|                           G_TEST_SUBPROCESS_DEFAULT);
 | ||
|   g_test_trap_assert_passed ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| key_changed_cb (GSettings *settings, const gchar *key, gpointer data)
 | ||
| {
 | ||
|   gboolean *b = data;
 | ||
|   (*b) = TRUE;
 | ||
| }
 | ||
| 
 | ||
| typedef struct
 | ||
| {
 | ||
|   const gchar *path;
 | ||
|   const gchar *root_group;
 | ||
|   const gchar *keyfile_group;
 | ||
|   const gchar *root_path;
 | ||
| } KeyfileTestData;
 | ||
| 
 | ||
| /*
 | ||
|  * Test that using a keyfile works
 | ||
|  */
 | ||
| static void
 | ||
| test_keyfile (Fixture       *fixture,
 | ||
|               gconstpointer  user_data)
 | ||
| {
 | ||
|   GSettingsBackend *kf_backend;
 | ||
|   GSettings *settings;
 | ||
|   GKeyFile *keyfile;
 | ||
|   gchar *str;
 | ||
|   gboolean writable;
 | ||
|   GError *error = NULL;
 | ||
|   gchar *data;
 | ||
|   gsize len;
 | ||
|   gboolean called = FALSE;
 | ||
|   gchar *keyfile_path = NULL, *store_path = NULL;
 | ||
| 
 | ||
|   keyfile_path = g_build_filename (fixture->tmp_dir, "keyfile", NULL);
 | ||
|   store_path = g_build_filename (keyfile_path, "gsettings.store", NULL);
 | ||
|   kf_backend = g_keyfile_settings_backend_new (store_path, "/", "root");
 | ||
|   settings = g_settings_new_with_backend ("org.gtk.test", kf_backend);
 | ||
|   g_object_unref (kf_backend);
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
|   str = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (str, ==, "Hello, earthlings");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   writable = g_settings_is_writable (settings, "greeting");
 | ||
|   g_assert_true (writable);
 | ||
|   g_settings_set (settings, "greeting", "s", "see if this works");
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (str, ==, "see if this works");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
|   g_settings_set (settings, "farewell", "s", "cheerio");
 | ||
|   g_settings_apply (settings);
 | ||
| 
 | ||
|   keyfile = g_key_file_new ();
 | ||
|   g_assert_true (g_key_file_load_from_file (keyfile, store_path, 0, NULL));
 | ||
| 
 | ||
|   str = g_key_file_get_string (keyfile, "tests", "greeting", NULL);
 | ||
|   g_assert_cmpstr (str, ==, "'see if this works'");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   str = g_key_file_get_string (keyfile, "tests", "farewell", NULL);
 | ||
|   g_assert_cmpstr (str, ==, "'cheerio'");
 | ||
|   g_free (str);
 | ||
|   g_key_file_free (keyfile);
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
|   g_settings_apply (settings);
 | ||
|   keyfile = g_key_file_new ();
 | ||
|   g_assert_true (g_key_file_load_from_file (keyfile, store_path, 0, NULL));
 | ||
| 
 | ||
|   str = g_key_file_get_string (keyfile, "tests", "greeting", NULL);
 | ||
|   g_assert_null (str);
 | ||
| 
 | ||
|   called = FALSE;
 | ||
|   g_signal_connect (settings, "changed::greeting", G_CALLBACK (key_changed_cb), &called);
 | ||
| 
 | ||
|   g_key_file_set_string (keyfile, "tests", "greeting", "'howdy'");
 | ||
|   data = g_key_file_to_data (keyfile, &len, NULL);
 | ||
|   g_file_set_contents (store_path, data, len, &error);
 | ||
|   g_assert_no_error (error);
 | ||
|   while (!called)
 | ||
|     g_main_context_iteration (NULL, FALSE);
 | ||
|   g_signal_handlers_disconnect_by_func (settings, key_changed_cb, &called);
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (str, ==, "howdy");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   /* Now check setting a string without quotes */
 | ||
|   called = FALSE;
 | ||
|   g_signal_connect (settings, "changed::greeting", G_CALLBACK (key_changed_cb), &called);
 | ||
| 
 | ||
|   g_key_file_set_string (keyfile, "tests", "greeting", "he\"l🤗uń");
 | ||
|   g_free (data);
 | ||
|   data = g_key_file_to_data (keyfile, &len, NULL);
 | ||
|   g_file_set_contents (store_path, data, len, &error);
 | ||
|   g_assert_no_error (error);
 | ||
|   while (!called)
 | ||
|     g_main_context_iteration (NULL, FALSE);
 | ||
|   g_signal_handlers_disconnect_by_func (settings, key_changed_cb, &called);
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (str, ==, "he\"l🤗uń");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_settings_set (settings, "farewell", "s", "cheerio");
 | ||
| 
 | ||
|   /* Check that empty keys/groups are not allowed. */
 | ||
|   g_assert_false (g_settings_is_writable (settings, ""));
 | ||
|   g_assert_false (g_settings_is_writable (settings, "/"));
 | ||
| 
 | ||
|   /* When executing as root, changing the mode of the keyfile will have
 | ||
|    * no effect on the writability of the settings.
 | ||
|    */
 | ||
|   if (geteuid () != 0)
 | ||
|     {
 | ||
|       called = FALSE;
 | ||
|       g_signal_connect (settings, "writable-changed::greeting",
 | ||
|                         G_CALLBACK (key_changed_cb), &called);
 | ||
| 
 | ||
|       g_assert_no_errno (g_chmod (keyfile_path, 0500));
 | ||
|       while (!called)
 | ||
|         g_main_context_iteration (NULL, FALSE);
 | ||
|       g_signal_handlers_disconnect_by_func (settings, key_changed_cb, &called);
 | ||
| 
 | ||
|       writable = g_settings_is_writable (settings, "greeting");
 | ||
|       g_assert_false (writable);
 | ||
|     }
 | ||
| 
 | ||
|   g_key_file_free (keyfile);
 | ||
|   g_free (data);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   /* Clean up the temporary directory. */
 | ||
|   g_assert_no_errno (g_chmod (keyfile_path, 0777));
 | ||
|   g_assert_no_errno (g_remove (store_path));
 | ||
|   g_assert_no_errno (g_rmdir (keyfile_path));
 | ||
|   g_free (store_path);
 | ||
|   g_free (keyfile_path);
 | ||
| }
 | ||
| 
 | ||
| /*
 | ||
|  * Test that using a keyfile works with a schema with no path set.
 | ||
|  */
 | ||
| static void
 | ||
| test_keyfile_no_path (Fixture       *fixture,
 | ||
|                       gconstpointer  user_data)
 | ||
| {
 | ||
|   const KeyfileTestData *test_data = user_data;
 | ||
|   GSettingsBackend *kf_backend;
 | ||
|   GSettings *settings;
 | ||
|   GKeyFile *keyfile;
 | ||
|   gboolean writable;
 | ||
|   gchar *key = NULL;
 | ||
|   GError *error = NULL;
 | ||
|   gchar *keyfile_path = NULL, *store_path = NULL;
 | ||
| 
 | ||
|   keyfile_path = g_build_filename (fixture->tmp_dir, "keyfile", NULL);
 | ||
|   store_path = g_build_filename (keyfile_path, "gsettings.store", NULL);
 | ||
|   kf_backend = g_keyfile_settings_backend_new (store_path, test_data->root_path, test_data->root_group);
 | ||
|   settings = g_settings_new_with_backend_and_path ("org.gtk.test.no-path", kf_backend, test_data->path);
 | ||
|   g_object_unref (kf_backend);
 | ||
| 
 | ||
|   g_settings_reset (settings, "test-boolean");
 | ||
|   g_assert_true (g_settings_get_boolean (settings, "test-boolean"));
 | ||
| 
 | ||
|   writable = g_settings_is_writable (settings, "test-boolean");
 | ||
|   g_assert_true (writable);
 | ||
|   g_settings_set (settings, "test-boolean", "b", FALSE);
 | ||
| 
 | ||
|   g_assert_false (g_settings_get_boolean (settings, "test-boolean"));
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
|   g_settings_set (settings, "test-boolean", "b", TRUE);
 | ||
|   g_settings_apply (settings);
 | ||
| 
 | ||
|   keyfile = g_key_file_new ();
 | ||
|   g_assert_true (g_key_file_load_from_file (keyfile, store_path, 0, NULL));
 | ||
| 
 | ||
|   g_assert_true (g_key_file_get_boolean (keyfile, test_data->keyfile_group, "test-boolean", NULL));
 | ||
| 
 | ||
|   g_key_file_free (keyfile);
 | ||
| 
 | ||
|   g_settings_reset (settings, "test-boolean");
 | ||
|   g_settings_apply (settings);
 | ||
|   keyfile = g_key_file_new ();
 | ||
|   g_assert_true (g_key_file_load_from_file (keyfile, store_path, 0, NULL));
 | ||
| 
 | ||
|   g_assert_false (g_key_file_get_string (keyfile, test_data->keyfile_group, "test-boolean", &error));
 | ||
|   g_assert_error (error, G_KEY_FILE_ERROR, G_KEY_FILE_ERROR_KEY_NOT_FOUND);
 | ||
|   g_clear_error (&error);
 | ||
| 
 | ||
|   /* Check that empty keys/groups are not allowed. */
 | ||
|   g_assert_false (g_settings_is_writable (settings, ""));
 | ||
|   g_assert_false (g_settings_is_writable (settings, "/"));
 | ||
| 
 | ||
|   /* Keys which ghost the root group name are not allowed. This can only be
 | ||
|    * tested when the path is `/` as otherwise it acts as a prefix and prevents
 | ||
|    * any ghosting. */
 | ||
|   if (g_str_equal (test_data->path, "/"))
 | ||
|     {
 | ||
|       key = g_strdup_printf ("%s/%s", test_data->root_group, "");
 | ||
|       g_assert_false (g_settings_is_writable (settings, key));
 | ||
|       g_free (key);
 | ||
| 
 | ||
|       key = g_strdup_printf ("%s/%s", test_data->root_group, "/");
 | ||
|       g_assert_false (g_settings_is_writable (settings, key));
 | ||
|       g_free (key);
 | ||
| 
 | ||
|       key = g_strdup_printf ("%s/%s", test_data->root_group, "test-boolean");
 | ||
|       g_assert_false (g_settings_is_writable (settings, key));
 | ||
|       g_free (key);
 | ||
|     }
 | ||
| 
 | ||
|   g_key_file_free (keyfile);
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   /* Clean up the temporary directory. */
 | ||
|   g_assert_no_errno (g_chmod (keyfile_path, 0777));
 | ||
|   g_assert_no_errno (g_remove (store_path));
 | ||
|   g_assert_no_errno (g_rmdir (keyfile_path));
 | ||
|   g_free (store_path);
 | ||
|   g_free (keyfile_path);
 | ||
| }
 | ||
| 
 | ||
| /*
 | ||
|  * Test that a keyfile rejects writes to keys outside its root path.
 | ||
|  */
 | ||
| static void
 | ||
| test_keyfile_outside_root_path (Fixture       *fixture,
 | ||
|                                 gconstpointer  user_data)
 | ||
| {
 | ||
|   GSettingsBackend *kf_backend;
 | ||
|   GSettings *settings;
 | ||
|   gchar *keyfile_path = NULL, *store_path = NULL;
 | ||
| 
 | ||
|   keyfile_path = g_build_filename (fixture->tmp_dir, "keyfile", NULL);
 | ||
|   store_path = g_build_filename (keyfile_path, "gsettings.store", NULL);
 | ||
|   kf_backend = g_keyfile_settings_backend_new (store_path, "/tests/basic-types/", "root");
 | ||
|   settings = g_settings_new_with_backend_and_path ("org.gtk.test.no-path", kf_backend, "/tests/");
 | ||
|   g_object_unref (kf_backend);
 | ||
| 
 | ||
|   g_assert_false (g_settings_is_writable (settings, "test-boolean"));
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   /* Clean up the temporary directory. The keyfile probably doesn’t exist, so
 | ||
|    * don’t error on failure. */
 | ||
|   g_remove (store_path);
 | ||
|   g_assert_no_errno (g_rmdir (keyfile_path));
 | ||
|   g_free (store_path);
 | ||
|   g_free (keyfile_path);
 | ||
| }
 | ||
| 
 | ||
| /*
 | ||
|  * Test that a keyfile rejects writes to keys in the root if no root group is set.
 | ||
|  */
 | ||
| static void
 | ||
| test_keyfile_no_root_group (Fixture       *fixture,
 | ||
|                             gconstpointer  user_data)
 | ||
| {
 | ||
|   GSettingsBackend *kf_backend;
 | ||
|   GSettings *settings;
 | ||
|   gchar *keyfile_path = NULL, *store_path = NULL;
 | ||
| 
 | ||
|   keyfile_path = g_build_filename (fixture->tmp_dir, "keyfile", NULL);
 | ||
|   store_path = g_build_filename (keyfile_path, "gsettings.store", NULL);
 | ||
|   kf_backend = g_keyfile_settings_backend_new (store_path, "/", NULL);
 | ||
|   settings = g_settings_new_with_backend_and_path ("org.gtk.test.no-path", kf_backend, "/");
 | ||
|   g_object_unref (kf_backend);
 | ||
| 
 | ||
|   g_assert_false (g_settings_is_writable (settings, "test-boolean"));
 | ||
|   g_assert_true (g_settings_is_writable (settings, "child/test-boolean"));
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   /* Clean up the temporary directory. The keyfile probably doesn’t exist, so
 | ||
|    * don’t error on failure. */
 | ||
|   g_remove (store_path);
 | ||
|   g_assert_no_errno (g_rmdir (keyfile_path));
 | ||
|   g_free (store_path);
 | ||
|   g_free (keyfile_path);
 | ||
| }
 | ||
| 
 | ||
| /* Test that getting child schemas works
 | ||
|  */
 | ||
| static void
 | ||
| test_child_schema (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettings *child;
 | ||
|   guint8 byte;
 | ||
| 
 | ||
|   /* first establish some known conditions */
 | ||
|   settings = g_settings_new ("org.gtk.test.basic-types");
 | ||
|   g_settings_set (settings, "test-byte", "y", 36);
 | ||
| 
 | ||
|   g_settings_get (settings, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpint (byte, ==, 36);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   child = g_settings_get_child (settings, "basic-types");
 | ||
|   g_assert_nonnull (child);
 | ||
| 
 | ||
|   g_settings_get (child, "test-byte", "y", &byte);
 | ||
|   g_assert_cmpint (byte, ==, 36);
 | ||
| 
 | ||
|   g_object_unref (child);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| #include "../strinfo.c"
 | ||
| 
 | ||
| static void
 | ||
| test_strinfo (void)
 | ||
| {
 | ||
|   /*  "foo" has a value of 1
 | ||
|    *  "bar" has a value of 2
 | ||
|    *  "baz" is an alias for "bar"
 | ||
|    */
 | ||
|   gchar array[] =
 | ||
|     "\1\0\0\0"      "\xff""foo"     "\0\0\0\xff"    "\2\0\0\0"
 | ||
|     "\xff" "bar"    "\0\0\0\xff"    "\3\0\0\0"      "\xfe""baz"
 | ||
|     "\0\0\0\xff";
 | ||
|   const guint32 *strinfo = (guint32 *) array;
 | ||
|   guint length = sizeof array / 4;
 | ||
|   guint result = 0;
 | ||
| 
 | ||
|   {
 | ||
|     /* build it and compare */
 | ||
|     GString *builder;
 | ||
| 
 | ||
|     builder = g_string_new (NULL);
 | ||
|     strinfo_builder_append_item (builder, "foo", 1);
 | ||
|     strinfo_builder_append_item (builder, "bar", 2);
 | ||
|     g_assert_true (strinfo_builder_append_alias (builder, "baz", "bar"));
 | ||
|     g_assert_cmpmem (builder->str, builder->len, strinfo, length * 4);
 | ||
|     g_string_free (builder, TRUE);
 | ||
|   }
 | ||
| 
 | ||
|   g_assert_cmpstr (strinfo_string_from_alias (strinfo, length, "foo"),
 | ||
|                    ==, NULL);
 | ||
|   g_assert_cmpstr (strinfo_string_from_alias (strinfo, length, "bar"),
 | ||
|                    ==, NULL);
 | ||
|   g_assert_cmpstr (strinfo_string_from_alias (strinfo, length, "baz"),
 | ||
|                    ==, "bar");
 | ||
|   g_assert_cmpstr (strinfo_string_from_alias (strinfo, length, "quux"),
 | ||
|                    ==, NULL);
 | ||
| 
 | ||
|   g_assert_true (strinfo_enum_from_string (strinfo, length, "foo", &result));
 | ||
|   g_assert_cmpint (result, ==, 1);
 | ||
|   g_assert_true (strinfo_enum_from_string (strinfo, length, "bar", &result));
 | ||
|   g_assert_cmpint (result, ==, 2);
 | ||
|   g_assert_false (strinfo_enum_from_string (strinfo, length, "baz", &result));
 | ||
|   g_assert_false (strinfo_enum_from_string (strinfo, length, "quux", &result));
 | ||
| 
 | ||
|   g_assert_cmpstr (strinfo_string_from_enum (strinfo, length, 0), ==, NULL);
 | ||
|   g_assert_cmpstr (strinfo_string_from_enum (strinfo, length, 1), ==, "foo");
 | ||
|   g_assert_cmpstr (strinfo_string_from_enum (strinfo, length, 2), ==, "bar");
 | ||
|   g_assert_cmpstr (strinfo_string_from_enum (strinfo, length, 3), ==, NULL);
 | ||
| 
 | ||
|   g_assert_true (strinfo_is_string_valid (strinfo, length, "foo"));
 | ||
|   g_assert_true (strinfo_is_string_valid (strinfo, length, "bar"));
 | ||
|   g_assert_false (strinfo_is_string_valid (strinfo, length, "baz"));
 | ||
|   g_assert_false (strinfo_is_string_valid (strinfo, length, "quux"));
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_enums_non_enum_key (void)
 | ||
| {
 | ||
|   GSettings *direct;
 | ||
| 
 | ||
|   direct = g_settings_new ("org.gtk.test.enums.direct");
 | ||
|   g_settings_get_enum (direct, "test");
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_enums_non_enum_value (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_set_enum (settings, "test", 42);
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_enums_range (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_set_string (settings, "test", "qux");
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_enums_non_flags (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_get_flags (settings, "test");
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_enums (void)
 | ||
| {
 | ||
|   GSettings *settings, *direct;
 | ||
|   gchar *str;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   direct = g_settings_new ("org.gtk.test.enums.direct");
 | ||
| 
 | ||
|   if (g_test_undefined () && !backend_set)
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/enums/subprocess/non-enum-key", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*not associated with an enum*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/enums/subprocess/non-enum-value", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*invalid enum value 42*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/enums/subprocess/range", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*g_settings_set_value*valid range*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/enums/subprocess/non-flags", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*not associated with a flags*");
 | ||
|     }
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "test");
 | ||
|   g_assert_cmpstr (str, ==, "bar");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_settings_set_enum (settings, "test", TEST_ENUM_FOO);
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "test");
 | ||
|   g_assert_cmpstr (str, ==, "foo");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_enum (settings, "test"), ==, TEST_ENUM_FOO);
 | ||
| 
 | ||
|   g_settings_set_string (direct, "test", "qux");
 | ||
| 
 | ||
|   str = g_settings_get_string (direct, "test");
 | ||
|   g_assert_cmpstr (str, ==, "qux");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "test");
 | ||
|   g_assert_cmpstr (str, ==, "quux");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_enum (settings, "test"), ==, TEST_ENUM_QUUX);
 | ||
| 
 | ||
|   g_object_unref (direct);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_flags_non_flags_key (void)
 | ||
| {
 | ||
|   GSettings *direct;
 | ||
| 
 | ||
|   direct = g_settings_new ("org.gtk.test.enums.direct");
 | ||
|   g_settings_get_flags (direct, "test");
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_flags_non_flags_value (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_set_flags (settings, "f-test", 0x42);
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_flags_range (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_set_strv (settings, "f-test",
 | ||
|                        (const gchar **) g_strsplit ("rock", ",", 0));
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_flags_non_enum (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   g_settings_get_enum (settings, "f-test");
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_flags (void)
 | ||
| {
 | ||
|   GSettings *settings, *direct;
 | ||
|   gchar **strv;
 | ||
|   gchar *str;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   direct = g_settings_new ("org.gtk.test.enums.direct");
 | ||
| 
 | ||
|   if (g_test_undefined () && !backend_set)
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/flags/subprocess/non-flags-key", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*not associated with a flags*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/flags/subprocess/non-flags-value", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*invalid flags value 0x00000042*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/flags/subprocess/range", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*g_settings_set_value*valid range*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/flags/subprocess/non-enum", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*not associated with an enum*");
 | ||
|     }
 | ||
| 
 | ||
|   strv = g_settings_get_strv (settings, "f-test");
 | ||
|   str = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (str, ==, "");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_settings_set_flags (settings, "f-test",
 | ||
|                         TEST_FLAGS_WALKING | TEST_FLAGS_TALKING);
 | ||
| 
 | ||
|   strv = g_settings_get_strv (settings, "f-test");
 | ||
|   str = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (str, ==, "talking,walking");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_flags (settings, "f-test"), ==,
 | ||
|                    TEST_FLAGS_WALKING | TEST_FLAGS_TALKING);
 | ||
| 
 | ||
|   strv = g_strsplit ("speaking,laughing", ",", 0);
 | ||
|   g_settings_set_strv (direct, "f-test", (const gchar **) strv);
 | ||
|   g_strfreev (strv);
 | ||
| 
 | ||
|   strv = g_settings_get_strv (direct, "f-test");
 | ||
|   str = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (str, ==, "speaking,laughing");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   strv = g_settings_get_strv (settings, "f-test");
 | ||
|   str = g_strjoinv (",", strv);
 | ||
|   g_assert_cmpstr (str, ==, "talking,laughing");
 | ||
|   g_strfreev (strv);
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_flags (settings, "f-test"), ==,
 | ||
|                    TEST_FLAGS_TALKING | TEST_FLAGS_LAUGHING);
 | ||
| 
 | ||
|   g_object_unref (direct);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_range_high (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.range");
 | ||
|   g_settings_set_int (settings, "val", 45);
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_range_low (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.range");
 | ||
|   g_settings_set_int (settings, "val", 1);
 | ||
|   g_assert_not_reached ();
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_range (void)
 | ||
| {
 | ||
|   GSettings *settings, *direct;
 | ||
|   GVariant *value;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.range");
 | ||
|   direct = g_settings_new ("org.gtk.test.range.direct");
 | ||
| 
 | ||
|   if (g_test_undefined () && !backend_set)
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/range/subprocess/high", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*g_settings_set_value*valid range*");
 | ||
| 
 | ||
|       g_test_trap_subprocess ("/gsettings/range/subprocess/low", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_failed ();
 | ||
|       g_test_trap_assert_stderr ("*g_settings_set_value*valid range*");
 | ||
|     }
 | ||
| 
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "val"), ==, 33);
 | ||
|   g_settings_set_int (direct, "val", 22);
 | ||
|   g_assert_cmpint (g_settings_get_int (direct, "val"), ==, 22);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "val"), ==, 22);
 | ||
|   g_settings_set_int (direct, "val", 45);
 | ||
|   g_assert_cmpint (g_settings_get_int (direct, "val"), ==, 45);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "val"), ==, 33);
 | ||
|   g_settings_set_int (direct, "val", 1);
 | ||
|   g_assert_cmpint (g_settings_get_int (direct, "val"), ==, 1);
 | ||
|   g_assert_cmpint (g_settings_get_int (settings, "val"), ==, 33);
 | ||
| 
 | ||
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS
 | ||
|   value = g_variant_new_int32 (1);
 | ||
|   g_assert_false (g_settings_range_check (settings, "val", value));
 | ||
|   g_variant_unref (value);
 | ||
|   value = g_variant_new_int32 (33);
 | ||
|   g_assert_true (g_settings_range_check (settings, "val", value));
 | ||
|   g_variant_unref (value);
 | ||
|   value = g_variant_new_int32 (45);
 | ||
|   g_assert_false (g_settings_range_check (settings, "val", value));
 | ||
|   g_variant_unref (value);
 | ||
| G_GNUC_END_IGNORE_DEPRECATIONS
 | ||
| 
 | ||
|   g_object_unref (direct);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| strv_set_equal (const gchar * const *strv, ...)
 | ||
| {
 | ||
|   gsize count;
 | ||
|   va_list list;
 | ||
|   const gchar *str;
 | ||
|   gboolean res;
 | ||
| 
 | ||
|   res = TRUE;
 | ||
|   count = 0;
 | ||
|   va_start (list, strv);
 | ||
|   while (1)
 | ||
|     {
 | ||
|       str = va_arg (list, const gchar *);
 | ||
|       if (str == NULL)
 | ||
|         break;
 | ||
|       if (!g_strv_contains (strv, str))
 | ||
|         {
 | ||
|           res = FALSE;
 | ||
|           break;
 | ||
|         }
 | ||
|       count++;
 | ||
|     }
 | ||
|   va_end (list);
 | ||
| 
 | ||
|   if (res)
 | ||
|     res = g_strv_length ((gchar**)strv) == count;
 | ||
| 
 | ||
|   return res;
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_list_items (void)
 | ||
| {
 | ||
|   GSettingsSchema *schema;
 | ||
|   GSettings *settings;
 | ||
|   gchar **children;
 | ||
|   gchar **keys;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   children = g_settings_list_children (settings);
 | ||
|   keys = g_settings_schema_list_keys (schema);
 | ||
| 
 | ||
|   g_assert_true (strv_set_equal ((const gchar * const *) children, "basic-types", "complex-types", "localized", NULL));
 | ||
|   g_assert_true (strv_set_equal ((const gchar * const *) keys, "greeting", "farewell", NULL));
 | ||
| 
 | ||
|   g_strfreev (children);
 | ||
|   g_strfreev (keys);
 | ||
| 
 | ||
|   g_settings_schema_unref (schema);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_list_schemas (void)
 | ||
| {
 | ||
|   const gchar * const *schemas;
 | ||
|   const gchar * const *relocs;
 | ||
| 
 | ||
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS
 | ||
|   relocs = g_settings_list_relocatable_schemas ();
 | ||
|   schemas = g_settings_list_schemas ();
 | ||
| G_GNUC_END_IGNORE_DEPRECATIONS
 | ||
| 
 | ||
|   g_assert_true (strv_set_equal (relocs,
 | ||
|                                  "org.gtk.test.no-path",
 | ||
|                                  "org.gtk.test.extends.base",
 | ||
|                                  "org.gtk.test.extends.extended",
 | ||
|                                  NULL));
 | ||
| 
 | ||
|   g_assert_true (strv_set_equal (schemas,
 | ||
|                                  "org.gtk.test",
 | ||
|                                  "org.gtk.test.basic-types",
 | ||
|                                  "org.gtk.test.complex-types",
 | ||
|                                  "org.gtk.test.localized",
 | ||
|                                  "org.gtk.test.binding",
 | ||
|                                  "org.gtk.test.enums",
 | ||
|                                  "org.gtk.test.enums.direct",
 | ||
|                                  "org.gtk.test.range",
 | ||
|                                  "org.gtk.test.range.direct",
 | ||
|                                  "org.gtk.test.mapped",
 | ||
|                                  "org.gtk.test.descriptions",
 | ||
|                                  "org.gtk.test.per-desktop",
 | ||
|                                  NULL));
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| map_func (GVariant *value,
 | ||
|           gpointer *result,
 | ||
|           gpointer  user_data)
 | ||
| {
 | ||
|   gint *state = user_data;
 | ||
|   gint v;
 | ||
| 
 | ||
|   if (value)
 | ||
|     v = g_variant_get_int32 (value);
 | ||
|   else
 | ||
|     v = -1;
 | ||
| 
 | ||
|   if (*state == 0)
 | ||
|     {
 | ||
|       g_assert_cmpint (v, ==, 1);
 | ||
|       (*state)++;
 | ||
|       return FALSE;
 | ||
|     }
 | ||
|   else if (*state == 1)
 | ||
|     {
 | ||
|       g_assert_cmpint (v, ==, 0);
 | ||
|       (*state)++;
 | ||
|       return FALSE;
 | ||
|     }
 | ||
|   else
 | ||
|     {
 | ||
|       g_assert_null (value);
 | ||
|       *result = g_variant_new_int32 (5);
 | ||
|       return TRUE;
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_get_mapped (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   gint state;
 | ||
|   gpointer p;
 | ||
|   gint val;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.mapped");
 | ||
|   g_settings_set_int (settings, "val", 1);
 | ||
| 
 | ||
|   state = 0;
 | ||
|   p = g_settings_get_mapped (settings, "val", map_func, &state);
 | ||
|   val = g_variant_get_int32 ((GVariant*)p);
 | ||
|   g_assert_cmpint (val, ==, 5);
 | ||
| 
 | ||
|   g_variant_unref (p);
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_get_range (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GVariant *range;
 | ||
| 
 | ||
| G_GNUC_BEGIN_IGNORE_DEPRECATIONS
 | ||
|   settings = g_settings_new ("org.gtk.test.range");
 | ||
|   range = g_settings_get_range (settings, "val");
 | ||
|   check_and_free (range, "('range', <(2, 44)>)");
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   range = g_settings_get_range (settings, "test");
 | ||
|   check_and_free (range, "('enum', <['foo', 'bar', 'baz', 'quux']>)");
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.enums");
 | ||
|   range = g_settings_get_range (settings, "f-test");
 | ||
|   check_and_free (range, "('flags', "
 | ||
|                   "<['mourning', 'laughing', 'talking', 'walking']>)");
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   range = g_settings_get_range (settings, "greeting");
 | ||
|   check_and_free (range, "('type', <@as []>)");
 | ||
|   g_object_unref (settings);
 | ||
| G_GNUC_END_IGNORE_DEPRECATIONS
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_schema_source (void)
 | ||
| {
 | ||
|   GSettingsSchemaSource *parent;
 | ||
|   GSettingsSchemaSource *source;
 | ||
|   GSettingsBackend *backend;
 | ||
|   GSettingsSchema *schema;
 | ||
|   GError *error = NULL;
 | ||
|   GSettings *settings, *child;
 | ||
|   gboolean enabled;
 | ||
| 
 | ||
|   backend = g_settings_backend_get_default ();
 | ||
| 
 | ||
|   /* make sure it fails properly */
 | ||
|   parent = g_settings_schema_source_get_default ();
 | ||
|   source = g_settings_schema_source_new_from_directory ("/path/that/does/not/exist", parent,  TRUE, &error);
 | ||
|   g_assert_null (source);
 | ||
|   g_assert_error (error, G_FILE_ERROR, G_FILE_ERROR_NOENT);
 | ||
|   g_clear_error (&error);
 | ||
| 
 | ||
|   /* Test error handling of corrupt compiled files. */
 | ||
|   source = g_settings_schema_source_new_from_directory ("schema-source-corrupt", parent, TRUE, &error);
 | ||
|   g_assert_error (error, G_FILE_ERROR, G_FILE_ERROR_INVAL);
 | ||
|   g_assert_null (source);
 | ||
|   g_clear_error (&error);
 | ||
| 
 | ||
|   /* Test error handling of empty compiled files. */
 | ||
|   source = g_settings_schema_source_new_from_directory ("schema-source-empty", parent, TRUE, &error);
 | ||
|   g_assert_error (error, G_FILE_ERROR, G_FILE_ERROR_INVAL);
 | ||
|   g_assert_null (source);
 | ||
|   g_clear_error (&error);
 | ||
| 
 | ||
|   /* create a source with the parent */
 | ||
|   source = g_settings_schema_source_new_from_directory ("schema-source", parent, TRUE, &error);
 | ||
|   g_assert_no_error (error);
 | ||
|   g_assert_nonnull (source);
 | ||
| 
 | ||
|   /* check recursive lookups are working */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.test", TRUE);
 | ||
|   g_assert_nonnull (schema);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| 
 | ||
|   /* check recursive lookups for non-existent schemas */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.doesnotexist", TRUE);
 | ||
|   g_assert_null (schema);
 | ||
| 
 | ||
|   /* check non-recursive for schema that only exists in lower layers */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.test", FALSE);
 | ||
|   g_assert_null (schema);
 | ||
| 
 | ||
|   /* check non-recursive lookup for non-existent */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.doesnotexist", FALSE);
 | ||
|   g_assert_null (schema);
 | ||
| 
 | ||
|   /* check non-recursive for schema that exists in toplevel */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.schemasourcecheck", FALSE);
 | ||
|   g_assert_nonnull (schema);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| 
 | ||
|   /* check recursive for schema that exists in toplevel */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.schemasourcecheck", TRUE);
 | ||
|   g_assert_nonnull (schema);
 | ||
| 
 | ||
|   /* try to use it for something */
 | ||
|   settings = g_settings_new_full (schema, backend, "/test/");
 | ||
|   g_settings_schema_unref (schema);
 | ||
|   enabled = FALSE;
 | ||
|   g_settings_get (settings, "enabled", "b", &enabled);
 | ||
|   g_assert_true (enabled);
 | ||
| 
 | ||
|   /* Check that child schemas are resolved from the correct schema source, see glib#1884 */
 | ||
|   child = g_settings_get_child (settings, "child");
 | ||
|   g_settings_get (settings, "enabled", "b", &enabled);
 | ||
| 
 | ||
|   g_object_unref (child);
 | ||
|   g_object_unref (settings);
 | ||
|   g_settings_schema_source_unref (source);
 | ||
| 
 | ||
|   /* try again, but with no parent */
 | ||
|   source = g_settings_schema_source_new_from_directory ("schema-source", NULL, FALSE, NULL);
 | ||
|   g_assert_nonnull (source);
 | ||
| 
 | ||
|   /* should not find it this time, even if recursive... */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.test", FALSE);
 | ||
|   g_assert_null (schema);
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.test", TRUE);
 | ||
|   g_assert_null (schema);
 | ||
| 
 | ||
|   /* should still find our own... */
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.schemasourcecheck", TRUE);
 | ||
|   g_assert_nonnull (schema);
 | ||
|   g_settings_schema_unref (schema);
 | ||
|   schema = g_settings_schema_source_lookup (source, "org.gtk.schemasourcecheck", FALSE);
 | ||
|   g_assert_nonnull (schema);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| 
 | ||
|   g_settings_schema_source_unref (source);
 | ||
|   g_object_unref (backend);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_schema_list_keys (void)
 | ||
| {
 | ||
|   gchar                 **keys;
 | ||
|   GSettingsSchemaSource  *src    = g_settings_schema_source_get_default ();
 | ||
|   GSettingsSchema        *schema = g_settings_schema_source_lookup (src, "org.gtk.test", TRUE);
 | ||
|   g_assert_nonnull (schema);
 | ||
| 
 | ||
|   keys = g_settings_schema_list_keys (schema);
 | ||
| 
 | ||
|   g_assert_true (strv_set_equal ((const gchar * const *) keys,
 | ||
|                                  "greeting",
 | ||
|                                  "farewell",
 | ||
|                                  NULL));
 | ||
| 
 | ||
|   g_strfreev (keys);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_actions (void)
 | ||
| {
 | ||
|   GAction *string, *toggle;
 | ||
|   gboolean c1, c2, c3;
 | ||
|   GSettings *settings;
 | ||
|   gchar *name;
 | ||
|   GVariantType *param_type;
 | ||
|   gboolean enabled;
 | ||
|   GVariantType *state_type;
 | ||
|   GVariant *state;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.basic-types");
 | ||
|   string = g_settings_create_action (settings, "test-string");
 | ||
|   toggle = g_settings_create_action (settings, "test-boolean");
 | ||
|   g_object_unref (settings); /* should be held by the actions */
 | ||
| 
 | ||
|   g_signal_connect (settings, "changed", G_CALLBACK (changed_cb2), &c1);
 | ||
|   g_signal_connect (string, "notify::state", G_CALLBACK (changed_cb2), &c2);
 | ||
|   g_signal_connect (toggle, "notify::state", G_CALLBACK (changed_cb2), &c3);
 | ||
| 
 | ||
|   c1 = c2 = c3 = FALSE;
 | ||
|   g_settings_set_string (settings, "test-string", "hello world");
 | ||
|   check_and_free (g_action_get_state (string), "'hello world'");
 | ||
|   g_assert_true (c1 && c2 && !c3);
 | ||
|   c1 = c2 = c3 = FALSE;
 | ||
| 
 | ||
|   g_action_activate (string, g_variant_new_string ("hihi"));
 | ||
|   check_and_free (g_settings_get_value (settings, "test-string"), "'hihi'");
 | ||
|   g_assert_true (c1 && c2 && !c3);
 | ||
|   c1 = c2 = c3 = FALSE;
 | ||
| 
 | ||
|   g_action_change_state (string, g_variant_new_string ("kthxbye"));
 | ||
|   check_and_free (g_settings_get_value (settings, "test-string"), "'kthxbye'");
 | ||
|   g_assert_true (c1 && c2 && !c3);
 | ||
|   c1 = c2 = c3 = FALSE;
 | ||
| 
 | ||
|   g_action_change_state (toggle, g_variant_new_boolean (TRUE));
 | ||
|   g_assert_true (g_settings_get_boolean (settings, "test-boolean"));
 | ||
|   g_assert_true (c1 && !c2 && c3);
 | ||
|   c1 = c2 = c3 = FALSE;
 | ||
| 
 | ||
|   g_action_activate (toggle, NULL);
 | ||
|   g_assert_false (g_settings_get_boolean (settings, "test-boolean"));
 | ||
|   g_assert_true (c1 && !c2 && c3);
 | ||
| 
 | ||
|   g_object_get (string,
 | ||
|                 "name", &name,
 | ||
|                 "parameter-type", ¶m_type,
 | ||
|                 "enabled", &enabled,
 | ||
|                 "state-type", &state_type,
 | ||
|                 "state", &state,
 | ||
|                 NULL);
 | ||
| 
 | ||
|   g_assert_cmpstr (name, ==, "test-string");
 | ||
|   g_assert_true (g_variant_type_equal (param_type, G_VARIANT_TYPE_STRING));
 | ||
|   g_assert_true (enabled);
 | ||
|   g_assert_true (g_variant_type_equal (state_type, G_VARIANT_TYPE_STRING));
 | ||
|   g_assert_cmpstr (g_variant_get_string (state, NULL), ==, "kthxbye");
 | ||
| 
 | ||
|   g_free (name);
 | ||
|   g_variant_type_free (param_type);
 | ||
|   g_variant_type_free (state_type);
 | ||
|   g_variant_unref (state);
 | ||
| 
 | ||
|   g_object_unref (string);
 | ||
|   g_object_unref (toggle);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_null_backend (void)
 | ||
| {
 | ||
|   GSettingsBackend *backend;
 | ||
|   GSettings *settings;
 | ||
|   gchar *str;
 | ||
|   gboolean writable;
 | ||
| 
 | ||
|   backend = g_null_settings_backend_new ();
 | ||
|   settings = g_settings_new_with_backend_and_path ("org.gtk.test", backend, "/tests/");
 | ||
| 
 | ||
|   g_object_get (settings, "schema-id", &str, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "org.gtk.test");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "goodbye world");
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   writable = g_settings_is_writable (settings, "greeting");
 | ||
|   g_assert_false (writable);
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
| 
 | ||
|   g_settings_delay (settings);
 | ||
|   g_settings_set (settings, "greeting", "s", "goodbye world");
 | ||
|   g_settings_apply (settings);
 | ||
|   settings_assert_cmpstr (settings, "greeting", ==, "Hello, earthlings");
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
|   g_object_unref (backend);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_memory_backend (void)
 | ||
| {
 | ||
|   GSettingsBackend *backend;
 | ||
| 
 | ||
|   backend = g_memory_settings_backend_new ();
 | ||
|   g_assert_true (G_IS_SETTINGS_BACKEND (backend));
 | ||
|   g_object_unref (backend);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_read_descriptions (void)
 | ||
| {
 | ||
|   GSettingsSchema *schema;
 | ||
|   GSettingsSchemaKey *key;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   key = g_settings_schema_get_key (schema, "greeting");
 | ||
| 
 | ||
|   g_assert_cmpstr (g_settings_schema_key_get_summary (key), ==, "A greeting");
 | ||
|   g_assert_cmpstr (g_settings_schema_key_get_description (key), ==, "Greeting of the invading martians");
 | ||
| 
 | ||
|   g_settings_schema_key_unref (key);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.descriptions");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   key = g_settings_schema_get_key (schema, "a");
 | ||
| 
 | ||
|   g_assert_cmpstr (g_settings_schema_key_get_summary (key), ==,
 | ||
|                    "a paragraph.\n\n"
 | ||
|                    "with some whitespace.\n\n"
 | ||
|                    "because not everyone has a great editor.\n\n"
 | ||
|                    "lots of space is as one.");
 | ||
| 
 | ||
|   g_settings_schema_key_unref (key);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_default_value (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GSettingsSchema *schema;
 | ||
|   GSettingsSchemaKey *key;
 | ||
|   GVariant *v;
 | ||
|   const gchar *str;
 | ||
|   gchar *s;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   key = g_settings_schema_get_key (schema, "greeting");
 | ||
|   g_settings_schema_unref (schema);
 | ||
|   g_settings_schema_key_ref (key);
 | ||
| 
 | ||
|   g_assert_true (g_variant_type_equal (g_settings_schema_key_get_value_type (key), G_VARIANT_TYPE_STRING));
 | ||
| 
 | ||
|   v = g_settings_schema_key_get_default_value (key);
 | ||
|   str = g_variant_get_string (v, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "Hello, earthlings");
 | ||
|   g_variant_unref (v);
 | ||
| 
 | ||
|   g_settings_schema_key_unref (key);
 | ||
|   g_settings_schema_key_unref (key);
 | ||
| 
 | ||
|   g_settings_set (settings, "greeting", "s", "goodbye world");
 | ||
| 
 | ||
|   v = g_settings_get_user_value (settings, "greeting");
 | ||
|   str = g_variant_get_string (v, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "goodbye world");
 | ||
|   g_variant_unref (v);
 | ||
| 
 | ||
|   v = g_settings_get_default_value (settings, "greeting");
 | ||
|   str = g_variant_get_string (v, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "Hello, earthlings");
 | ||
|   g_variant_unref (v);
 | ||
| 
 | ||
|   g_settings_reset (settings, "greeting");
 | ||
| 
 | ||
|   v = g_settings_get_user_value (settings, "greeting");
 | ||
|   g_assert_null (v);
 | ||
| 
 | ||
|   s = g_settings_get_string (settings, "greeting");
 | ||
|   g_assert_cmpstr (s, ==, "Hello, earthlings");
 | ||
|   g_free (s);
 | ||
| 
 | ||
|   g_object_unref (settings);
 | ||
| }
 | ||
| 
 | ||
| static gboolean
 | ||
| string_map_func (GVariant *value,
 | ||
|                  gpointer *result,
 | ||
|                  gpointer  user_data)
 | ||
| {
 | ||
|   const gchar *str;
 | ||
| 
 | ||
|   str = g_variant_get_string (value, NULL);
 | ||
|   *result = g_variant_new_string (str);
 | ||
| 
 | ||
|   return TRUE;
 | ||
| }
 | ||
| 
 | ||
| /* Test that per-desktop values from org.gtk.test.gschema.override
 | ||
|  * does not change default value if current desktop is not listed in
 | ||
|  * $XDG_CURRENT_DESKTOP.
 | ||
|  */
 | ||
| static void
 | ||
| test_per_desktop (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GAction *action_string;
 | ||
|   TestObject *obj;
 | ||
|   gpointer p;
 | ||
|   gchar *str;
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.per-desktop");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   if (!g_test_subprocess ())
 | ||
|     {
 | ||
|       g_test_trap_subprocess ("/gsettings/per-desktop/subprocess", 0,
 | ||
|                               G_TEST_SUBPROCESS_DEFAULT);
 | ||
|       g_test_trap_assert_passed ();
 | ||
|     }
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "desktop");
 | ||
|   g_assert_cmpstr (str, ==, "GNOME");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   p = g_settings_get_mapped (settings, "desktop", string_map_func, NULL);
 | ||
| 
 | ||
|   str = g_variant_dup_string (p, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "GNOME");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_variant_unref (p);
 | ||
| 
 | ||
|   g_settings_bind (settings, "desktop", obj, "string", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_get (obj, "string", &str, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "GNOME");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   action_string = g_settings_create_action (settings, "desktop");
 | ||
|   check_and_free (g_action_get_state (action_string), "'GNOME'");
 | ||
| 
 | ||
|   g_clear_object (&action_string);
 | ||
|   g_object_unref (settings);
 | ||
|   g_object_unref (obj);
 | ||
| }
 | ||
| 
 | ||
| /* Test that per-desktop values from org.gtk.test.gschema.override
 | ||
|  * are successfully loaded based on the value of $XDG_CURRENT_DESKTOP.
 | ||
|  */
 | ||
| static void
 | ||
| test_per_desktop_subprocess (void)
 | ||
| {
 | ||
|   GSettings *settings;
 | ||
|   GAction *action_string;
 | ||
|   TestObject *obj;
 | ||
|   gpointer p;
 | ||
|   gchar *str;
 | ||
| 
 | ||
|   g_setenv ("XDG_CURRENT_DESKTOP", "GNOME-Classic:GNOME", TRUE);
 | ||
| 
 | ||
|   settings = g_settings_new ("org.gtk.test.per-desktop");
 | ||
|   obj = test_object_new ();
 | ||
| 
 | ||
|   str = g_settings_get_string (settings, "desktop");
 | ||
|   g_assert_cmpstr (str, ==, "GNOME Classic");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   p = g_settings_get_mapped (settings, "desktop", string_map_func, NULL);
 | ||
| 
 | ||
|   str = g_variant_dup_string (p, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "GNOME Classic");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   g_variant_unref (p);
 | ||
| 
 | ||
|   g_settings_bind (settings, "desktop", obj, "string", G_SETTINGS_BIND_DEFAULT);
 | ||
| 
 | ||
|   g_object_get (obj, "string", &str, NULL);
 | ||
|   g_assert_cmpstr (str, ==, "GNOME Classic");
 | ||
|   g_free (str);
 | ||
| 
 | ||
|   action_string = g_settings_create_action (settings, "desktop");
 | ||
|   check_and_free (g_action_get_state (action_string), "'GNOME Classic'");
 | ||
| 
 | ||
|   g_clear_object (&action_string);
 | ||
|   g_object_unref (settings);
 | ||
|   g_object_unref (obj);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_extended_schema (void)
 | ||
| {
 | ||
|   GSettingsSchema *schema;
 | ||
|   GSettings *settings;
 | ||
|   gchar **keys;
 | ||
| 
 | ||
|   settings = g_settings_new_with_path ("org.gtk.test.extends.extended", "/test/extends/");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   keys = g_settings_schema_list_keys (schema);
 | ||
|   g_assert_true (strv_set_equal ((const gchar * const *) keys, "int32", "string", "another-int32", NULL));
 | ||
|   g_strfreev (keys);
 | ||
|   g_object_unref (settings);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| }
 | ||
| 
 | ||
| static void
 | ||
| test_extended_schema_has_key (void)
 | ||
| {
 | ||
|   GSettingsSchema *schema;
 | ||
|   GSettings *settings;
 | ||
| 
 | ||
|   settings = g_settings_new_with_path ("org.gtk.test.extends.extended", "/test/extends/");
 | ||
|   g_object_get (settings, "settings-schema", &schema, NULL);
 | ||
|   g_assert_true (g_settings_schema_has_key (schema, "int32"));
 | ||
|   g_assert_true (g_settings_schema_has_key (schema, "string"));
 | ||
|   g_assert_true (g_settings_schema_has_key (schema, "another-int32"));
 | ||
|   g_object_unref (settings);
 | ||
|   g_settings_schema_unref (schema);
 | ||
| }
 | ||
| 
 | ||
| int
 | ||
| main (int argc, char *argv[])
 | ||
| {
 | ||
|   gchar *schema_text;
 | ||
|   gchar *override_text;
 | ||
|   gchar *enums;
 | ||
|   gint result;
 | ||
|   const KeyfileTestData keyfile_test_data_explicit_path = { "/tests/", "root", "tests", "/" };
 | ||
|   const KeyfileTestData keyfile_test_data_empty_path = { "/", "root", "root", "/" };
 | ||
|   const KeyfileTestData keyfile_test_data_long_path = {
 | ||
|     "/tests/path/is/very/long/and/this/makes/some/comparisons/take/a/different/branch/",
 | ||
|     "root",
 | ||
|     "tests/path/is/very/long/and/this/makes/some/comparisons/take/a/different/branch",
 | ||
|     "/"
 | ||
|   };
 | ||
| 
 | ||
| /* Meson build sets this */
 | ||
| #ifdef TEST_LOCALE_PATH
 | ||
|   if (g_str_has_suffix (TEST_LOCALE_PATH, "LC_MESSAGES"))
 | ||
|     {
 | ||
|       locale_dir = TEST_LOCALE_PATH G_DIR_SEPARATOR_S ".." G_DIR_SEPARATOR_S "..";
 | ||
|     }
 | ||
| #endif
 | ||
| 
 | ||
|   setlocale (LC_ALL, "");
 | ||
| 
 | ||
|   g_test_init (&argc, &argv, G_TEST_OPTION_ISOLATE_DIRS, NULL);
 | ||
| 
 | ||
|   if (!g_test_subprocess ())
 | ||
|     {
 | ||
|       GError *local_error = NULL;
 | ||
|       char *subprocess_stdout = NULL;
 | ||
| 
 | ||
|       /* A GVDB header is 6 guint32s, and requires a magic number in the first
 | ||
|        * two guint32s. A set of zero bytes of a greater length is considered
 | ||
|        * corrupt. */
 | ||
|       const guint8 gschemas_compiled_corrupt[sizeof (guint32) * 7] = { 0, };
 | ||
| 
 | ||
|       backend_set = g_getenv ("GSETTINGS_BACKEND") != NULL;
 | ||
| 
 | ||
|       g_setenv ("XDG_DATA_DIRS", ".", TRUE);
 | ||
|       g_setenv ("XDG_DATA_HOME", ".", TRUE);
 | ||
|       g_setenv ("GSETTINGS_SCHEMA_DIR", ".", TRUE);
 | ||
|       g_setenv ("XDG_CURRENT_DESKTOP", "", TRUE);
 | ||
| 
 | ||
|       if (!backend_set)
 | ||
|         g_setenv ("GSETTINGS_BACKEND", "memory", TRUE);
 | ||
| 
 | ||
|       g_remove ("org.gtk.test.enums.xml");
 | ||
|       /* #GLIB_MKENUMS is defined in meson.build */
 | ||
|       g_assert_true (g_spawn_command_line_sync (GLIB_MKENUMS " "
 | ||
|                                                 "--template " SRCDIR "/enums.xml.template "
 | ||
|                                                 SRCDIR "/testenum.h",
 | ||
|                                                 &enums, NULL, &result, NULL));
 | ||
|       g_assert_cmpint (result, ==, 0);
 | ||
|       g_assert_true (g_file_set_contents ("org.gtk.test.enums.xml", enums, -1, NULL));
 | ||
|       g_free (enums);
 | ||
| 
 | ||
|       g_assert_true (g_file_get_contents (SRCDIR "/org.gtk.test.gschema.xml.orig", &schema_text, NULL, NULL));
 | ||
|       g_assert_true (g_file_set_contents ("org.gtk.test.gschema.xml", schema_text, -1, NULL));
 | ||
|       g_free (schema_text);
 | ||
| 
 | ||
|       g_assert_true (g_file_get_contents (SRCDIR "/org.gtk.test.gschema.override.orig", &override_text, NULL, NULL));
 | ||
|       g_assert_true (g_file_set_contents ("org.gtk.test.gschema.override", override_text, -1, NULL));
 | ||
|       g_free (override_text);
 | ||
| 
 | ||
|       g_remove ("gschemas.compiled");
 | ||
|       /* #GLIB_COMPILE_SCHEMAS is defined in meson.build */
 | ||
|       g_assert_true (g_spawn_command_line_sync (GLIB_COMPILE_SCHEMAS " --targetdir=. "
 | ||
|                                                 "--schema-file=org.gtk.test.enums.xml "
 | ||
|                                                 "--schema-file=org.gtk.test.gschema.xml "
 | ||
|                                                 "--override-file=org.gtk.test.gschema.override",
 | ||
|                                                 &subprocess_stdout, NULL, &result, NULL));
 | ||
|       if (subprocess_stdout && *g_strstrip (subprocess_stdout) != '\0')
 | ||
|         g_test_message ("%s", subprocess_stdout);
 | ||
|       g_clear_pointer (&subprocess_stdout, g_free);
 | ||
|       g_assert_cmpint (result, ==, 0);
 | ||
| 
 | ||
|       g_remove ("schema-source/gschemas.compiled");
 | ||
|       g_mkdir ("schema-source", 0777);
 | ||
|       g_assert_true (g_spawn_command_line_sync (GLIB_COMPILE_SCHEMAS " --targetdir=schema-source "
 | ||
|                                                 "--schema-file=" SRCDIR "/org.gtk.schemasourcecheck.gschema.xml",
 | ||
|                                                 &subprocess_stdout, NULL, &result, NULL));
 | ||
|       if (subprocess_stdout && *g_strstrip (subprocess_stdout) != '\0')
 | ||
|         g_test_message ("%s", subprocess_stdout);
 | ||
|       g_clear_pointer (&subprocess_stdout, g_free);
 | ||
|       g_assert_cmpint (result, ==, 0);
 | ||
| 
 | ||
|       g_remove ("schema-source-corrupt/gschemas.compiled");
 | ||
|       g_mkdir ("schema-source-corrupt", 0777);
 | ||
|       g_file_set_contents ("schema-source-corrupt/gschemas.compiled",
 | ||
|                            (const gchar *) gschemas_compiled_corrupt,
 | ||
|                            sizeof (gschemas_compiled_corrupt),
 | ||
|                            &local_error);
 | ||
|       g_assert_no_error (local_error);
 | ||
| 
 | ||
|       g_remove ("schema-source-empty/gschemas.compiled");
 | ||
|       g_mkdir ("schema-source-empty", 0777);
 | ||
|       g_file_set_contents ("schema-source-empty/gschemas.compiled",
 | ||
|                            "", 0,
 | ||
|                            &local_error);
 | ||
|       g_assert_no_error (local_error);
 | ||
|    }
 | ||
| 
 | ||
|   g_test_add_func ("/gsettings/basic", test_basic);
 | ||
| 
 | ||
|   if (!backend_set)
 | ||
|     {
 | ||
|       g_test_add_func ("/gsettings/no-schema", test_no_schema);
 | ||
|       g_test_add_func ("/gsettings/unknown-key", test_unknown_key);
 | ||
|       g_test_add_func ("/gsettings/wrong-type", test_wrong_type);
 | ||
|       g_test_add_func ("/gsettings/wrong-path", test_wrong_path);
 | ||
|       g_test_add_func ("/gsettings/no-path", test_no_path);
 | ||
|     }
 | ||
| 
 | ||
|   g_test_add_func ("/gsettings/basic-types", test_basic_types);
 | ||
|   g_test_add_func ("/gsettings/complex-types", test_complex_types);
 | ||
|   g_test_add_func ("/gsettings/changes", test_changes);
 | ||
| 
 | ||
|   g_test_add_func ("/gsettings/l10n", test_l10n);
 | ||
|   g_test_add_func ("/gsettings/l10n-context", test_l10n_context);
 | ||
|   g_test_add_func ("/gsettings/l10n-time", test_l10n_time);
 | ||
| 
 | ||
|   g_test_add_func ("/gsettings/delay-apply", test_delay_apply);
 | ||
|   g_test_add_func ("/gsettings/delay-revert", test_delay_revert);
 | ||
|   g_test_add_func ("/gsettings/delay-child", test_delay_child);
 | ||
|   g_test_add_func ("/gsettings/delay-reset-key", test_delay_reset_key);
 | ||
|   g_test_add_func ("/gsettings/atomic", test_atomic);
 | ||
| 
 | ||
|   g_test_add_func ("/gsettings/simple-binding", test_simple_binding);
 | ||
|   g_test_add_func ("/gsettings/directional-binding", test_directional_binding);
 | ||
|   g_test_add_func ("/gsettings/custom-binding", test_custom_binding);
 | ||
|   g_test_add_func ("/gsettings/bind-with-mapping-closures", test_bind_with_mapping_closures);
 | ||
|   g_test_add_func ("/gsettings/bind-with-mapping-closures-parameters", test_bind_with_mapping_closures_parameters);
 | ||
|   g_test_add_func ("/gsettings/no-change-binding", test_no_change_binding);
 | ||
|   g_test_add_func ("/gsettings/unbinding", test_unbind);
 | ||
|   g_test_add_func ("/gsettings/writable-binding", test_bind_writable);
 | ||
| 
 | ||
|   if (!backend_set)
 | ||
|     {
 | ||
|       g_test_add_func ("/gsettings/typesafe-binding", test_typesafe_binding);
 | ||
|       g_test_add_func ("/gsettings/no-read-binding", test_no_read_binding);
 | ||
|       g_test_add_func ("/gsettings/no-read-binding/subprocess/fail", test_no_read_binding_fail);
 | ||
|       g_test_add_func ("/gsettings/no-read-binding/subprocess/pass", test_no_read_binding_pass);
 | ||
|       g_test_add_func ("/gsettings/no-write-binding", test_no_write_binding);
 | ||
|       g_test_add_func ("/gsettings/no-write-binding/subprocess/fail", test_no_write_binding_fail);
 | ||
|       g_test_add_func ("/gsettings/no-write-binding/subprocess/pass", test_no_write_binding_pass);
 | ||
|     }
 | ||
| 
 | ||
|   g_test_add ("/gsettings/keyfile", Fixture, NULL, setup, test_keyfile, teardown);
 | ||
|   g_test_add ("/gsettings/keyfile/explicit-path", Fixture, &keyfile_test_data_explicit_path, setup, test_keyfile_no_path, teardown);
 | ||
|   g_test_add ("/gsettings/keyfile/empty-path", Fixture, &keyfile_test_data_empty_path, setup, test_keyfile_no_path, teardown);
 | ||
|   g_test_add ("/gsettings/keyfile/long-path", Fixture, &keyfile_test_data_long_path, setup, test_keyfile_no_path, teardown);
 | ||
|   g_test_add ("/gsettings/keyfile/outside-root-path", Fixture, NULL, setup, test_keyfile_outside_root_path, teardown);
 | ||
|   g_test_add ("/gsettings/keyfile/no-root-group", Fixture, NULL, setup, test_keyfile_no_root_group, teardown);
 | ||
|   g_test_add_func ("/gsettings/child-schema", test_child_schema);
 | ||
|   g_test_add_func ("/gsettings/strinfo", test_strinfo);
 | ||
|   g_test_add_func ("/gsettings/enums", test_enums);
 | ||
|   g_test_add_func ("/gsettings/enums/subprocess/non-enum-key", test_enums_non_enum_key);
 | ||
|   g_test_add_func ("/gsettings/enums/subprocess/non-enum-value", test_enums_non_enum_value);
 | ||
|   g_test_add_func ("/gsettings/enums/subprocess/range", test_enums_range);
 | ||
|   g_test_add_func ("/gsettings/enums/subprocess/non-flags", test_enums_non_flags);
 | ||
|   g_test_add_func ("/gsettings/flags", test_flags);
 | ||
|   g_test_add_func ("/gsettings/flags/subprocess/non-flags-key", test_flags_non_flags_key);
 | ||
|   g_test_add_func ("/gsettings/flags/subprocess/non-flags-value", test_flags_non_flags_value);
 | ||
|   g_test_add_func ("/gsettings/flags/subprocess/range", test_flags_range);
 | ||
|   g_test_add_func ("/gsettings/flags/subprocess/non-enum", test_flags_non_enum);
 | ||
|   g_test_add_func ("/gsettings/range", test_range);
 | ||
|   g_test_add_func ("/gsettings/range/subprocess/high", test_range_high);
 | ||
|   g_test_add_func ("/gsettings/range/subprocess/low", test_range_low);
 | ||
|   g_test_add_func ("/gsettings/list-items", test_list_items);
 | ||
|   g_test_add_func ("/gsettings/list-schemas", test_list_schemas);
 | ||
|   g_test_add_func ("/gsettings/mapped", test_get_mapped);
 | ||
|   g_test_add_func ("/gsettings/get-range", test_get_range);
 | ||
|   g_test_add_func ("/gsettings/schema-source", test_schema_source);
 | ||
|   g_test_add_func ("/gsettings/schema-list-keys", test_schema_list_keys);
 | ||
|   g_test_add_func ("/gsettings/actions", test_actions);
 | ||
|   g_test_add_func ("/gsettings/null-backend", test_null_backend);
 | ||
|   g_test_add_func ("/gsettings/memory-backend", test_memory_backend);
 | ||
|   g_test_add_func ("/gsettings/read-descriptions", test_read_descriptions);
 | ||
|   g_test_add_func ("/gsettings/test-extended-schema", test_extended_schema);
 | ||
|   g_test_add_func ("/gsettings/test-extended-schema-has-key", test_extended_schema_has_key);
 | ||
|   g_test_add_func ("/gsettings/default-value", test_default_value);
 | ||
|   g_test_add_func ("/gsettings/per-desktop", test_per_desktop);
 | ||
|   g_test_add_func ("/gsettings/per-desktop/subprocess", test_per_desktop_subprocess);
 | ||
| 
 | ||
|   result = g_test_run ();
 | ||
| 
 | ||
|   g_settings_sync ();
 | ||
| 
 | ||
|   /* FIXME: Due to the way #GSettings objects can be used without specifying a
 | ||
|    * backend, the default backend is leaked. In order to be able to run this
 | ||
|    * test under valgrind and get meaningful checking for real leaks, use this
 | ||
|    * hack to drop the final reference to the default #GSettingsBackend.
 | ||
|    *
 | ||
|    * This should not be used in production code. */
 | ||
|     {
 | ||
|       GSettingsBackend *backend;
 | ||
| 
 | ||
|       backend = g_settings_backend_get_default ();
 | ||
|       g_object_unref (backend);  /* reference from the *_get_default() call */
 | ||
|       g_assert_finalize_object (backend);  /* singleton reference owned by GLib */
 | ||
|     }
 | ||
| 
 | ||
|   return result;
 | ||
| }
 |