2007-05-07 19:09:24 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2007 The GNOME Foundation
|
|
|
|
* Written by Thomas Wood <thos@gnome.org>
|
|
|
|
* All Rights Reserved
|
|
|
|
*
|
|
|
|
* 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.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
2007-05-09 19:50:39 +00:00
|
|
|
#include "appearance.h"
|
|
|
|
|
2007-05-07 19:09:24 +00:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "gnome-theme-info.h"
|
2007-05-07 22:00:05 +00:00
|
|
|
#include "gconf-property-editor.h"
|
2007-05-07 19:09:24 +00:00
|
|
|
|
|
|
|
enum ThemeType {
|
|
|
|
GTK_THEMES,
|
|
|
|
METACITY_THEMES,
|
|
|
|
ICON_THEMES,
|
2007-05-13 18:23:12 +00:00
|
|
|
CURSOR_THEMES,
|
|
|
|
COLOR_SCHEME
|
2007-05-07 19:09:24 +00:00
|
|
|
};
|
|
|
|
|
2007-05-09 19:50:39 +00:00
|
|
|
static const gchar *gconf_keys[] = {
|
2007-05-07 19:09:24 +00:00
|
|
|
"/desktop/gnome/interface/gtk_theme",
|
|
|
|
"/apps/metacity/general/theme",
|
|
|
|
"/desktop/gnome/interface/icon_theme",
|
2007-05-13 18:23:12 +00:00
|
|
|
"/desktop/gnome/peripherals/mouse/cursor_theme",
|
|
|
|
"/desktop/gnome/interface/gtk_color_scheme"
|
2007-05-07 19:09:24 +00:00
|
|
|
};
|
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
|
|
|
|
static void prepare_combo (AppearanceData *data, GtkWidget *combo, enum ThemeType type);
|
|
|
|
static void update_color_buttons_from_string (gchar *color_scheme, AppearanceData *data);
|
|
|
|
|
|
|
|
|
|
|
|
/* GUI Callbacks */
|
|
|
|
|
|
|
|
static void color_button_clicked_cb (GtkWidget *colorbutton, AppearanceData *data);
|
|
|
|
static GConfValue *conv_to_widget_cb (GConfPropertyEditor *peditor, GConfValue *value);
|
|
|
|
static GConfValue *conv_from_widget_cb (GConfPropertyEditor *peditor, GConfValue *value);
|
|
|
|
void
|
|
|
|
style_init (AppearanceData *data)
|
|
|
|
{
|
|
|
|
|
|
|
|
GtkSettings *settings;
|
|
|
|
gchar *colour_scheme;
|
|
|
|
|
|
|
|
prepare_combo (data, glade_xml_get_widget (data->xml, "gtk_themes_combobox"), GTK_THEMES);
|
|
|
|
prepare_combo (data, glade_xml_get_widget (data->xml, "window_themes_combobox"), METACITY_THEMES);
|
|
|
|
prepare_combo (data, glade_xml_get_widget (data->xml, "icon_themes_combobox"), ICON_THEMES);
|
|
|
|
|
|
|
|
settings = gtk_settings_get_default ();
|
|
|
|
g_object_get (G_OBJECT (settings), "gtk-color-scheme", &colour_scheme, NULL);
|
|
|
|
update_color_buttons_from_string (colour_scheme, data);
|
|
|
|
g_free (colour_scheme);
|
|
|
|
|
|
|
|
/* connect signals */
|
|
|
|
/* color buttons */
|
|
|
|
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "bg_colorbutton")), "color-set", (GCallback) color_button_clicked_cb, data);
|
|
|
|
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "base_colorbutton")), "color-set", (GCallback) color_button_clicked_cb, data);
|
|
|
|
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "selected_bg_colorbutton")), "color-set", (GCallback) color_button_clicked_cb, data);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
prepare_combo (AppearanceData *data, GtkWidget *combo, enum ThemeType type)
|
|
|
|
{
|
|
|
|
GtkListStore *store;
|
|
|
|
GList *l, *list = NULL;
|
|
|
|
GtkCellRenderer *renderer;
|
|
|
|
GnomeThemeElement element = 0;
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case GTK_THEMES:
|
|
|
|
element = GNOME_THEME_GTK_2;
|
|
|
|
case METACITY_THEMES:
|
|
|
|
if (!element) element = GNOME_THEME_METACITY;
|
|
|
|
list = gnome_theme_info_find_by_type (element);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ICON_THEMES:
|
|
|
|
list = gnome_theme_icon_info_find_all ();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case CURSOR_THEMES:
|
|
|
|
list = NULL; /* don't know what to do yet */
|
|
|
|
|
|
|
|
default:
|
|
|
|
/* we don't deal with any other type of themes here */
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!list)
|
|
|
|
return;
|
|
|
|
|
|
|
|
renderer = gtk_cell_renderer_text_new ();
|
|
|
|
store = gtk_list_store_new (1, G_TYPE_STRING);
|
|
|
|
|
|
|
|
for (l = list; l; l = g_list_next (l))
|
|
|
|
{
|
|
|
|
gchar *name = NULL;
|
|
|
|
GtkTreeIter i;
|
|
|
|
|
|
|
|
if (type < ICON_THEMES)
|
|
|
|
name = ((GnomeThemeInfo*) l->data)->name;
|
|
|
|
else if (type == ICON_THEMES)
|
|
|
|
name = ((GnomeThemeIconInfo*) l->data)->name;
|
|
|
|
|
|
|
|
if (!name)
|
|
|
|
continue; /* just in case... */
|
|
|
|
|
|
|
|
gtk_list_store_insert_with_values (store, &i, 0, 0, name, -1);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
gtk_combo_box_set_model (GTK_COMBO_BOX (combo), GTK_TREE_MODEL (store));
|
|
|
|
gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (combo), renderer, TRUE);
|
|
|
|
gtk_cell_layout_add_attribute (GTK_CELL_LAYOUT (combo), renderer, "text", 0);
|
|
|
|
|
|
|
|
gconf_peditor_new_combo_box (NULL, gconf_keys[type], combo,
|
|
|
|
"conv-to-widget-cb", conv_to_widget_cb,
|
|
|
|
"conv-from-widget-cb", conv_from_widget_cb,
|
|
|
|
NULL);
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Callbacks */
|
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
static GConfValue *
|
|
|
|
conv_to_widget_cb (GConfPropertyEditor *peditor, GConfValue *value)
|
|
|
|
{
|
|
|
|
GtkListStore *store;
|
|
|
|
GtkTreeIter iter;
|
|
|
|
gboolean valid;
|
|
|
|
gchar *test = NULL;
|
|
|
|
GtkWidget *combo;
|
|
|
|
GConfValue *new_value;
|
|
|
|
gint index = -1;
|
|
|
|
|
|
|
|
/* find value in model */
|
|
|
|
combo = GTK_WIDGET (gconf_property_editor_get_ui_control (peditor));
|
|
|
|
store = GTK_LIST_STORE (gtk_combo_box_get_model (GTK_COMBO_BOX (combo)));
|
|
|
|
valid = gtk_tree_model_get_iter_first (GTK_TREE_MODEL (store), &iter);
|
|
|
|
while (valid)
|
|
|
|
{
|
|
|
|
index++;
|
|
|
|
g_free (test);
|
|
|
|
gtk_tree_model_get (GTK_TREE_MODEL (store), &iter, 0, &test, -1);
|
|
|
|
if (test && !strcmp (test, gconf_value_get_string (value)))
|
|
|
|
break;
|
|
|
|
valid = gtk_tree_model_iter_next (GTK_TREE_MODEL (store), &iter);
|
|
|
|
}
|
|
|
|
g_free (test);
|
|
|
|
|
2007-05-07 22:29:37 +00:00
|
|
|
/* FIXME: Add a temporary item if we can't find a match */
|
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
new_value = gconf_value_new (GCONF_VALUE_INT);
|
|
|
|
gconf_value_set_int (new_value, index);
|
|
|
|
|
|
|
|
return new_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GConfValue *
|
|
|
|
conv_from_widget_cb (GConfPropertyEditor *peditor, GConfValue *value)
|
2007-05-07 19:09:24 +00:00
|
|
|
{
|
2007-05-07 22:00:05 +00:00
|
|
|
GConfValue *new_value;
|
|
|
|
gchar *combo_value = NULL;
|
2007-05-07 19:09:24 +00:00
|
|
|
GtkTreeIter iter;
|
|
|
|
GtkTreeModel *model;
|
2007-05-07 22:00:05 +00:00
|
|
|
GtkWidget *combo;
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
combo = GTK_WIDGET (gconf_property_editor_get_ui_control (peditor));
|
2007-05-07 19:09:24 +00:00
|
|
|
model = gtk_combo_box_get_model (GTK_COMBO_BOX (combo));
|
|
|
|
gtk_combo_box_get_active_iter (GTK_COMBO_BOX (combo), &iter);
|
2007-05-07 22:00:05 +00:00
|
|
|
gtk_tree_model_get (model, &iter, 0, &combo_value, -1);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
new_value = gconf_value_new (GCONF_VALUE_STRING);
|
|
|
|
gconf_value_set_string (new_value, combo_value);
|
|
|
|
g_free (combo_value);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
return new_value;
|
2007-05-07 19:09:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2007-05-13 18:23:12 +00:00
|
|
|
update_color_buttons_from_string (gchar *color_scheme, AppearanceData *data)
|
2007-05-07 19:09:24 +00:00
|
|
|
{
|
2007-05-13 18:23:12 +00:00
|
|
|
GdkColor color_scheme_colors[6];
|
|
|
|
gchar **color_scheme_strings, **color_scheme_pair, *current_string;
|
|
|
|
gint i;
|
|
|
|
GtkWidget *widget;
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
if (!color_scheme) return;
|
|
|
|
if (!strcmp (color_scheme, "")) return;
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
/* The color scheme string consists of name:color pairs, seperated by
|
|
|
|
* newlines, so first we split the string up by new line */
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
color_scheme_strings = g_strsplit (color_scheme, "\n", 0);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
/* loop through the name:color pairs, and save the colour if we recognise the name */
|
|
|
|
i = 0;
|
|
|
|
while ((current_string = color_scheme_strings[i++]))
|
2007-05-07 19:09:24 +00:00
|
|
|
{
|
2007-05-13 18:23:12 +00:00
|
|
|
color_scheme_pair = g_strsplit (current_string, ":", 0);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
if (color_scheme_pair[0] != NULL && color_scheme_pair[1] != NULL)
|
|
|
|
{
|
|
|
|
g_strstrip (color_scheme_pair[0]);
|
|
|
|
g_strstrip (color_scheme_pair[1]);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
if (!strcmp ("fg_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[0]);
|
|
|
|
else if (!strcmp ("bg_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[1]);
|
|
|
|
else if (!strcmp ("text_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[2]);
|
|
|
|
else if (!strcmp ("base_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[3]);
|
|
|
|
else if (!strcmp ("selected_fg_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[4]);
|
|
|
|
else if (!strcmp ("selected_bg_color", color_scheme_pair[0]))
|
|
|
|
gdk_color_parse (color_scheme_pair[1], &color_scheme_colors[5]);
|
|
|
|
}
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
g_strfreev (color_scheme_pair);
|
2007-05-07 19:09:24 +00:00
|
|
|
}
|
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
g_strfreev (color_scheme_strings);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
/* not sure whether we need to do this, but it can't hurt */
|
|
|
|
for (i = 0; i < 6; i++)
|
|
|
|
gdk_colormap_alloc_color (gdk_colormap_get_system (), &color_scheme_colors[i], FALSE, TRUE);
|
|
|
|
|
|
|
|
/* now set all the buttons to the correct settings */
|
|
|
|
widget = glade_xml_get_widget (data->xml, "bg_colorbutton");
|
|
|
|
gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[1]);
|
|
|
|
widget = glade_xml_get_widget (data->xml, "base_colorbutton");
|
|
|
|
gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[3]);
|
|
|
|
widget = glade_xml_get_widget (data->xml, "selected_bg_colorbutton");
|
|
|
|
gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[5]);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
|
|
|
|
2007-05-07 22:00:05 +00:00
|
|
|
}
|
2007-05-13 18:23:12 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
color_scheme_changed (GObject *settings,
|
|
|
|
GParamSpec *pspec,
|
|
|
|
AppearanceData *data)
|
2007-05-07 19:09:24 +00:00
|
|
|
{
|
2007-05-13 18:23:12 +00:00
|
|
|
gchar *theme;
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
theme = gconf_client_get_string (data->client, gconf_keys[COLOR_SCHEME], NULL);
|
|
|
|
if (theme == NULL || strcmp (theme, "") == 0)
|
|
|
|
g_object_get (settings, "gtk-color-scheme", &theme, NULL);
|
2007-05-07 19:09:24 +00:00
|
|
|
|
2007-05-13 18:23:12 +00:00
|
|
|
update_color_buttons_from_string (theme, data);
|
|
|
|
g_free (theme);
|
2007-05-07 19:09:24 +00:00
|
|
|
}
|
2007-05-13 18:23:12 +00:00
|
|
|
|
|
|
|
static void
|
|
|
|
color_button_clicked_cb (GtkWidget *colorbutton, AppearanceData *data)
|
|
|
|
{
|
|
|
|
gchar *new_scheme;
|
|
|
|
GdkColor colors[6];
|
|
|
|
gchar *bg, *fg, *text, *base, *selected_fg, *selected_bg;
|
|
|
|
GtkWidget *widget;
|
|
|
|
|
|
|
|
widget = glade_xml_get_widget (data->xml, "bg_colorbutton");
|
|
|
|
gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[1]);
|
|
|
|
widget = glade_xml_get_widget (data->xml, "base_colorbutton");
|
|
|
|
gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[3]);
|
|
|
|
widget = glade_xml_get_widget (data->xml, "selected_bg_colorbutton");
|
|
|
|
gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[5]);
|
|
|
|
|
|
|
|
/* TODO: calculate proper colours here */
|
|
|
|
gdk_color_parse ("black", &colors[0]);
|
|
|
|
gdk_color_parse ("black", &colors[2]);
|
|
|
|
gdk_color_parse ("black", &colors[4]);
|
|
|
|
|
|
|
|
fg = g_strdup_printf ("fg_color:#%04x%04x%04x\n", colors[0].red, colors[0].green, colors[0].blue);
|
|
|
|
bg = g_strdup_printf ("bg_color:#%04x%04x%04x\n", colors[1].red, colors[1].green, colors[1].blue);
|
|
|
|
text = g_strdup_printf ("text_color:#%04x%04x%04x\n", colors[2].red, colors[2].green, colors[2].blue);
|
|
|
|
base = g_strdup_printf ("base_color:#%04x%04x%04x\n", colors[3].red, colors[3].green, colors[3].blue);
|
|
|
|
selected_fg = g_strdup_printf ("selected_fg_color:#%04x%04x%04x\n", colors[4].red, colors[4].green, colors[4].blue);
|
|
|
|
selected_bg = g_strdup_printf ("selected_bg_color:#%04x%04x%04x", colors[5].red, colors[5].green, colors[5].blue);
|
|
|
|
|
|
|
|
new_scheme = g_strconcat (fg, bg, text, base, selected_fg, selected_bg, NULL);
|
|
|
|
|
|
|
|
/* Currently we assume this has only been called when one of the colours has
|
|
|
|
* actually changed, so we don't check the original key first
|
|
|
|
*/
|
|
|
|
gconf_client_set_string (data->client, gconf_keys[COLOR_SCHEME], new_scheme, NULL);
|
|
|
|
|
|
|
|
g_free (fg);
|
|
|
|
g_free (bg);
|
|
|
|
g_free (text);
|
|
|
|
g_free (base);
|
|
|
|
g_free (selected_fg);
|
|
|
|
g_free (selected_bg);
|
|
|
|
g_free (new_scheme);
|
|
|
|
}
|
|
|
|
|
|
|
|
|