gnome-control-center/capplets/appearance/appearance-themes.c
Jens Granseuer f5fe47e66e add capplets/appearance/gnome-theme-package.xml to generated files
2007-06-05  Jens Granseuer  <jensgr@gmx.net>

	* configure.in: add capplets/appearance/gnome-theme-package.xml
	to generated files

	* capplets/appearance/Makefile.am:
	* capplets/appearance/gnome-theme-package.xml.in: move theme
	package mime type over from the theme-manager (closes bug #444336)

svn path=/trunk/; revision=7693
2007-06-05 18:32:05 +00:00

405 lines
12 KiB
C

/*
* 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.
*/
#include "appearance.h"
#include "theme-thumbnail.h"
#include "gnome-theme-apply.h"
#include <libwindow-settings/gnome-wm-manager.h>
#include <string.h>
#define GTK_THEME_KEY "/desktop/gnome/interface/gtk_theme"
#define METACITY_THEME_KEY "/apps/metacity/general/theme"
#define ICON_THEME_KEY "/desktop/gnome/interface/icon_theme"
#define COLOR_SCHEME_KEY "/desktop/gnome/interface/gtk_color_scheme"
#define CUSTOM_THEME_NAME "__custom__"
enum {
COL_LABEL,
COL_THUMBNAIL,
COL_NAME,
NUM_COLS
};
static gboolean theme_thumbnail_generate (AppearanceData *data);
static gchar *
get_default_string_from_key (GConfClient *client, const char *key)
{
GConfValue *value;
gchar *str = NULL;
value = gconf_client_get_default_from_schema (client, key, NULL);
if (value) {
if (value->type == GCONF_VALUE_STRING)
str = gconf_value_to_string (value);
gconf_value_free (value);
}
return str;
}
static gboolean
find_in_model (GtkTreeModel *model, const gchar *value, gint column, GtkTreeIter *hit)
{
GtkTreeIter iter;
gboolean valid;
gchar *test;
if (!value)
return FALSE;
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) {
if (hit)
*hit = iter;
return TRUE;
}
}
}
return FALSE;
}
/** Theme Callbacks **/
static void
theme_changed_func (gpointer uri, AppearanceData *data)
{
/* TODO: add/change/remove themes from the model as appropriate */
}
static void
theme_thumbnail_func (GdkPixbuf *pixbuf, AppearanceData *data)
{
GnomeThemeMetaInfo *info = data->theme_queue->data;
/* find item in model and update thumbnail */
if (pixbuf) {
GtkTreeIter iter;
GtkTreeModel *model = GTK_TREE_MODEL (data->theme_store);
if (find_in_model (model, info->name, COL_NAME, &iter))
gtk_list_store_set (data->theme_store, &iter, COL_THUMBNAIL, pixbuf, -1);
g_object_unref (pixbuf);
}
data->theme_queue = g_slist_remove (data->theme_queue, info);
if (data->theme_queue)
/* we can't call theme_thumbnail_generate directly since the thumbnail
* factory hasn't yet cleaned up at this point */
g_idle_add ((GSourceFunc) theme_thumbnail_generate, data);
}
/** GUI Callbacks **/
static void
theme_selection_changed_cb (GtkWidget *icon_view, AppearanceData *data)
{
GtkTreeModel *model;
GtkTreeIter iter;
GnomeThemeMetaInfo *theme = NULL;
gchar *name;
GList *selection;
selection = gtk_icon_view_get_selected_items (GTK_ICON_VIEW (icon_view));
model = gtk_icon_view_get_model (GTK_ICON_VIEW (icon_view));
gtk_tree_model_get_iter (model, &iter, selection->data);
gtk_tree_model_get (model, &iter, COL_NAME, &name, -1);
if (!strcmp (name, CUSTOM_THEME_NAME))
theme = data->theme_custom;
else
theme = gnome_theme_meta_info_find (name);
if (theme)
gnome_meta_theme_set (theme);
g_free (name);
g_list_foreach (selection, (GFunc) gtk_tree_path_free, NULL);
g_list_free (selection);
}
static void
theme_custom_cb (GtkWidget *button, AppearanceData *data)
{
GtkWidget *w, *parent, *icon_view;
w = glade_xml_get_widget (data->xml, "theme_details");
parent = glade_xml_get_widget (data->xml, "appearance_window");
icon_view = glade_xml_get_widget (data->xml, "themes_list");
/* select the "custom" metatheme */
theme_set_custom_from_selected (icon_view, data);
gtk_window_set_transient_for (GTK_WINDOW (w), GTK_WINDOW (parent));
gtk_widget_show_all (w);
}
static void
theme_install_cb (GtkWidget *button, AppearanceData *data)
{
/* TODO: Install a new theme */
}
static void
theme_delete_cb (GtkWidget *button, AppearanceData *data)
{
/* TODO: Delete the selected theme */
}
static void
theme_select_after_realize (GtkIconView *icon_view,
const gchar *theme)
{
GtkTreeIter iter;
GtkTreeModel *model = gtk_icon_view_get_model (icon_view);
if (find_in_model (model, theme, COL_NAME, &iter)) {
GtkTreePath *path = gtk_tree_model_get_path (model, &iter);
gtk_icon_view_select_path (icon_view, path);
gtk_icon_view_scroll_to_path (icon_view, path, FALSE, 0.5, 0.0);
gtk_tree_path_free (path);
}
}
static gboolean
theme_thumbnail_generate (AppearanceData *data)
{
generate_theme_thumbnail_async (data->theme_queue->data, (ThemeThumbnailFunc) theme_thumbnail_func, data, NULL);
return FALSE;
}
static void
theme_queue_for_thumbnail (GnomeThemeMetaInfo *theme, AppearanceData *data)
{
if (data->theme_queue == NULL)
g_idle_add ((GSourceFunc) theme_thumbnail_generate, data);
data->theme_queue = g_slist_append (data->theme_queue, theme);
}
static GnomeThemeMetaInfo *
theme_get_selected (GtkIconView *icon_view)
{
GnomeThemeMetaInfo *theme = NULL;
GList *selected = gtk_icon_view_get_selected_items (icon_view);
if (selected) {
theme = selected->data;
g_list_foreach (selected, gtk_tree_path_free, NULL);
g_list_free (selected);
}
return theme;
}
static void
theme_set_custom_from_selected (GtkIconView *icon_view, AppearanceData *data)
{
GnomeThemeMetaInfo *custom = data->theme_custom;
GnomeThemeMetaInfo *theme = theme_get_selected (icon_view);
if (theme == custom)
return;
/* if info is not NULL, we'll copy those theme settings over */
if (theme != NULL) {
g_free (custom->gtk_theme_name);
g_free (custom->icon_theme_name);
g_free (custom->metacity_theme_name);
g_free (custom->gtk_color_scheme);
custom->gtk_theme_name = NULL;
custom->icon_theme_name = NULL;
custom->metacity_theme_name = NULL;
custom->gtk_color_scheme = NULL;
if (info->gtk_theme_name)
custom->gtk_theme_name = g_strdup (info->gtk_theme_name);
else
custom->gtk_theme_name = get_default_string_from_key (data->client, GTK_THEME_KEY);
if (info->icon_theme_name)
custom->icon_theme_name = g_strdup (info->icon_theme_name);
else
custom->icon_theme_name = get_default_string_from_key (data->client, ICON_THEME_KEY);
if (info->metacity_theme_name)
custom->metacity_theme_name = g_strdup (info->metacity_theme_name);
else
custom->metacity_theme_name = get_default_string_from_key (data->client, METACITY_THEME_KEY);
if (info->gtk_color_scheme)
custom->gtk_color_scheme = g_strdup (info->gtk_color_scheme);
else
custom->gtk_color_scheme = get_default_string_from_key (data->client, COLOR_SCHEME_KEY);
}
if (!find_in_model (data->theme_store, custom->name, COL_NAME, NULL)) {
gtk_list_store_insert_with_values (data->theme_store, NULL, 0,
COL_LABEL, custom->readable_name,
COL_NAME, custom->name,
-1);
}
/* update the theme thumbnail */
theme_queue_for_thumbnail (custom, data);
}
static void
theme_remove_custom (GtkIconView *icon_view, AppearanceData *data)
{
GtkTreeIter iter;
if (find_in_model (data->theme_store, CUSTOM_THEME_NAME, COL_NAME, &iter))
gtk_list_store_remove (data->theme_store, &iter);
}
static gboolean
theme_is_equal (const GnomeThemeMetaInfo *a, const GnomeThemeMetaInfo *b)
{
if (!(a->gtk_theme_name && b->gtk_theme_name) ||
strcmp (a->gtk_theme_name, b->gtk_theme_name))
return FALSE;
if (!(a->gtk_color_scheme && b->gtk_color_scheme) ||
strcmp (a->gtk_color_scheme, b->gtk_color_scheme))
return FALSE;
if (!(a->icon_theme_name && b->icon_theme_name) ||
strcmp (a->icon_theme_name, b->icon_theme_name))
return FALSE;
if (!(a->metacity_theme_name && b->metacity_theme_name) ||
strcmp (a->metacity_theme_name, b->metacity_theme_name))
return FALSE;
return TRUE;
}
void
themes_init (AppearanceData *data)
{
GtkWidget *w;
GList *theme_list, *l;
GtkListStore *theme_store;
GtkTreeModel *sort_model;
gchar *meta_theme = NULL;
GdkPixbuf *temp;
/* initialise some stuff */
gnome_theme_init (NULL);
gnome_wm_manager_init ();
data->theme_queue = NULL;
data->theme_custom = gnome_theme_meta_info_new ();
/* connect button signals */
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "theme_custom")), "clicked", (GCallback) theme_custom_cb, data);
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "theme_install")), "clicked", (GCallback) theme_install_cb, data);
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "theme_delete")), "clicked", (GCallback) theme_delete_cb, data);
/* connect theme list signals */
g_signal_connect (G_OBJECT (glade_xml_get_widget (data->xml, "theme_list")), "selection-changed", (GCallback) theme_selection_changed_cb, data);
/* set up theme list */
data->theme_store = theme_store =
gtk_list_store_new (NUM_COLS, G_TYPE_STRING, GDK_TYPE_PIXBUF, G_TYPE_STRING);
theme_list = gnome_theme_meta_info_find_all ();
gnome_theme_info_register_theme_change ((GFunc) theme_changed_func, data);
data->theme_custom->name = g_strdup (CUSTOM_THEME_NAME);
data->theme_custom->readable_name = g_strdup ("Custom"); /* FIXME: translate */
data->theme_custom->gtk_theme_name =
gconf_client_get_string (data->client, GTK_THEME_KEY, NULL);
data->theme_custom->gtk_color_scheme =
gconf_client_get_string (data->client, COLOR_SCHEME_KEY, NULL);
data->theme_custom->metacity_theme_name =
gconf_client_get_string (data->client, METACITY_THEME_KEY, NULL);
data->theme_custom->icon_theme_name =
gconf_client_get_string (data->client, ICON_THEME_KEY, NULL);
temp = gdk_pixbuf_new_from_file (GNOMECC_PIXMAP_DIR "/theme-thumbnailing.png", NULL);
for (l = theme_list; l; l = l->next) {
GnomeThemeMetaInfo *info = l->data;
data->theme_queue = g_slist_prepend (data->theme_queue, info);
gtk_list_store_insert_with_values (theme_store, NULL, 0,
COL_LABEL, info->readable_name,
COL_NAME, info->name,
COL_THUMBNAIL, temp,
-1);
if (!meta_theme && theme_is_equal (data->theme_custom, info))
meta_theme = info->name;
}
if (!meta_theme) {
/* add custom theme */
meta_theme = data->theme_custom->name;
data->theme_queue = g_slist_prepend (data->theme_queue, data->theme_custom);
gtk_list_store_insert_with_values (theme_store, NULL, 0,
COL_LABEL, data->theme_custom->readable_name,
COL_NAME, meta_theme,
COL_THUMBNAIL, temp,
-1);
}
if (temp)
g_object_unref (temp);
w = glade_xml_get_widget (data->xml, "theme_list");
sort_model = gtk_tree_model_sort_new_with_model (GTK_TREE_MODEL (theme_store));
gtk_icon_view_set_model (GTK_ICON_VIEW (w), GTK_TREE_MODEL (sort_model));
gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (sort_model), COL_LABEL, GTK_SORT_ASCENDING);
g_signal_connect_after (G_OBJECT (w), "realize",
G_CALLBACK (theme_select_after_realize), meta_theme);
w = glade_xml_get_widget (data->xml, "theme_install");
gtk_button_set_image (GTK_BUTTON (w),
gtk_image_new_from_stock (GTK_STOCK_OPEN, GTK_ICON_SIZE_BUTTON));
theme_thumbnail_generate (data);
}
void
themes_shutdown (AppearanceData *data)
{
gnome_theme_meta_info_free (data->theme_custom);
g_slist_free (data->theme_queue);
}