/* * Copyright (C) 2007 The GNOME Foundation * Written by Thomas Wood * 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. */ #include "appearance.h" #include #include #include "theme-util.h" #include "gtkrc-utils.h" #include "gconf-property-editor.h" #include "theme-thumbnail.h" static void prepare_list (AppearanceData *data, GtkWidget *list, ThemeType type); static void update_color_buttons_from_string (const gchar *color_scheme, AppearanceData *data); static void update_color_buttons_from_settings (GtkSettings *settings, AppearanceData *data); static void check_color_schemes_enabled (GtkSettings *settings, AppearanceData *data); static void color_scheme_changed (GObject *settings, GParamSpec *pspec, AppearanceData *data); static void theme_name_changed (GObject *settings, GParamSpec *pspec, AppearanceData *data); /* GUI Callbacks */ static void color_button_clicked_cb (GtkWidget *colorbutton, AppearanceData *data); static void color_scheme_defaults_button_clicked_cb (GtkWidget *button, AppearanceData *data); static GConfValue *conv_to_widget_cb (GConfPropertyEditor *peditor, const GConfValue *value); static GConfValue *conv_from_widget_cb (GConfPropertyEditor *peditor, const GConfValue *value); static void style_response_cb (GtkDialog *dialog, gint response_id) { if (response_id == GTK_RESPONSE_HELP) { /* FIXME: help */ } else { gtk_widget_hide (GTK_WIDGET (dialog)); } } void style_init (AppearanceData *data) { GObject *settings; GtkWidget *w; w = glade_xml_get_widget (data->xml, "theme_details"); g_signal_connect (w, "response", (GCallback) style_response_cb, NULL); g_signal_connect (w, "delete_event", (GCallback) gtk_true, NULL); prepare_list (data, glade_xml_get_widget (data->xml, "gtk_themes_list"), THEME_TYPE_GTK); prepare_list (data, glade_xml_get_widget (data->xml, "window_themes_list"), THEME_TYPE_WINDOW); prepare_list (data, glade_xml_get_widget (data->xml, "icon_themes_list"), THEME_TYPE_ICON); w = glade_xml_get_widget (data->xml, "color_scheme_message_hbox"); gtk_widget_set_no_show_all (w, TRUE); w = glade_xml_get_widget (data->xml, "color_scheme_defaults_button"); gtk_button_set_image (GTK_BUTTON (w), gtk_image_new_from_stock (GTK_STOCK_REVERT_TO_SAVED, GTK_ICON_SIZE_BUTTON)); settings = G_OBJECT (gtk_settings_get_default ()); g_signal_connect (settings, "notify::gtk-theme-name", (GCallback) theme_name_changed, data); check_color_schemes_enabled (GTK_SETTINGS (settings), data); g_signal_connect (settings, "notify::gtk-color-scheme", (GCallback) color_scheme_changed, data); update_color_buttons_from_settings (GTK_SETTINGS (settings), data); /* connect signals */ /* color buttons */ g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "fg_colorbutton")), "color-set", (GCallback) color_button_clicked_cb, data); 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, "text_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_fg_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); /* "Reset To Defaults" button */ g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "color_scheme_defaults_button")), "clicked", (GCallback) color_scheme_defaults_button_clicked_cb, data); } static void prepare_list (AppearanceData *data, GtkWidget *list, ThemeType type) { GtkListStore *store; GList *l, *themes = NULL; GtkCellRenderer *renderer; GtkTreeViewColumn *column; GtkTreeModel *sort_model; const gchar *key; switch (type) { case THEME_TYPE_GTK: themes = gnome_theme_info_find_by_type (GNOME_THEME_GTK_2); key = GTK_THEME_KEY; break; case THEME_TYPE_WINDOW: themes = gnome_theme_info_find_by_type (GNOME_THEME_METACITY); key = METACITY_THEME_KEY; break; case THEME_TYPE_ICON: themes = gnome_theme_icon_info_find_all (); key = ICON_THEME_KEY; break; case THEME_TYPE_CURSOR: themes = NULL; /* don't know what to do yet */ key = CURSOR_THEME_KEY; default: /* we don't deal with any other type of themes here */ return; } if (!list) return; store = gtk_list_store_new (NUM_COLS, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING); for (l = themes; l; l = g_list_next (l)) { const gchar *name = NULL; const gchar *label = NULL; GdkPixbuf *thumbnail; GtkTreeIter i; if (type == THEME_TYPE_GTK || type == THEME_TYPE_WINDOW) { name = ((GnomeThemeInfo *) l->data)->name; } else if (type == THEME_TYPE_ICON) { name = ((GnomeThemeIconInfo *) l->data)->name; label = ((GnomeThemeIconInfo *) l->data)->readable_name; } if (!name) continue; /* just in case... */ switch (type) { case THEME_TYPE_GTK: thumbnail = generate_gtk_theme_thumbnail ((GnomeThemeInfo *) l->data); break; case THEME_TYPE_ICON: thumbnail = generate_icon_theme_thumbnail ((GnomeThemeIconInfo *) l->data); break; case THEME_TYPE_WINDOW: thumbnail = generate_metacity_theme_thumbnail ((GnomeThemeInfo *) l->data); break; default: thumbnail = NULL; } gtk_list_store_insert_with_values (store, &i, 0, COL_THUMBNAIL, thumbnail, COL_LABEL, label ? label : name, COL_NAME, name, -1); if (thumbnail) g_object_unref (thumbnail); } g_list_free (themes); sort_model = gtk_tree_model_sort_new_with_model (GTK_TREE_MODEL (store)); gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (sort_model), COL_LABEL, GTK_SORT_ASCENDING); gtk_tree_view_set_model (GTK_TREE_VIEW (list), GTK_TREE_MODEL (sort_model)); renderer = gtk_cell_renderer_pixbuf_new (); g_object_set (G_OBJECT (renderer), "xpad", 3, "ypad", 3, NULL); column = gtk_tree_view_column_new (); gtk_tree_view_column_pack_start (column, renderer, FALSE); gtk_tree_view_column_add_attribute (column, renderer, "pixbuf", COL_THUMBNAIL); gtk_tree_view_append_column (GTK_TREE_VIEW (list), column); renderer = gtk_cell_renderer_text_new (); g_object_set (G_OBJECT (renderer), "weight", PANGO_WEIGHT_BOLD, "weight-set", TRUE, NULL); column = gtk_tree_view_column_new (); gtk_tree_view_column_pack_start (column, renderer, FALSE); gtk_tree_view_column_add_attribute (column, renderer, "text", COL_LABEL); gtk_tree_view_append_column (GTK_TREE_VIEW (list), column); gconf_peditor_new_tree_view (NULL, key, list, "conv-to-widget-cb", conv_to_widget_cb, "conv-from-widget-cb", conv_from_widget_cb, NULL); } /* Callbacks */ static gchar * find_string_in_model (GtkTreeModel *model, const gchar *value, gint column) { GtkTreeIter iter; gboolean valid; gchar *path = NULL, *test; if (!value) return NULL; for (valid = gtk_tree_model_get_iter_first (model, &iter); valid; valid = gtk_tree_model_iter_next (model, &iter)) { gtk_tree_model_get (model, &iter, column, &test, -1); if (test) { gint cmp = strcmp (test, value); g_free (test); if (!cmp) { path = gtk_tree_model_get_string_from_iter (model, &iter); break; } } } return path; } static GConfValue * conv_to_widget_cb (GConfPropertyEditor *peditor, const GConfValue *value) { GtkTreeModel *store; GtkTreeView *list; const gchar *curr_value; GConfValue *new_value; gchar *path; /* find value in model */ curr_value = gconf_value_get_string (value); list = GTK_TREE_VIEW (gconf_property_editor_get_ui_control (peditor)); store = gtk_tree_view_get_model (list); path = find_string_in_model (store, curr_value, COL_NAME); /* Add a temporary item if we can't find a match * TODO: delete this item if it is no longer selected? */ if (!path) { GtkListStore *list_store; GtkTreeIter iter, sort_iter; list_store = GTK_LIST_STORE (gtk_tree_model_sort_get_model (GTK_TREE_MODEL_SORT (store))); gtk_list_store_insert_with_values (list_store, &iter, 0, COL_LABEL, curr_value, COL_NAME, curr_value, -1); /* convert the tree store iter for use with the sort model */ gtk_tree_model_sort_convert_child_iter_to_iter (GTK_TREE_MODEL_SORT (store), &sort_iter, &iter); path = gtk_tree_model_get_string_from_iter (store, &sort_iter); } new_value = gconf_value_new (GCONF_VALUE_STRING); gconf_value_set_string (new_value, path); g_free (path); return new_value; } static GConfValue * conv_from_widget_cb (GConfPropertyEditor *peditor, const GConfValue *value) { GConfValue *new_value; gchar *list_value = NULL; GtkTreeIter iter; GtkTreeSelection *selection; GtkTreeModel *model; GtkTreeView *list; list = GTK_TREE_VIEW (gconf_property_editor_get_ui_control (peditor)); model = gtk_tree_view_get_model (list); selection = gtk_tree_view_get_selection (list); gtk_tree_selection_get_selected (selection, NULL, &iter); gtk_tree_model_get (model, &iter, COL_NAME, &list_value, -1); new_value = gconf_value_new (GCONF_VALUE_STRING); gconf_value_set_string (new_value, list_value); g_free (list_value); return new_value; } static void update_color_buttons_from_string (const gchar *color_scheme, AppearanceData *data) { GdkColor color_scheme_colors[6]; gchar **color_scheme_strings, **color_scheme_pair, *current_string; gint i; GtkWidget *widget; if (!color_scheme || !strcmp (color_scheme, "")) return; /* The color scheme string consists of name:color pairs, seperated by * newlines, so first we split the string up by new line */ color_scheme_strings = g_strsplit (color_scheme, "\n", 0); /* loop through the name:color pairs, and save the colour if we recognise the name */ i = 0; while ((current_string = color_scheme_strings[i++])) { color_scheme_pair = g_strsplit (current_string, ":", 0); if (color_scheme_pair[0] != NULL && color_scheme_pair[1] != NULL) { g_strstrip (color_scheme_pair[0]); g_strstrip (color_scheme_pair[1]); 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]); } g_strfreev (color_scheme_pair); } g_strfreev (color_scheme_strings); /* 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, "fg_colorbutton"); gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[0]); 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, "text_colorbutton"); gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[2]); 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_fg_colorbutton"); gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[4]); widget = glade_xml_get_widget (data->xml, "selected_bg_colorbutton"); gtk_color_button_set_color (GTK_COLOR_BUTTON (widget), &color_scheme_colors[5]); } static void update_color_buttons_from_settings (GtkSettings *settings, AppearanceData *data) { gchar *scheme; scheme = gconf_client_get_string (data->client, COLOR_SCHEME_KEY, NULL); if (scheme == NULL || strcmp (scheme, "") == 0) { gtk_widget_set_sensitive (glade_xml_get_widget (data->xml, "color_scheme_defaults_button"), FALSE); g_free (scheme); g_object_get (G_OBJECT (settings), "gtk-color-scheme", &scheme, NULL); } update_color_buttons_from_string (scheme, data); g_free (scheme); } static void color_scheme_changed (GObject *settings, GParamSpec *pspec, AppearanceData *data) { update_color_buttons_from_settings (GTK_SETTINGS (settings), data); } static void check_color_schemes_enabled (GtkSettings *settings, AppearanceData *data) { gchar *theme = NULL; gchar *filename; GSList *symbolic_colors = NULL; gboolean fg, bg, base, text, fg_s, bg_s, enable_colors; g_object_get (G_OBJECT (settings), "gtk-theme-name", &theme, NULL); filename = gtkrc_find_named (theme); g_free (theme); gtkrc_get_details (filename, NULL, &symbolic_colors); g_free (filename); fg = (g_slist_find_custom (symbolic_colors, "fg_color", g_str_equal) != NULL); bg = (g_slist_find_custom (symbolic_colors, "bg_color", g_str_equal) != NULL); base = (g_slist_find_custom (symbolic_colors, "base_color", g_str_equal) != NULL); text = (g_slist_find_custom (symbolic_colors, "text_color", g_str_equal) != NULL); fg_s = (g_slist_find_custom (symbolic_colors, "selected_fg_color", g_str_equal) != NULL); bg_s = (g_slist_find_custom (symbolic_colors, "selected_bg_color", g_str_equal) != NULL); g_slist_foreach (symbolic_colors, (GFunc) g_free, NULL); g_slist_free (symbolic_colors); enable_colors = (fg && bg && base && text && fg_s && bg_s); gtk_widget_set_sensitive (glade_xml_get_widget (data->xml, "color_scheme_table"), enable_colors); gtk_widget_set_sensitive (glade_xml_get_widget (data->xml, "color_scheme_defaults_button"), enable_colors); if (enable_colors) gtk_widget_hide (glade_xml_get_widget (data->xml, "color_scheme_message_hbox")); else gtk_widget_show (glade_xml_get_widget (data->xml, "color_scheme_message_hbox")); } static void theme_name_changed (GObject *settings, GParamSpec *pspec, AppearanceData *data) { gchar *current_theme, *new_theme; /* manually update GtkSettings to new gtk+ theme. */ g_object_get (settings, "gtk-theme-name", ¤t_theme, NULL); new_theme = gconf_client_get_string (data->client, GTK_THEME_KEY, NULL); if (strcmp (current_theme, new_theme) != 0) g_object_set (settings, "gtk-theme-name", new_theme, NULL); g_free (current_theme); g_free (new_theme); check_color_schemes_enabled (GTK_SETTINGS (settings), data); update_color_buttons_from_settings (GTK_SETTINGS (settings), data); } 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, "fg_colorbutton"); gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[0]); 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, "text_colorbutton"); gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[2]); 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_fg_colorbutton"); gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[4]); widget = glade_xml_get_widget (data->xml, "selected_bg_colorbutton"); gtk_color_button_get_color (GTK_COLOR_BUTTON (widget), &colors[5]); 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, COLOR_SCHEME_KEY, new_scheme, NULL); gtk_widget_set_sensitive (glade_xml_get_widget (data->xml, "color_scheme_defaults_button"), TRUE); g_free (fg); g_free (bg); g_free (text); g_free (base); g_free (selected_fg); g_free (selected_bg); g_free (new_scheme); } static void color_scheme_defaults_button_clicked_cb (GtkWidget *button, AppearanceData *data) { gconf_client_set_string (data->client, COLOR_SCHEME_KEY, "", NULL); gtk_widget_set_sensitive (glade_xml_get_widget (data->xml, "color_scheme_defaults_button"), FALSE); }