gnome-control-center/gnome-settings-daemon/utils.c
Alexander Larsson d8426a9677 Also check registered directories for notification. Fixes theme e.g.
2007-04-10  Alexander Larsson <alexl@redhat.com>

	* utils.c (config_notify):
	Also check registered directories for notification.
	Fixes theme e.g. switching.


svn path=/trunk/; revision=7456
2007-04-10 11:50:49 +00:00

193 lines
4.7 KiB
C

/*
* Copyright (C) 2007 The GNOME Foundation
*
* Authors: Rodrigo Moya
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
* USA
*/
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include "utils.h"
static GHashTable *directories = NULL;
static void
child_watch_cb (GPid pid, gint status, gpointer user_data)
{
gchar *command = user_data;
if (!WIFEXITED (status) || WEXITSTATUS (status)) {
g_warning ("Command %s failed", command);
}
g_free (command);
}
/*
* Helper function for spawn_with_input() - write an entire
* string to a fd.
*/
static gboolean
write_all (int fd,
const char *buf,
gsize to_write)
{
while (to_write > 0) {
gssize count = write (fd, buf, to_write);
if (count < 0) {
if (errno != EINTR)
return FALSE;
} else {
to_write -= count;
buf += count;
}
}
return TRUE;
}
/**
* gnome_settings_spawn_with_input:
* @argv: command line to run
* @input: string to write to the child process.
*
* Spawns a child process specified by @argv, writes the text in
* @input to it, then waits for the child to exit. Any failures
* are output through g_warning(); if you wanted to use this in
* cases where errors need to be presented to the user, some
* modification would be needed.
**/
void
gnome_settings_spawn_with_input (char **argv, const char *input)
{
int child_pid;
int inpipe;
GError *err = NULL;
gchar *command;
if (!g_spawn_async_with_pipes (NULL /* working directory */, argv, NULL /* envp */,
G_SPAWN_SEARCH_PATH | G_SPAWN_DO_NOT_REAP_CHILD,
NULL, NULL, /* child setup and data */
&child_pid,
&inpipe, NULL, NULL, /* stdin, stdout, stderr */
&err)) {
command = g_strjoinv (" ", argv);
g_warning ("Could not execute %s: %s", command, err->message);
g_error_free (err);
g_free (command);
return;
}
command = g_strjoinv (" ", argv);
if (input) {
if (!write_all (inpipe, input, strlen (input))) {
g_warning ("Could not write input to %s", command);
}
close (inpipe);
}
g_child_watch_add (child_pid, (GChildWatchFunc) child_watch_cb, command);
}
extern GConfClient *conf_client;
GConfClient *
gnome_settings_get_config_client (void)
{
if (!conf_client)
conf_client = gconf_client_get_default ();
return conf_client;
}
static void
config_notify (GConfClient *client,
guint cnxn_id,
GConfEntry *entry,
gpointer user_data)
{
GSList *callbacks;
char *dir, *last_slash;
callbacks = g_hash_table_lookup (directories, entry->key);
if (callbacks) {
GSList *sl;
for (sl = callbacks; sl; sl = sl->next)
((GnomeSettingsConfigCallback) sl->data) (entry);
}
/* Also check if the directory, not the specific key was registered */
dir = g_strdup (entry->key);
last_slash = strrchr (dir, '/');
if (last_slash)
*last_slash = 0;
callbacks = g_hash_table_lookup (directories, dir);
g_free (dir);
if (callbacks) {
GSList *sl;
for (sl = callbacks; sl; sl = sl->next)
((GnomeSettingsConfigCallback) sl->data) (entry);
}
}
void
gnome_settings_register_config_callback (const char *dir, GnomeSettingsConfigCallback func)
{
GSList *callbacks;
if (!directories)
directories = g_hash_table_new (g_str_hash, g_str_equal);
callbacks = g_hash_table_lookup (directories, dir);
if (callbacks) {
GSList *sl;
for (sl = callbacks; sl; sl = sl->next) {
if (func == sl->data)
return;
}
/* callback not registered, so add it */
callbacks = g_slist_append (callbacks, func);
} else {
GError *error = NULL;
callbacks = g_slist_append (callbacks, func);
g_hash_table_insert (directories, g_strdup (dir), callbacks);
gconf_client_add_dir (gnome_settings_get_config_client (), dir,
GCONF_CLIENT_PRELOAD_ONELEVEL, NULL);
gconf_client_notify_add (gnome_settings_get_config_client (), dir,
config_notify, NULL, NULL, &error);
if (error) {
g_warning ("Could not listen for changes to configuration in '%s': %s\n",
dir, error->message);
g_error_free (error);
}
}
}