mirror of
https://gitlab.gnome.org/GNOME/glib.git
synced 2025-08-23 09:28:54 +02:00
some GSettings stuff
This commit is contained in:
@@ -250,6 +250,11 @@ libgio_2_0_la_SOURCES = \
|
||||
gpollfilemonitor.h \
|
||||
gresolver.c \
|
||||
gseekable.c \
|
||||
gsettings.c \
|
||||
gsettingsschema.c \
|
||||
gdelayedsettingsbackend.c\
|
||||
gsettingsbackend.c \
|
||||
gmemorysettingsbackend.c\
|
||||
gsimpleasyncresult.c \
|
||||
gsocket.c \
|
||||
gsocketaddress.c \
|
||||
|
329
gio/gdelayedsettingsbackend.c
Normal file
329
gio/gdelayedsettingsbackend.c
Normal file
@@ -0,0 +1,329 @@
|
||||
/*
|
||||
* Copyright © 2009 Codethink Limited
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of version 3 of the GNU General Public License as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* See the included COPYING file for more information.
|
||||
*/
|
||||
|
||||
#include "gdelayedsettingsbackend.h"
|
||||
|
||||
#include <string.h>
|
||||
|
||||
#include "gioalias.h"
|
||||
|
||||
enum
|
||||
{
|
||||
PROP_NONE,
|
||||
PROP_BACKEND,
|
||||
PROP_BASE_PATH,
|
||||
PROP_HAS_UNAPPLIED
|
||||
};
|
||||
|
||||
struct _GDelayedSettingsBackendPrivate {
|
||||
GSettingsBackend *backend;
|
||||
guint handler_id;
|
||||
gchar *base_path;
|
||||
GTree *delayed;
|
||||
};
|
||||
|
||||
G_DEFINE_TYPE (GDelayedSettingsBackend,
|
||||
g_delayed_settings_backend,
|
||||
G_TYPE_SETTINGS_BACKEND)
|
||||
|
||||
static gboolean
|
||||
g_delayed_settings_backend_add_to_tree (gpointer key,
|
||||
gpointer value,
|
||||
gpointer user_data)
|
||||
{
|
||||
gpointer *args = user_data;
|
||||
|
||||
g_tree_insert (args[0],
|
||||
g_strjoin (NULL, args[1], key, NULL),
|
||||
g_variant_ref (value));
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_write (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *tree,
|
||||
gpointer origin_tag)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (backend);
|
||||
gconstpointer args[2] = { delayed->priv->delayed, prefix };
|
||||
gboolean was_empty;
|
||||
|
||||
was_empty = g_tree_nnodes (delayed->priv->delayed) == 0;
|
||||
|
||||
g_tree_foreach (tree, g_delayed_settings_backend_add_to_tree, args);
|
||||
|
||||
g_settings_backend_changed_tree (backend, prefix, tree, origin_tag);
|
||||
|
||||
if (was_empty)
|
||||
g_object_notify (G_OBJECT (delayed), "has-unapplied");
|
||||
}
|
||||
|
||||
static GVariant *
|
||||
g_delayed_settings_backend_read (GSettingsBackend *backend,
|
||||
const gchar *key,
|
||||
const GVariantType *expected_type)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (backend);
|
||||
GVariant *result = NULL;
|
||||
gchar *path;
|
||||
|
||||
if ((result = g_tree_lookup (delayed->priv->delayed, key)))
|
||||
return g_variant_ref (result);
|
||||
|
||||
path = g_strconcat (delayed->priv->base_path, key, NULL);
|
||||
result = g_settings_backend_read (delayed->priv->backend,
|
||||
path, expected_type);
|
||||
g_free (path);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
gboolean
|
||||
g_delayed_settings_backend_get_has_unapplied (GDelayedSettingsBackend *delayed)
|
||||
{
|
||||
return g_tree_nnodes (delayed->priv->delayed) > 0;
|
||||
}
|
||||
|
||||
void
|
||||
g_delayed_settings_backend_apply (GDelayedSettingsBackend *delayed)
|
||||
{
|
||||
if (g_tree_nnodes (delayed->priv->delayed))
|
||||
{
|
||||
GTree *tmp;
|
||||
|
||||
tmp = delayed->priv->delayed;
|
||||
delayed->priv->delayed = g_settings_backend_create_tree ();
|
||||
|
||||
g_settings_backend_write (delayed->priv->backend,
|
||||
delayed->priv->base_path,
|
||||
tmp, delayed->priv);
|
||||
g_tree_unref (tmp);
|
||||
|
||||
g_object_notify (G_OBJECT (delayed), "has-unapplied");
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
g_delayed_settings_backend_revert (GDelayedSettingsBackend *delayed)
|
||||
{
|
||||
if (g_tree_nnodes (delayed->priv->delayed))
|
||||
{
|
||||
GTree *tmp;
|
||||
|
||||
tmp = delayed->priv->delayed;
|
||||
delayed->priv->delayed = g_settings_backend_create_tree ();
|
||||
g_settings_backend_changed_tree (G_SETTINGS_BACKEND (delayed),
|
||||
"", tmp, NULL);
|
||||
g_tree_destroy (tmp);
|
||||
|
||||
g_object_notify (G_OBJECT (delayed), "has-unapplied");
|
||||
}
|
||||
}
|
||||
|
||||
static gboolean
|
||||
g_delayed_settings_backend_get_writable (GSettingsBackend *backend,
|
||||
const gchar *name)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (backend);
|
||||
gboolean sensitive;
|
||||
gchar *path;
|
||||
|
||||
path = g_strconcat (delayed->priv->base_path, name, NULL);
|
||||
sensitive = g_settings_backend_get_writable (delayed->priv->backend, path);
|
||||
g_free (path);
|
||||
|
||||
return sensitive;
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_subscribe (GSettingsBackend *base,
|
||||
const char *name)
|
||||
{
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_get_property (GObject *object, guint prop_id,
|
||||
GValue *value, GParamSpec *pspec)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (object);
|
||||
|
||||
switch (prop_id)
|
||||
{
|
||||
gboolean has;
|
||||
|
||||
case PROP_HAS_UNAPPLIED:
|
||||
has = g_delayed_settings_backend_get_has_unapplied (delayed);
|
||||
g_value_set_boolean (value, has);
|
||||
break;
|
||||
|
||||
default:
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_set_property (GObject *object, guint prop_id,
|
||||
const GValue *value, GParamSpec *pspec)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (object);
|
||||
|
||||
switch (prop_id)
|
||||
{
|
||||
case PROP_BACKEND:
|
||||
g_assert (delayed->priv->backend == NULL);
|
||||
delayed->priv->backend = g_value_dup_object (value);
|
||||
break;
|
||||
|
||||
case PROP_BASE_PATH:
|
||||
g_assert (delayed->priv->base_path == NULL);
|
||||
delayed->priv->base_path = g_value_dup_string (value);
|
||||
break;
|
||||
|
||||
default:
|
||||
g_assert_not_reached ();
|
||||
}
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_backend_changed (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
const gchar * const *items,
|
||||
gint n_items,
|
||||
gpointer origin_tag,
|
||||
gpointer user_data)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (user_data);
|
||||
|
||||
if (origin_tag == delayed->priv)
|
||||
return;
|
||||
|
||||
if (g_str_has_prefix (prefix, delayed->priv->base_path))
|
||||
{
|
||||
g_settings_backend_changed (G_SETTINGS_BACKEND (delayed),
|
||||
prefix + strlen (delayed->priv->base_path),
|
||||
items, n_items, origin_tag);
|
||||
}
|
||||
|
||||
else if (g_str_has_prefix (delayed->priv->base_path, prefix))
|
||||
{
|
||||
const gchar **my_items;
|
||||
const gchar *relative;
|
||||
gint relative_length;
|
||||
gint i, j;
|
||||
|
||||
relative = delayed->priv->base_path + strlen (prefix);
|
||||
relative_length = strlen (relative);
|
||||
|
||||
my_items = g_new (const gchar *, n_items + 1);
|
||||
|
||||
for (i = j = 0; i < n_items; i++)
|
||||
if (g_str_has_prefix (items[i], relative))
|
||||
my_items[j++] = items[i] + relative_length;
|
||||
my_items[j] = NULL;
|
||||
|
||||
if (j > 0)
|
||||
g_settings_backend_changed (G_SETTINGS_BACKEND (delayed),
|
||||
"", my_items, j, origin_tag);
|
||||
g_free (my_items);
|
||||
}
|
||||
|
||||
else
|
||||
/* do nothing */;
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_constructed (GObject *object)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (object);
|
||||
|
||||
g_assert (delayed->priv->backend != NULL);
|
||||
g_assert (delayed->priv->base_path != NULL);
|
||||
|
||||
delayed->priv->handler_id =
|
||||
g_signal_connect (delayed->priv->backend, "changed",
|
||||
G_CALLBACK (g_delayed_settings_backend_backend_changed),
|
||||
delayed);
|
||||
|
||||
g_settings_backend_subscribe (delayed->priv->backend,
|
||||
delayed->priv->base_path);
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_finalize (GObject *object)
|
||||
{
|
||||
GDelayedSettingsBackend *delayed = G_DELAYED_SETTINGS_BACKEND (object);
|
||||
|
||||
g_signal_handler_disconnect (delayed->priv->backend,
|
||||
delayed->priv->handler_id);
|
||||
g_settings_backend_unsubscribe (delayed->priv->backend,
|
||||
delayed->priv->base_path);
|
||||
g_object_unref (delayed->priv->backend);
|
||||
g_free (delayed->priv->base_path);
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_class_init (GDelayedSettingsBackendClass *class)
|
||||
{
|
||||
GSettingsBackendClass *backend_class = G_SETTINGS_BACKEND_CLASS (class);
|
||||
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
||||
|
||||
g_type_class_add_private (class, sizeof (GDelayedSettingsBackendPrivate));
|
||||
|
||||
backend_class->write = g_delayed_settings_backend_write;
|
||||
backend_class->read = g_delayed_settings_backend_read;
|
||||
backend_class->get_writable = g_delayed_settings_backend_get_writable;
|
||||
backend_class->subscribe = g_delayed_settings_backend_subscribe;
|
||||
backend_class->unsubscribe = g_delayed_settings_backend_subscribe;
|
||||
|
||||
object_class->get_property = g_delayed_settings_backend_get_property;
|
||||
object_class->set_property = g_delayed_settings_backend_set_property;
|
||||
object_class->constructed = g_delayed_settings_backend_constructed;
|
||||
object_class->finalize = g_delayed_settings_backend_finalize;
|
||||
|
||||
g_object_class_install_property (object_class, PROP_BACKEND,
|
||||
g_param_spec_object ("backend", "backend backend", "backend",
|
||||
G_TYPE_SETTINGS_BACKEND, G_PARAM_CONSTRUCT_ONLY |
|
||||
G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
g_object_class_install_property (object_class, PROP_BASE_PATH,
|
||||
g_param_spec_string ("base-path", "base path", "base",
|
||||
"", G_PARAM_CONSTRUCT_ONLY |
|
||||
G_PARAM_WRITABLE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
g_object_class_install_property (object_class, PROP_HAS_UNAPPLIED,
|
||||
g_param_spec_boolean ("has-unapplied", "has unapplied", "unapplied",
|
||||
FALSE, G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
||||
|
||||
}
|
||||
|
||||
static void
|
||||
g_delayed_settings_backend_init (GDelayedSettingsBackend *delayed)
|
||||
{
|
||||
delayed->priv =
|
||||
G_TYPE_INSTANCE_GET_PRIVATE (delayed, G_TYPE_DELAYED_SETTINGS_BACKEND,
|
||||
GDelayedSettingsBackendPrivate);
|
||||
|
||||
delayed->priv->delayed = g_settings_backend_create_tree ();
|
||||
}
|
||||
|
||||
GSettingsBackend *
|
||||
g_delayed_settings_backend_new (GSettingsBackend *backend,
|
||||
const gchar *base_path)
|
||||
{
|
||||
return g_object_new (G_TYPE_DELAYED_SETTINGS_BACKEND,
|
||||
"backend", backend,
|
||||
"base-path", base_path,
|
||||
NULL);
|
||||
}
|
||||
|
||||
#define _gsettingsdelayedbackend_c_
|
||||
#include "gioaliasdef.c"
|
59
gio/gdelayedsettingsbackend.h
Normal file
59
gio/gdelayedsettingsbackend.h
Normal file
@@ -0,0 +1,59 @@
|
||||
/*
|
||||
* Copyright © 2009 Codethink Limited
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of version 3 of the GNU General Public License as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* See the included COPYING file for more information.
|
||||
*/
|
||||
|
||||
#ifndef _gsettingsdelayedbackend_h_
|
||||
#define _gsettingsdelayedbackend_h_
|
||||
|
||||
#include <glib-object.h>
|
||||
|
||||
#include <gio/gsettingsbackend.h>
|
||||
|
||||
#define G_TYPE_DELAYED_SETTINGS_BACKEND (g_delayed_settings_backend_get_type ())
|
||||
#define G_DELAYED_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_CAST ((inst), \
|
||||
G_TYPE_DELAYED_SETTINGS_BACKEND, \
|
||||
GDelayedSettingsBackend))
|
||||
#define G_DELAYED_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_CAST ((class), \
|
||||
G_TYPE_DELAYED_SETTINGS_BACKEND, \
|
||||
GDelayedSettingsBackendClass))
|
||||
#define G_IS_DELAYED_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_TYPE ((inst), \
|
||||
G_TYPE_DELAYED_SETTINGS_BACKEND))
|
||||
#define G_IS_DELAYED_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_TYPE ((class), \
|
||||
G_TYPE_DELAYED_SETTINGS_BACKEND))
|
||||
#define G_DELAYED_SETTINGS_BACKEND_GET_CLASS(inst) (G_TYPE_INSTANCE_GET_CLASS ((inst), \
|
||||
G_TYPE_DELAYED_SETTINGS_BACKEND, \
|
||||
GDelayedSettingsBackendClass))
|
||||
|
||||
typedef struct _GDelayedSettingsBackendPrivate GDelayedSettingsBackendPrivate;
|
||||
typedef struct _GDelayedSettingsBackendClass GDelayedSettingsBackendClass;
|
||||
typedef struct _GDelayedSettingsBackend GDelayedSettingsBackend;
|
||||
|
||||
struct _GDelayedSettingsBackendClass
|
||||
{
|
||||
GSettingsBackendClass parent_class;
|
||||
};
|
||||
|
||||
struct _GDelayedSettingsBackend
|
||||
{
|
||||
GSettingsBackend parent_instance;
|
||||
GDelayedSettingsBackendPrivate *priv;
|
||||
};
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
GType g_delayed_settings_backend_get_type (void);
|
||||
GSettingsBackend * g_delayed_settings_backend_new (GSettingsBackend *backend,
|
||||
const gchar *base_path);
|
||||
void g_delayed_settings_backend_revert (GDelayedSettingsBackend *delayed);
|
||||
void g_delayed_settings_backend_apply (GDelayedSettingsBackend *delayed);
|
||||
gboolean g_delayed_settings_backend_get_has_unapplied (GDelayedSettingsBackend *delayed);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* _gsettingsdelayedbackend_h_ */
|
@@ -4,3 +4,5 @@ VOID:BOOLEAN,POINTER
|
||||
VOID:OBJECT,OBJECT,ENUM
|
||||
BOOLEAN:OBJECT,OBJECT
|
||||
VOID:STRING,BOXED,BOXED
|
||||
VOID:STRING,BOXED,INT,POINTER
|
||||
VOID:POINTER,INT
|
||||
|
98
gio/gmemorysettingsbackend.c
Normal file
98
gio/gmemorysettingsbackend.c
Normal file
@@ -0,0 +1,98 @@
|
||||
#include "gmemorysettingsbackend.h"
|
||||
|
||||
G_DEFINE_TYPE_WITH_CODE (GMemorySettingsBackend,
|
||||
g_memory_settings_backend,
|
||||
G_TYPE_SETTINGS_BACKEND,
|
||||
g_io_extension_point_implement (G_SETTINGS_BACKEND_EXTENSION_POINT_NAME,
|
||||
g_define_type_id, "memory", 0))
|
||||
|
||||
struct _GMemorySettingsBackendPrivate
|
||||
{
|
||||
GHashTable *table;
|
||||
};
|
||||
|
||||
static GVariant *
|
||||
g_memory_settings_backend_read (GSettingsBackend *backend,
|
||||
const gchar *key,
|
||||
const GVariantType *expected_type)
|
||||
{
|
||||
GVariant *value;
|
||||
|
||||
GMemorySettingsBackend *memory = G_MEMORY_SETTINGS_BACKEND (backend);
|
||||
|
||||
value = g_hash_table_lookup (memory->priv->table, key);
|
||||
|
||||
if (value != NULL)
|
||||
g_variant_ref (value);
|
||||
|
||||
return value;
|
||||
}
|
||||
|
||||
static gboolean
|
||||
g_memory_settings_backend_write_one (gpointer key,
|
||||
gpointer value,
|
||||
gpointer data)
|
||||
{
|
||||
GMemorySettingsBackend *memory = ((gpointer *) data)[0];
|
||||
const gchar *prefix = ((gpointer *) data)[1];
|
||||
|
||||
g_hash_table_insert (memory->priv->table,
|
||||
g_strjoin ("", prefix, key, NULL),
|
||||
g_variant_ref (value));
|
||||
|
||||
return FALSE;
|
||||
}
|
||||
|
||||
static void
|
||||
g_memory_settings_backend_write (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *tree,
|
||||
gpointer origin_tag)
|
||||
{
|
||||
gpointer write_info[] = { backend, (gpointer) prefix };
|
||||
g_tree_foreach (tree, g_memory_settings_backend_write_one, write_info);
|
||||
g_settings_backend_changed_tree (backend, prefix, tree, origin_tag);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
g_memory_settings_backend_get_writable (GSettingsBackend *backend,
|
||||
const gchar *name)
|
||||
{
|
||||
return TRUE;
|
||||
}
|
||||
|
||||
static void
|
||||
g_memory_settings_backend_finalize (GObject *object)
|
||||
{
|
||||
GMemorySettingsBackend *memory = G_MEMORY_SETTINGS_BACKEND (object);
|
||||
|
||||
g_hash_table_unref (memory->priv->table);
|
||||
|
||||
G_OBJECT_CLASS (g_memory_settings_backend_parent_class)
|
||||
->finalize (object);
|
||||
}
|
||||
|
||||
static void
|
||||
g_memory_settings_backend_init (GMemorySettingsBackend *memory)
|
||||
{
|
||||
memory->priv = G_TYPE_INSTANCE_GET_PRIVATE (memory,
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND,
|
||||
GMemorySettingsBackendPrivate);
|
||||
memory->priv->table =
|
||||
g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
|
||||
(GDestroyNotify) g_variant_unref);
|
||||
}
|
||||
|
||||
static void
|
||||
g_memory_settings_backend_class_init (GMemorySettingsBackendClass *class)
|
||||
{
|
||||
GSettingsBackendClass *backend_class = G_SETTINGS_BACKEND_CLASS (class);
|
||||
GObjectClass *object_class = G_OBJECT_CLASS (class);
|
||||
|
||||
backend_class->read = g_memory_settings_backend_read;
|
||||
backend_class->write = g_memory_settings_backend_write;
|
||||
backend_class->get_writable = g_memory_settings_backend_get_writable;
|
||||
object_class->finalize = g_memory_settings_backend_finalize;
|
||||
|
||||
g_type_class_add_private (class, sizeof (GMemorySettingsBackendPrivate));
|
||||
}
|
51
gio/gmemorysettingsbackend.h
Normal file
51
gio/gmemorysettingsbackend.h
Normal file
@@ -0,0 +1,51 @@
|
||||
#ifndef _gmemorysettingsbackend_h_
|
||||
#define _gmemorysettingsbackend_h_
|
||||
|
||||
#include <gio/gsettingsbackend.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
#define G_TYPE_MEMORY_SETTINGS_BACKEND (g_memory_settings_backend_get_type ())
|
||||
#define G_MEMORY_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_CAST ((inst), \
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND, \
|
||||
GMemorySettingsBackend))
|
||||
#define G_MEMORY_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_CAST ((class), \
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND, \
|
||||
GMemorySettingsBackendClass))
|
||||
#define G_IS_MEMORY_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_TYPE ((inst), \
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND))
|
||||
#define G_IS_MEMORY_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_TYPE ((class), \
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND))
|
||||
#define G_MEMORY_SETTINGS_BACKEND_GET_CLASS(inst) (G_TYPE_INSTANCE_GET_CLASS ((inst), \
|
||||
G_TYPE_MEMORY_SETTINGS_BACKEND, \
|
||||
GMemorySettingsBackendClass))
|
||||
|
||||
#define G_MEMORY_SETTINGS_BACKEND_EXTENSION_POINT_NAME "gsettings-backend"
|
||||
|
||||
/**
|
||||
* GMemorySettingsBackend:
|
||||
*
|
||||
* A backend to GSettings that stores the settings in memory.
|
||||
**/
|
||||
typedef struct _GMemorySettingsBackendPrivate GMemorySettingsBackendPrivate;
|
||||
typedef struct _GMemorySettingsBackendClass GMemorySettingsBackendClass;
|
||||
typedef struct _GMemorySettingsBackend GMemorySettingsBackend;
|
||||
|
||||
struct _GMemorySettingsBackendClass
|
||||
{
|
||||
GSettingsBackendClass parent_class;
|
||||
};
|
||||
|
||||
struct _GMemorySettingsBackend
|
||||
{
|
||||
GSettingsBackend parent_instance;
|
||||
|
||||
/*< private >*/
|
||||
GMemorySettingsBackendPrivate *priv;
|
||||
};
|
||||
|
||||
GType g_memory_settings_backend_get_type (void);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* _gmemorysettingsbackend_h_ */
|
332
gio/gsettingsbackend.c
Normal file
332
gio/gsettingsbackend.c
Normal file
@@ -0,0 +1,332 @@
|
||||
/*
|
||||
* Copyright © 2009 Codethink Limited
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of version 3 of the GNU General Public License as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* See the included COPYING file for more information.
|
||||
*/
|
||||
|
||||
#include "gsettingsbackend.h"
|
||||
|
||||
#include "giomodule-priv.h"
|
||||
#include "gio-marshal.h"
|
||||
|
||||
#include <string.h>
|
||||
#include <stdlib.h>
|
||||
#include <glib.h>
|
||||
|
||||
#include "gioalias.h"
|
||||
|
||||
G_DEFINE_ABSTRACT_TYPE (GSettingsBackend, g_settings_backend, G_TYPE_OBJECT)
|
||||
|
||||
static guint changed_signal;
|
||||
|
||||
/**
|
||||
* SECTION:gsettingsbackend
|
||||
* @short_description: an interface for settings backend implementations
|
||||
*
|
||||
* The #GSettingsBackend inteface defines a generic interface for
|
||||
* non-strictly-typed data backend in a hierarchy.
|
||||
*
|
||||
* The interface defines methods for reading and writing values, a
|
||||
* method for determining if writing of certain values will fail
|
||||
* (lockdown) and a change notification signal.
|
||||
*
|
||||
* The semantics of the interface are very precisely defined and
|
||||
* implementations must carefully adhere to the expectations of
|
||||
* callers that are documented on each of the interface methods.
|
||||
**/
|
||||
|
||||
/**
|
||||
* g_settings_backend_changed:
|
||||
* @backend: a #GSettingsBackend implementation
|
||||
* @prefix: the longest common prefix
|
||||
* @items: the NULL-terminated list of changed keys
|
||||
* @n_items: the number of items in @items
|
||||
* @origin_tag: the origin tag
|
||||
*
|
||||
* Emits the changed signal on @backend. This function should only be
|
||||
* called by the implementation itself, to indicate that a change has
|
||||
* occurred.
|
||||
*
|
||||
* The list of changed keys, relative to the root of the settings store,
|
||||
* is @prefix prepended to each of the @items. It must either be the
|
||||
* case that @prefix is equal to "" or ends in "/" or that @items
|
||||
* contains exactly one item: "". @prefix need not be the largest
|
||||
* possible prefix.
|
||||
*
|
||||
* The implementation must call this function during any call to
|
||||
* g_settings_backend_write(), before the call returns (except in the
|
||||
* case that no keys are actually changed). It may not rely on the
|
||||
* existence of a mainloop for dispatching the signal later.
|
||||
*
|
||||
* The implementation may call this function at any other time it likes
|
||||
* in response to other events (such as changes occuring outside of the
|
||||
* program). These calls may originate from a mainloop or may originate
|
||||
* in response to any other action (including from calls to
|
||||
* g_settings_backend_write()).
|
||||
*
|
||||
* In the case that this call is in response to a call to
|
||||
* g_settings_backend_write() then @origin_tag must be set to the same
|
||||
* value that was passed to that call.
|
||||
**/
|
||||
void
|
||||
g_settings_backend_changed (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
gchar const * const *items,
|
||||
gint n_items,
|
||||
gpointer origin_tag)
|
||||
{
|
||||
if (n_items == -1)
|
||||
for (n_items = 0; items[n_items]; n_items++);
|
||||
|
||||
g_assert (items[n_items] == NULL);
|
||||
|
||||
g_signal_emit (backend, changed_signal, 0,
|
||||
prefix, items, n_items, origin_tag);
|
||||
}
|
||||
|
||||
static gboolean
|
||||
g_settings_backend_append_to_list (gpointer key,
|
||||
gpointer value,
|
||||
gpointer user_data)
|
||||
{
|
||||
return (*((*((gchar ***) user_data))++) = key, FALSE);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_changed_tree:
|
||||
* @backend: a #GSettingsBackend implementation
|
||||
* @prefix: the longest common prefix
|
||||
* @tree: a #GTree
|
||||
* @origin_tag: the origin tag
|
||||
*
|
||||
* This call is a convenience wrapper around
|
||||
* g_settings_backend_changed(). It gets the list of changes from
|
||||
* @tree.
|
||||
**/
|
||||
void
|
||||
g_settings_backend_changed_tree (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *tree,
|
||||
gpointer origin_tag)
|
||||
{
|
||||
gchar **list;
|
||||
|
||||
list = g_new (gchar *, g_tree_nnodes (tree) + 1);
|
||||
|
||||
{
|
||||
gchar **ptr = list;
|
||||
g_tree_foreach (tree, g_settings_backend_append_to_list, &ptr);
|
||||
*ptr = NULL;
|
||||
|
||||
g_assert (list + g_tree_nnodes (tree) == ptr);
|
||||
}
|
||||
|
||||
g_signal_emit (backend, changed_signal, 0,
|
||||
prefix, list, g_tree_nnodes (tree), origin_tag);
|
||||
g_free (list);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_read:
|
||||
* @backend: a #GSettingsBackend implementation
|
||||
* @key: the key to read
|
||||
* @expected_type: a #GVariantType hint
|
||||
* @returns: the values that was read, or %NULL
|
||||
*
|
||||
* Reads a keys. This call will never block.
|
||||
*
|
||||
* If the key exists, it will be returned. If the key does not exist,
|
||||
* %NULL will be returned.
|
||||
*
|
||||
* If @expected_type is given, it serves as a type hint to the backend.
|
||||
* If you expect a key of a certain type then you should give
|
||||
* @expected_type to increase your chances of getting it. Some backends
|
||||
* may ignore this argument and return values of a different type; it is
|
||||
* mostly used by backends that don't store strong type information.
|
||||
**/
|
||||
GVariant *
|
||||
g_settings_backend_read (GSettingsBackend *backend,
|
||||
const gchar *key,
|
||||
const GVariantType *expected_type)
|
||||
{
|
||||
return G_SETTINGS_BACKEND_GET_CLASS (backend)
|
||||
->read (backend, key, expected_type);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_write:
|
||||
* @backend: a #GSettingsBackend implementation
|
||||
* @prefix: the longest common prefix
|
||||
* @values: a #GTree containing values to write
|
||||
* @origin_tag: the origin tag
|
||||
*
|
||||
* Writes one or more keys. This call will never block.
|
||||
*
|
||||
* For each item in @values, a key is written. The key to be written is
|
||||
* @prefix prepended to the key used in the tree. The value stored in
|
||||
* the tree is expected to be a #GVariant instance. It must either be
|
||||
* the case that @prefix is equal to "" or ends in "/" or that @values
|
||||
* contains exactly one item, with a key of "". @prefix need not be the
|
||||
* largest possible prefix.
|
||||
*
|
||||
* This call does not fail. During this call a "changed" signal will be
|
||||
* emitted if any keys have been changed. The new values of all updated
|
||||
* keys will be visible to any signal callbacks.
|
||||
*
|
||||
* One possible method that an implementation might deal with failures
|
||||
* is to emit a second "backend-changed" signal (either during this
|
||||
* call, or later) to indicate that the affected keys have suddenly
|
||||
* "changed back" to their old values.
|
||||
**/
|
||||
void
|
||||
g_settings_backend_write (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *values,
|
||||
gpointer origin_tag)
|
||||
{
|
||||
G_SETTINGS_BACKEND_GET_CLASS (backend)
|
||||
->write (backend, prefix, values, origin_tag);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_get_writable:
|
||||
* @backend: a #GSettingsBackend implementation
|
||||
* @name: the name of a key, relative to the root of the backend
|
||||
* @returns: %TRUE if the key is writable
|
||||
*
|
||||
* Ensures that a key is available for writing to. This is the
|
||||
* interface through which 'lockdown' is implemented. Locked down
|
||||
* keys will have %FALSE returned by this call.
|
||||
*
|
||||
* You should not write to locked-down keys, but if you do, the
|
||||
* implementation will deal with it.
|
||||
**/
|
||||
gboolean
|
||||
g_settings_backend_get_writable (GSettingsBackend *backend,
|
||||
const gchar *name)
|
||||
{
|
||||
return G_SETTINGS_BACKEND_GET_CLASS (backend)
|
||||
->get_writable (backend, name);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_subscribe:
|
||||
* @backend: a #GSettingsBackend
|
||||
* @name: a key or path to subscribe to
|
||||
*
|
||||
* Reverses the effect of a previous call to
|
||||
* g_settings_backend_subscribe().
|
||||
**/
|
||||
void
|
||||
g_settings_backend_unsubscribe (GSettingsBackend *backend,
|
||||
const char *name)
|
||||
{
|
||||
G_SETTINGS_BACKEND_GET_CLASS (backend)
|
||||
->unsubscribe (backend, name);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_subscribe:
|
||||
* @backend: a #GSettingsBackend
|
||||
* @name: a key or path to subscribe to
|
||||
*
|
||||
* Requests that change signals be emitted for events on @name.
|
||||
**/
|
||||
void
|
||||
g_settings_backend_subscribe (GSettingsBackend *backend,
|
||||
const gchar *name)
|
||||
{
|
||||
G_SETTINGS_BACKEND_GET_CLASS (backend)
|
||||
->subscribe (backend, name);
|
||||
}
|
||||
|
||||
static void
|
||||
g_settings_backend_class_init (GSettingsBackendClass *class)
|
||||
{
|
||||
changed_signal =
|
||||
g_signal_new ("changed", G_TYPE_SETTINGS_BACKEND,
|
||||
G_SIGNAL_RUN_LAST,
|
||||
G_STRUCT_OFFSET (GSettingsBackendClass, changed),
|
||||
NULL, NULL,
|
||||
_gio_marshal_VOID__STRING_BOXED_INT_POINTER, G_TYPE_NONE,
|
||||
4, G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE,
|
||||
G_TYPE_STRV | G_SIGNAL_TYPE_STATIC_SCOPE,
|
||||
G_TYPE_INT, G_TYPE_POINTER);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_create_tree:
|
||||
* @returns: a new #GTree
|
||||
*
|
||||
* This is a convenience function for creating a tree that is compatible
|
||||
* with g_settings_backend_write(). It merely calls g_tree_new_full()
|
||||
* with strcmp() g_free() and g_variant_unref().
|
||||
**/
|
||||
GTree *
|
||||
g_settings_backend_create_tree (void)
|
||||
{
|
||||
return g_tree_new_full ((GCompareDataFunc) strcmp, NULL,
|
||||
g_free, (GDestroyNotify) g_variant_unref);
|
||||
}
|
||||
|
||||
static gpointer
|
||||
get_default_backend (gpointer user_data)
|
||||
{
|
||||
GIOExtension *extension = NULL;
|
||||
GIOExtensionPoint *point;
|
||||
GList *extensions;
|
||||
const gchar *env;
|
||||
GType type;
|
||||
|
||||
_g_io_modules_ensure_loaded ();
|
||||
|
||||
point = g_io_extension_point_lookup (G_SETTINGS_BACKEND_EXTENSION_POINT_NAME);
|
||||
|
||||
if ((env = getenv ("GSETTINGS_BACKEND")))
|
||||
{
|
||||
extension = g_io_extension_point_get_extension_by_name (point, env);
|
||||
|
||||
if (extension == NULL)
|
||||
g_warning ("Can't find GSettings backend '%s' given in "
|
||||
"GSETTINGS_BACKEND environment variable", env);
|
||||
}
|
||||
|
||||
if (extension == NULL)
|
||||
{
|
||||
extensions = g_io_extension_point_get_extensions (point);
|
||||
|
||||
if (extensions == NULL)
|
||||
g_error ("No GSettingsBackend implementations exist.");
|
||||
|
||||
extension = extensions->data;
|
||||
}
|
||||
|
||||
type = g_io_extension_get_type (extension);
|
||||
|
||||
return g_object_new (type, NULL);
|
||||
}
|
||||
|
||||
/**
|
||||
* g_settings_backend_get_default:
|
||||
* @returns: the default #GSettingsBackend
|
||||
*
|
||||
* Returns the default #GSettingsBackend.
|
||||
*
|
||||
* The user does not own the return value and it must not be freed.
|
||||
**/
|
||||
GSettingsBackend *
|
||||
g_settings_backend_get_default (void)
|
||||
{
|
||||
static GOnce once = G_ONCE_INIT;
|
||||
|
||||
return g_once (&once, get_default_backend, NULL);
|
||||
}
|
||||
|
||||
static void
|
||||
g_settings_backend_init (GSettingsBackend *backend)
|
||||
{
|
||||
}
|
108
gio/gsettingsbackend.h
Normal file
108
gio/gsettingsbackend.h
Normal file
@@ -0,0 +1,108 @@
|
||||
/*
|
||||
* Copyright © 2009 Codethink Limited
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of version 3 of the GNU General Public License as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* See the included COPYING file for more information.
|
||||
*/
|
||||
|
||||
#ifndef _gsettingsbackend_h_
|
||||
#define _gsettingsbackend_h_
|
||||
|
||||
#include <glib-object.h>
|
||||
|
||||
G_BEGIN_DECLS
|
||||
|
||||
#define G_TYPE_SETTINGS_BACKEND (g_settings_backend_get_type ())
|
||||
#define G_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_CAST ((inst), \
|
||||
G_TYPE_SETTINGS_BACKEND, GSettingsBackend))
|
||||
#define G_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_CAST ((class), \
|
||||
G_TYPE_SETTINGS_BACKEND, GSettingsBackendClass))
|
||||
#define G_IS_SETTINGS_BACKEND(inst) (G_TYPE_CHECK_INSTANCE_TYPE ((inst), \
|
||||
G_TYPE_SETTINGS_BACKEND))
|
||||
#define G_IS_SETTINGS_BACKEND_CLASS(class) (G_TYPE_CHECK_CLASS_TYPE ((class), \
|
||||
G_TYPE_SETTINGS_BACKEND))
|
||||
#define G_SETTINGS_BACKEND_GET_CLASS(inst) (G_TYPE_INSTANCE_GET_CLASS ((inst), \
|
||||
G_TYPE_SETTINGS_BACKEND, GSettingsBackendClass))
|
||||
|
||||
#define G_SETTINGS_BACKEND_EXTENSION_POINT_NAME "gsettings-backend"
|
||||
|
||||
/**
|
||||
* GSettingsBackend:
|
||||
*
|
||||
* An implementation of a settings storage repository.
|
||||
**/
|
||||
typedef struct _GSettingsBackendPrivate GSettingsBackendPrivate;
|
||||
typedef struct _GSettingsBackendClass GSettingsBackendClass;
|
||||
typedef struct _GSettingsBackend GSettingsBackend;
|
||||
|
||||
struct _GSettingsBackendClass
|
||||
{
|
||||
GObjectClass parent_class;
|
||||
|
||||
void (*changed) (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
gchar const * const *names,
|
||||
gint names_len,
|
||||
gpointer origin_tag);
|
||||
|
||||
GVariant * (*read) (GSettingsBackend *backend,
|
||||
const gchar *key,
|
||||
const GVariantType *expected_type);
|
||||
void (*write) (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *tree,
|
||||
gpointer origin_tag);
|
||||
gboolean (*get_writable) (GSettingsBackend *backend,
|
||||
const gchar *name);
|
||||
void (*subscribe) (GSettingsBackend *backend,
|
||||
const gchar *name);
|
||||
void (*unsubscribe) (GSettingsBackend *backend,
|
||||
const gchar *name);
|
||||
};
|
||||
|
||||
struct _GSettingsBackend
|
||||
{
|
||||
GObject parent_instance;
|
||||
|
||||
/*< private >*/
|
||||
GSettingsBackendPrivate *priv;
|
||||
};
|
||||
|
||||
GType g_settings_backend_get_type (void);
|
||||
GSettingsBackend * g_settings_backend_get_default (void);
|
||||
void g_settings_backend_set_default (GSettingsBackend *backend);
|
||||
GTree * g_settings_backend_create_tree (void);
|
||||
|
||||
GVariant * g_settings_backend_read (GSettingsBackend *backend,
|
||||
const gchar *key,
|
||||
const GVariantType *expected_type);
|
||||
|
||||
void g_settings_backend_write (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *values,
|
||||
gpointer origin_tag);
|
||||
|
||||
gboolean g_settings_backend_get_writable (GSettingsBackend *backend,
|
||||
const char *name);
|
||||
|
||||
void g_settings_backend_unsubscribe (GSettingsBackend *backend,
|
||||
const char *name);
|
||||
void g_settings_backend_subscribe (GSettingsBackend *backend,
|
||||
const char *name);
|
||||
|
||||
void g_settings_backend_changed (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
gchar const * const *items,
|
||||
gint n_items,
|
||||
gpointer origin_tag);
|
||||
void g_settings_backend_changed_tree (GSettingsBackend *backend,
|
||||
const gchar *prefix,
|
||||
GTree *tree,
|
||||
gpointer origin_tag);
|
||||
|
||||
G_END_DECLS
|
||||
|
||||
#endif /* _gsettingsbackend_h_ */
|
Reference in New Issue
Block a user