/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */ /* * Copyright (C) 2003-2007 Imendio AB * * 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 St, Fifth Floor, * Boston, MA 02110-1301 USA * * Authors: Mikael Hallendal * Richard Hult * Martyn Russell */ #include #include #include #include #include #include #include #include #include #include #include #ifdef HAVE_WEBKIT #include #endif #include "empathy-preferences.h" typedef struct { GtkWidget *dialog; GtkWidget *notebook; GtkWidget *checkbutton_show_avatars; GtkWidget *checkbutton_compact_contact_list; GtkWidget *checkbutton_show_smileys; GtkWidget *checkbutton_show_contacts_in_rooms; GtkWidget *combobox_chat_theme; GtkWidget *hbox_adium_theme; GtkWidget *filechooserbutton_adium_theme; GtkWidget *label_invalid_adium_theme; GtkWidget *checkbutton_separate_chat_windows; GtkWidget *checkbutton_autoconnect; GtkWidget *radiobutton_contact_list_sort_by_name; GtkWidget *radiobutton_contact_list_sort_by_state; GtkWidget *checkbutton_sounds_enabled; GtkWidget *checkbutton_sounds_disabled_away; GtkWidget *treeview_sounds; GtkWidget *checkbutton_notifications_enabled; GtkWidget *checkbutton_notifications_disabled_away; GtkWidget *checkbutton_notifications_focus; GtkWidget *treeview_spell_checker; GtkWidget *checkbutton_location_publish; GtkWidget *checkbutton_location_reduce_accuracy; GtkWidget *checkbutton_location_resource_network; GtkWidget *checkbutton_location_resource_cell; GtkWidget *checkbutton_location_resource_gps; GList *notify_ids; } EmpathyPreferences; static void preferences_setup_widgets (EmpathyPreferences *preferences); static void preferences_languages_setup (EmpathyPreferences *preferences); static void preferences_languages_add (EmpathyPreferences *preferences); static void preferences_languages_save (EmpathyPreferences *preferences); static gboolean preferences_languages_save_foreach (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gchar **languages); static void preferences_languages_load (EmpathyPreferences *preferences); static gboolean preferences_languages_load_foreach (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gchar **languages); static void preferences_languages_cell_toggled_cb (GtkCellRendererToggle *cell, gchar *path_string, EmpathyPreferences *preferences); static void preferences_widget_sync_bool (const gchar *key, GtkWidget *widget); static void preferences_widget_sync_string (const gchar *key, GtkWidget *widget); static void preferences_notify_string_cb (EmpathyConf *conf, const gchar *key, gpointer user_data); static void preferences_notify_bool_cb (EmpathyConf *conf, const gchar *key, gpointer user_data); static void preferences_notify_sensitivity_cb (EmpathyConf *conf, const gchar *key, gpointer user_data); static void preferences_hookup_toggle_button (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget); static void preferences_hookup_radio_button (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget); static void preferences_hookup_sensitivity (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget); static void preferences_toggle_button_toggled_cb (GtkWidget *button, gpointer user_data); static void preferences_radio_button_toggled_cb (GtkWidget *button, gpointer user_data); static void preferences_destroy_cb (GtkWidget *widget, EmpathyPreferences *preferences); static void preferences_response_cb (GtkWidget *widget, gint response, EmpathyPreferences *preferences); enum { COL_LANG_ENABLED, COL_LANG_CODE, COL_LANG_NAME, COL_LANG_COUNT }; enum { COL_COMBO_VISIBLE_NAME, COL_COMBO_NAME, COL_COMBO_COUNT }; enum { COL_SOUND_ENABLED, COL_SOUND_NAME, COL_SOUND_KEY, COL_SOUND_COUNT }; typedef struct { const char *name; const char *key; } SoundEventEntry; /* TODO: add phone related sounds also? */ static SoundEventEntry sound_entries [] = { { N_("Message received"), EMPATHY_PREFS_SOUNDS_INCOMING_MESSAGE }, { N_("Message sent"), EMPATHY_PREFS_SOUNDS_OUTGOING_MESSAGE }, { N_("New conversation"), EMPATHY_PREFS_SOUNDS_NEW_CONVERSATION }, { N_("Contact goes online"), EMPATHY_PREFS_SOUNDS_CONTACT_LOGIN }, { N_("Contact goes offline"), EMPATHY_PREFS_SOUNDS_CONTACT_LOGOUT }, { N_("Account connected"), EMPATHY_PREFS_SOUNDS_SERVICE_LOGIN }, { N_("Account disconnected"), EMPATHY_PREFS_SOUNDS_SERVICE_LOGOUT } }; static void preferences_add_id (EmpathyPreferences *preferences, guint id) { preferences->notify_ids = g_list_prepend (preferences->notify_ids, GUINT_TO_POINTER (id)); } static void preferences_compact_contact_list_changed_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { EmpathyPreferences *preferences = user_data; gboolean value; if (empathy_conf_get_bool (empathy_conf_get (), key, &value)) { gtk_widget_set_sensitive (preferences->checkbutton_show_avatars, !value); } } static void preferences_setup_widgets (EmpathyPreferences *preferences) { guint id; preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_NOTIFICATIONS_ENABLED, preferences->checkbutton_notifications_enabled); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_NOTIFICATIONS_DISABLED_AWAY, preferences->checkbutton_notifications_disabled_away); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_NOTIFICATIONS_FOCUS, preferences->checkbutton_notifications_focus); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_NOTIFICATIONS_ENABLED, preferences->checkbutton_notifications_disabled_away); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_NOTIFICATIONS_ENABLED, preferences->checkbutton_notifications_focus); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_SOUNDS_ENABLED, preferences->checkbutton_sounds_enabled); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_SOUNDS_DISABLED_AWAY, preferences->checkbutton_sounds_disabled_away); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_SOUNDS_ENABLED, preferences->checkbutton_sounds_disabled_away); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_SOUNDS_ENABLED, preferences->treeview_sounds); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_UI_SEPARATE_CHAT_WINDOWS, preferences->checkbutton_separate_chat_windows); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_UI_SHOW_AVATARS, preferences->checkbutton_show_avatars); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_UI_COMPACT_CONTACT_LIST, preferences->checkbutton_compact_contact_list); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_CHAT_SHOW_SMILEYS, preferences->checkbutton_show_smileys); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_CHAT_SHOW_CONTACTS_IN_ROOMS, preferences->checkbutton_show_contacts_in_rooms); preferences_hookup_radio_button (preferences, EMPATHY_PREFS_CONTACTS_SORT_CRITERIUM, preferences->radiobutton_contact_list_sort_by_name); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_AUTOCONNECT, preferences->checkbutton_autoconnect); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_LOCATION_PUBLISH, preferences->checkbutton_location_publish); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_LOCATION_RESOURCE_NETWORK, preferences->checkbutton_location_resource_network); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_LOCATION_PUBLISH, preferences->checkbutton_location_resource_network); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_LOCATION_RESOURCE_CELL, preferences->checkbutton_location_resource_cell); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_LOCATION_PUBLISH, preferences->checkbutton_location_resource_cell); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_LOCATION_RESOURCE_GPS, preferences->checkbutton_location_resource_gps); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_LOCATION_PUBLISH, preferences->checkbutton_location_resource_gps); preferences_hookup_toggle_button (preferences, EMPATHY_PREFS_LOCATION_REDUCE_ACCURACY, preferences->checkbutton_location_reduce_accuracy); preferences_hookup_sensitivity (preferences, EMPATHY_PREFS_LOCATION_PUBLISH, preferences->checkbutton_location_reduce_accuracy); id = empathy_conf_notify_add (empathy_conf_get (), EMPATHY_PREFS_UI_COMPACT_CONTACT_LIST, preferences_compact_contact_list_changed_cb, preferences); if (id) { preferences_add_id (preferences, id); } preferences_compact_contact_list_changed_cb (empathy_conf_get (), EMPATHY_PREFS_UI_COMPACT_CONTACT_LIST, preferences); } static void preferences_sound_cell_toggled_cb (GtkCellRendererToggle *toggle, char *path_string, EmpathyPreferences *preferences) { GtkTreePath *path; gboolean toggled, instore; GtkTreeIter iter; GtkTreeView *view; GtkTreeModel *model; char *key; view = GTK_TREE_VIEW (preferences->treeview_sounds); model = gtk_tree_view_get_model (view); path = gtk_tree_path_new_from_string (path_string); toggled = gtk_cell_renderer_toggle_get_active (toggle); gtk_tree_model_get_iter (model, &iter, path); gtk_tree_model_get (model, &iter, COL_SOUND_KEY, &key, COL_SOUND_ENABLED, &instore, -1); instore ^= 1; gtk_list_store_set (GTK_LIST_STORE (model), &iter, COL_SOUND_ENABLED, instore, -1); empathy_conf_set_bool (empathy_conf_get (), key, instore); g_free (key); gtk_tree_path_free (path); } static void preferences_sound_load (EmpathyPreferences *preferences) { int i; GtkTreeView *view; GtkListStore *store; GtkTreeIter iter; gboolean set; EmpathyConf *conf; view = GTK_TREE_VIEW (preferences->treeview_sounds); store = GTK_LIST_STORE (gtk_tree_view_get_model (view)); conf = empathy_conf_get (); for (i = 0; i < G_N_ELEMENTS (sound_entries); i++) { empathy_conf_get_bool (conf, sound_entries[i].key, &set); gtk_list_store_insert_with_values (store, &iter, i, COL_SOUND_NAME, gettext (sound_entries[i].name), COL_SOUND_KEY, sound_entries[i].key, COL_SOUND_ENABLED, set, -1); } } static void preferences_sound_setup (EmpathyPreferences *preferences) { GtkTreeView *view; GtkListStore *store; GtkCellRenderer *renderer; GtkTreeViewColumn *column; view = GTK_TREE_VIEW (preferences->treeview_sounds); store = gtk_list_store_new (COL_SOUND_COUNT, G_TYPE_BOOLEAN, /* enabled */ G_TYPE_STRING, /* name */ G_TYPE_STRING); /* key */ gtk_tree_view_set_model (view, GTK_TREE_MODEL (store)); renderer = gtk_cell_renderer_toggle_new (); g_signal_connect (renderer, "toggled", G_CALLBACK (preferences_sound_cell_toggled_cb), preferences); column = gtk_tree_view_column_new (); gtk_tree_view_column_pack_start (column, renderer, FALSE); gtk_tree_view_column_add_attribute (column, renderer, "active", COL_SOUND_ENABLED); renderer = gtk_cell_renderer_text_new (); gtk_tree_view_column_pack_start (column, renderer, FALSE); gtk_tree_view_column_add_attribute (column, renderer, "text", COL_SOUND_NAME); gtk_tree_view_append_column (view, column); gtk_tree_view_column_set_resizable (column, FALSE); gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (column), TRUE); g_object_unref (store); } static void preferences_languages_setup (EmpathyPreferences *preferences) { GtkTreeView *view; GtkListStore *store; GtkTreeSelection *selection; GtkTreeModel *model; GtkTreeViewColumn *column; GtkCellRenderer *renderer; guint col_offset; view = GTK_TREE_VIEW (preferences->treeview_spell_checker); store = gtk_list_store_new (COL_LANG_COUNT, G_TYPE_BOOLEAN, /* enabled */ G_TYPE_STRING, /* code */ G_TYPE_STRING); /* name */ gtk_tree_view_set_model (view, GTK_TREE_MODEL (store)); selection = gtk_tree_view_get_selection (view); gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE); model = GTK_TREE_MODEL (store); renderer = gtk_cell_renderer_toggle_new (); g_signal_connect (renderer, "toggled", G_CALLBACK (preferences_languages_cell_toggled_cb), preferences); column = gtk_tree_view_column_new_with_attributes (NULL, renderer, "active", COL_LANG_ENABLED, NULL); gtk_tree_view_append_column (view, column); renderer = gtk_cell_renderer_text_new (); col_offset = gtk_tree_view_insert_column_with_attributes (view, -1, _("Language"), renderer, "text", COL_LANG_NAME, NULL); g_object_set_data (G_OBJECT (renderer), "column", GINT_TO_POINTER (COL_LANG_NAME)); column = gtk_tree_view_get_column (view, col_offset - 1); gtk_tree_view_column_set_sort_column_id (column, COL_LANG_NAME); gtk_tree_view_column_set_resizable (column, FALSE); gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (column), TRUE); g_object_unref (store); } static void preferences_languages_add (EmpathyPreferences *preferences) { GtkTreeView *view; GtkListStore *store; GList *codes, *l; view = GTK_TREE_VIEW (preferences->treeview_spell_checker); store = GTK_LIST_STORE (gtk_tree_view_get_model (view)); codes = empathy_spell_get_language_codes (); empathy_conf_set_bool (empathy_conf_get (), EMPATHY_PREFS_CHAT_SPELL_CHECKER_ENABLED, codes != NULL); if (!codes) { gtk_widget_set_sensitive (preferences->treeview_spell_checker, FALSE); } for (l = codes; l; l = l->next) { GtkTreeIter iter; const gchar *code; const gchar *name; code = l->data; name = empathy_spell_get_language_name (code); if (!name) { continue; } gtk_list_store_append (store, &iter); gtk_list_store_set (store, &iter, COL_LANG_CODE, code, COL_LANG_NAME, name, -1); } empathy_spell_free_language_codes (codes); } static void preferences_languages_save (EmpathyPreferences *preferences) { GtkTreeView *view; GtkTreeModel *model; gchar *languages = NULL; view = GTK_TREE_VIEW (preferences->treeview_spell_checker); model = gtk_tree_view_get_model (view); gtk_tree_model_foreach (model, (GtkTreeModelForeachFunc) preferences_languages_save_foreach, &languages); /* if user selects no languages, we don't want spell check */ empathy_conf_set_bool (empathy_conf_get (), EMPATHY_PREFS_CHAT_SPELL_CHECKER_ENABLED, languages != NULL); empathy_conf_set_string (empathy_conf_get (), EMPATHY_PREFS_CHAT_SPELL_CHECKER_LANGUAGES, languages ? languages : ""); g_free (languages); } static gboolean preferences_languages_save_foreach (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gchar **languages) { gboolean enabled; gchar *code; if (!languages) { return TRUE; } gtk_tree_model_get (model, iter, COL_LANG_ENABLED, &enabled, -1); if (!enabled) { return FALSE; } gtk_tree_model_get (model, iter, COL_LANG_CODE, &code, -1); if (!code) { return FALSE; } if (!(*languages)) { *languages = g_strdup (code); } else { gchar *str = *languages; *languages = g_strdup_printf ("%s,%s", str, code); g_free (str); } g_free (code); return FALSE; } static void preferences_languages_load (EmpathyPreferences *preferences) { GtkTreeView *view; GtkTreeModel *model; gchar *value; gchar **vlanguages; if (!empathy_conf_get_string (empathy_conf_get (), EMPATHY_PREFS_CHAT_SPELL_CHECKER_LANGUAGES, &value) || !value) { return; } vlanguages = g_strsplit (value, ",", -1); g_free (value); view = GTK_TREE_VIEW (preferences->treeview_spell_checker); model = gtk_tree_view_get_model (view); gtk_tree_model_foreach (model, (GtkTreeModelForeachFunc) preferences_languages_load_foreach, vlanguages); g_strfreev (vlanguages); } static gboolean preferences_languages_load_foreach (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gchar **languages) { gchar *code; gchar *lang; gint i; gboolean found = FALSE; if (!languages) { return TRUE; } gtk_tree_model_get (model, iter, COL_LANG_CODE, &code, -1); if (!code) { return FALSE; } for (i = 0, lang = languages[i]; lang; lang = languages[++i]) { if (strcmp (lang, code) == 0) { found = TRUE; } } g_free (code); gtk_list_store_set (GTK_LIST_STORE (model), iter, COL_LANG_ENABLED, found, -1); return FALSE; } static void preferences_languages_cell_toggled_cb (GtkCellRendererToggle *cell, gchar *path_string, EmpathyPreferences *preferences) { GtkTreeView *view; GtkTreeModel *model; GtkListStore *store; GtkTreePath *path; GtkTreeIter iter; gboolean enabled; view = GTK_TREE_VIEW (preferences->treeview_spell_checker); model = gtk_tree_view_get_model (view); store = GTK_LIST_STORE (model); path = gtk_tree_path_new_from_string (path_string); gtk_tree_model_get_iter (model, &iter, path); gtk_tree_model_get (model, &iter, COL_LANG_ENABLED, &enabled, -1); enabled ^= 1; gtk_list_store_set (store, &iter, COL_LANG_ENABLED, enabled, -1); gtk_tree_path_free (path); preferences_languages_save (preferences); } static void preferences_widget_sync_bool (const gchar *key, GtkWidget *widget) { gboolean value; if (empathy_conf_get_bool (empathy_conf_get (), key, &value)) { gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (widget), value); } } static void preferences_widget_sync_string (const gchar *key, GtkWidget *widget) { gchar *value; if (empathy_conf_get_string (empathy_conf_get (), key, &value) && value) { if (GTK_IS_ENTRY (widget)) { gtk_entry_set_text (GTK_ENTRY (widget), value); } else if (GTK_IS_RADIO_BUTTON (widget)) { if (strcmp (key, EMPATHY_PREFS_CONTACTS_SORT_CRITERIUM) == 0) { GType type; GEnumClass *enum_class; GEnumValue *enum_value; GSList *list; GtkWidget *toggle_widget; /* Get index from new string */ type = empathy_contact_list_store_sort_get_type (); enum_class = G_ENUM_CLASS (g_type_class_peek (type)); enum_value = g_enum_get_value_by_nick (enum_class, value); if (enum_value) { list = gtk_radio_button_get_group (GTK_RADIO_BUTTON (widget)); toggle_widget = g_slist_nth_data (list, enum_value->value); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle_widget), TRUE); } } else { g_warning ("Unhandled key:'%s' just had string change", key); } } g_free (value); } } static void preferences_notify_string_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { preferences_widget_sync_string (key, user_data); } static void preferences_notify_bool_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { preferences_widget_sync_bool (key, user_data); } static void preferences_notify_sensitivity_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { gboolean value; if (empathy_conf_get_bool (conf, key, &value)) { gtk_widget_set_sensitive (GTK_WIDGET (user_data), value); } } #if 0 static void preferences_widget_sync_int (const gchar *key, GtkWidget *widget) { gint value; if (empathy_conf_get_int (empathy_conf_get (), key, &value)) { if (GTK_IS_SPIN_BUTTON (widget)) { gtk_spin_button_set_value (GTK_SPIN_BUTTON (widget), value); } } } static void preferences_notify_int_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { preferences_widget_sync_int (key, user_data); } static void preferences_hookup_spin_button (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget) { guint id; preferences_widget_sync_int (key, widget); g_object_set_data_full (G_OBJECT (widget), "key", g_strdup (key), g_free); g_signal_connect (widget, "value_changed", G_CALLBACK (preferences_spin_button_value_changed_cb), NULL); id = empathy_conf_notify_add (empathy_conf_get (), key, preferences_notify_int_cb, widget); if (id) { preferences_add_id (preferences, id); } } static void preferences_hookup_entry (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget) { guint id; preferences_widget_sync_string (key, widget); g_object_set_data_full (G_OBJECT (widget), "key", g_strdup (key), g_free); g_signal_connect (widget, "changed", G_CALLBACK (preferences_entry_value_changed_cb), NULL); id = empathy_conf_notify_add (empathy_conf_get (), key, preferences_notify_string_cb, widget); if (id) { preferences_add_id (preferences, id); } } static void preferences_spin_button_value_changed_cb (GtkWidget *button, gpointer user_data) { const gchar *key; key = g_object_get_data (G_OBJECT (button), "key"); empathy_conf_set_int (empathy_conf_get (), key, gtk_spin_button_get_value (GTK_SPIN_BUTTON (button))); } static void preferences_entry_value_changed_cb (GtkWidget *entry, gpointer user_data) { const gchar *key; key = g_object_get_data (G_OBJECT (entry), "key"); empathy_conf_set_string (empathy_conf_get (), key, gtk_entry_get_text (GTK_ENTRY (entry))); } #endif static void preferences_hookup_toggle_button (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget) { guint id; preferences_widget_sync_bool (key, widget); g_object_set_data_full (G_OBJECT (widget), "key", g_strdup (key), g_free); g_signal_connect (widget, "toggled", G_CALLBACK (preferences_toggle_button_toggled_cb), NULL); id = empathy_conf_notify_add (empathy_conf_get (), key, preferences_notify_bool_cb, widget); if (id) { preferences_add_id (preferences, id); } } static void preferences_hookup_radio_button (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget) { GSList *group, *l; guint id; preferences_widget_sync_string (key, widget); group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (widget)); for (l = group; l; l = l->next) { g_signal_connect (l->data, "toggled", G_CALLBACK (preferences_radio_button_toggled_cb), NULL); g_object_set_data_full (G_OBJECT (l->data), "key", g_strdup (key), g_free); } id = empathy_conf_notify_add (empathy_conf_get (), key, preferences_notify_string_cb, widget); if (id) { preferences_add_id (preferences, id); } } static void preferences_hookup_sensitivity (EmpathyPreferences *preferences, const gchar *key, GtkWidget *widget) { gboolean value; guint id; if (empathy_conf_get_bool (empathy_conf_get (), key, &value)) { gtk_widget_set_sensitive (widget, value); } id = empathy_conf_notify_add (empathy_conf_get (), key, preferences_notify_sensitivity_cb, widget); if (id) { preferences_add_id (preferences, id); } } static void preferences_toggle_button_toggled_cb (GtkWidget *button, gpointer user_data) { const gchar *key; key = g_object_get_data (G_OBJECT (button), "key"); empathy_conf_set_bool (empathy_conf_get (), key, gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button))); } static void preferences_radio_button_toggled_cb (GtkWidget *button, gpointer user_data) { const gchar *key; const gchar *value = NULL; if (!gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (button))) { return; } key = g_object_get_data (G_OBJECT (button), "key"); if (key && strcmp (key, EMPATHY_PREFS_CONTACTS_SORT_CRITERIUM) == 0) { GSList *group; GType type; GEnumClass *enum_class; GEnumValue *enum_value; group = gtk_radio_button_get_group (GTK_RADIO_BUTTON (button)); /* Get string from index */ type = empathy_contact_list_store_sort_get_type (); enum_class = G_ENUM_CLASS (g_type_class_peek (type)); enum_value = g_enum_get_value (enum_class, g_slist_index (group, button)); if (!enum_value) { g_warning ("No GEnumValue for EmpathyContactListSort with GtkRadioButton index:%d", g_slist_index (group, button)); return; } value = enum_value->value_nick; } else if (key && strcmp (key, EMPATHY_PREFS_CONTACTS_SORT_CRITERIUM) == 0) { return; } empathy_conf_set_string (empathy_conf_get (), key, value); } static void preferences_theme_adium_update_visibility (EmpathyPreferences *preferences, const gchar *name) { if (name && strcmp (name, "adium") == 0) { gtk_widget_show (preferences->hbox_adium_theme); } else { gtk_widget_hide (preferences->hbox_adium_theme); gtk_widget_hide (preferences->label_invalid_adium_theme); } } static void preferences_theme_adium_update_validity (EmpathyPreferences *preferences, const gchar *path) { #ifdef HAVE_WEBKIT if (empathy_adium_path_is_valid (path)) { gtk_widget_hide (preferences->label_invalid_adium_theme); } else { gtk_widget_show (preferences->label_invalid_adium_theme); } #endif } static void preferences_theme_adium_path_notify_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { EmpathyPreferences *preferences = user_data; GtkFileChooser *chooser; gchar *value; const gchar *path; if (!empathy_conf_get_string (conf, key, &value)) { return; } if (EMP_STR_EMPTY (value)) { path = g_get_home_dir (); } else { path = value; } chooser = GTK_FILE_CHOOSER (preferences->filechooserbutton_adium_theme); gtk_file_chooser_set_current_folder (chooser, path); preferences_theme_adium_update_validity (preferences, path); g_free (value); } static void preferences_theme_adium_file_set_cb (GtkFileChooser *chooser, EmpathyPreferences *preferences) { gchar *path; path = gtk_file_chooser_get_current_folder (chooser); empathy_conf_set_string (empathy_conf_get (), EMPATHY_PREFS_CHAT_ADIUM_PATH, path); preferences_theme_adium_update_validity (preferences, path); g_free (path); } static void preferences_theme_notify_cb (EmpathyConf *conf, const gchar *key, gpointer user_data) { EmpathyPreferences *preferences = user_data; GtkComboBox *combo; gchar *value; GtkTreeModel *model; GtkTreeIter iter; gboolean found = FALSE; if (!empathy_conf_get_string (conf, key, &value)) { return; } preferences_theme_adium_update_visibility (preferences, value); combo = GTK_COMBO_BOX (preferences->combobox_chat_theme); model = gtk_combo_box_get_model (combo); if (value && gtk_tree_model_get_iter_first (model, &iter)) { gchar *name; do { gtk_tree_model_get (model, &iter, COL_COMBO_NAME, &name, -1); if (strcmp (name, value) == 0) { found = TRUE; gtk_combo_box_set_active_iter (combo, &iter); break; } g_free (name); } while (gtk_tree_model_iter_next (model, &iter)); } /* Fallback to the first one. */ if (!found) { if (gtk_tree_model_get_iter_first (model, &iter)) { gtk_combo_box_set_active_iter (combo, &iter); } } g_free (value); } static void preferences_theme_changed_cb (GtkComboBox *combo, EmpathyPreferences *preferences) { GtkTreeModel *model; GtkTreeIter iter; gchar *name; if (gtk_combo_box_get_active_iter (combo, &iter)) { model = gtk_combo_box_get_model (combo); gtk_tree_model_get (model, &iter, COL_COMBO_NAME, &name, -1); preferences_theme_adium_update_visibility (preferences, name); empathy_conf_set_string (empathy_conf_get (), EMPATHY_PREFS_CHAT_THEME, name); g_free (name); } } static void preferences_themes_setup (EmpathyPreferences *preferences) { GtkComboBox *combo; GtkCellLayout *cell_layout; GtkCellRenderer *renderer; GtkListStore *store; const gchar **themes; gint i; guint id; combo = GTK_COMBO_BOX (preferences->combobox_chat_theme); cell_layout = GTK_CELL_LAYOUT (combo); /* Create the model */ store = gtk_list_store_new (COL_COMBO_COUNT, G_TYPE_STRING, /* Display name */ G_TYPE_STRING); /* Theme name */ /* Fill the model */ themes = empathy_theme_manager_get_themes (); for (i = 0; themes[i]; i += 2) { gtk_list_store_insert_with_values (store, NULL, -1, COL_COMBO_VISIBLE_NAME, _(themes[i + 1]), COL_COMBO_NAME, themes[i], -1); } /* Add cell renderer */ renderer = gtk_cell_renderer_text_new (); gtk_cell_layout_pack_start (cell_layout, renderer, TRUE); gtk_cell_layout_set_attributes (cell_layout, renderer, "text", COL_COMBO_VISIBLE_NAME, NULL); gtk_combo_box_set_model (combo, GTK_TREE_MODEL (store)); g_object_unref (store); g_signal_connect (combo, "changed", G_CALLBACK (preferences_theme_changed_cb), preferences); /* Select the theme from the gconf key and track changes */ preferences_theme_notify_cb (empathy_conf_get (), EMPATHY_PREFS_CHAT_THEME, preferences); id = empathy_conf_notify_add (empathy_conf_get (), EMPATHY_PREFS_CHAT_THEME, preferences_theme_notify_cb, preferences); if (id) { preferences_add_id (preferences, id); } g_signal_connect (preferences->filechooserbutton_adium_theme, "file-set", G_CALLBACK (preferences_theme_adium_file_set_cb), preferences); /* Select the adium path from the gconf key and track changes */ preferences_theme_adium_path_notify_cb (empathy_conf_get (), EMPATHY_PREFS_CHAT_ADIUM_PATH, preferences); id = empathy_conf_notify_add (empathy_conf_get (), EMPATHY_PREFS_CHAT_ADIUM_PATH, preferences_theme_adium_path_notify_cb, preferences); if (id) { preferences_add_id (preferences, id); } } static void preferences_response_cb (GtkWidget *widget, gint response, EmpathyPreferences *preferences) { gtk_widget_destroy (widget); } static void preferences_destroy_cb (GtkWidget *widget, EmpathyPreferences *preferences) { GList *l; for (l = preferences->notify_ids; l; l = l->next) { guint id; id = GPOINTER_TO_UINT (l->data); empathy_conf_notify_remove (empathy_conf_get (), id); } g_list_free (preferences->notify_ids); g_free (preferences); } GtkWidget * empathy_preferences_show (GtkWindow *parent) { static EmpathyPreferences *preferences; GtkBuilder *gui; gchar *filename; GtkWidget *page; if (preferences) { gtk_window_present (GTK_WINDOW (preferences->dialog)); return preferences->dialog; } preferences = g_new0 (EmpathyPreferences, 1); filename = empathy_file_lookup ("empathy-preferences.ui", "src"); gui = empathy_builder_get_file (filename, "preferences_dialog", &preferences->dialog, "notebook", &preferences->notebook, "checkbutton_show_avatars", &preferences->checkbutton_show_avatars, "checkbutton_compact_contact_list", &preferences->checkbutton_compact_contact_list, "checkbutton_show_smileys", &preferences->checkbutton_show_smileys, "checkbutton_show_contacts_in_rooms", &preferences->checkbutton_show_contacts_in_rooms, "combobox_chat_theme", &preferences->combobox_chat_theme, "hbox_adium_theme", &preferences->hbox_adium_theme, "filechooserbutton_adium_theme", &preferences->filechooserbutton_adium_theme, "label_invalid_adium_theme", &preferences->label_invalid_adium_theme, "checkbutton_separate_chat_windows", &preferences->checkbutton_separate_chat_windows, "checkbutton_autoconnect", &preferences->checkbutton_autoconnect, "radiobutton_contact_list_sort_by_name", &preferences->radiobutton_contact_list_sort_by_name, "radiobutton_contact_list_sort_by_state", &preferences->radiobutton_contact_list_sort_by_state, "checkbutton_notifications_enabled", &preferences->checkbutton_notifications_enabled, "checkbutton_notifications_disabled_away", &preferences->checkbutton_notifications_disabled_away, "checkbutton_notifications_focus", &preferences->checkbutton_notifications_focus, "checkbutton_sounds_enabled", &preferences->checkbutton_sounds_enabled, "checkbutton_sounds_disabled_away", &preferences->checkbutton_sounds_disabled_away, "treeview_sounds", &preferences->treeview_sounds, "treeview_spell_checker", &preferences->treeview_spell_checker, "checkbutton_location_publish", &preferences->checkbutton_location_publish, "checkbutton_location_reduce_accuracy", &preferences->checkbutton_location_reduce_accuracy, "checkbutton_location_resource_network", &preferences->checkbutton_location_resource_network, "checkbutton_location_resource_cell", &preferences->checkbutton_location_resource_cell, "checkbutton_location_resource_gps", &preferences->checkbutton_location_resource_gps, NULL); g_free (filename); empathy_builder_connect (gui, preferences, "preferences_dialog", "destroy", preferences_destroy_cb, "preferences_dialog", "response", preferences_response_cb, NULL); g_object_unref (gui); g_object_add_weak_pointer (G_OBJECT (preferences->dialog), (gpointer) &preferences); preferences_themes_setup (preferences); preferences_setup_widgets (preferences); preferences_languages_setup (preferences); preferences_languages_add (preferences); preferences_languages_load (preferences); preferences_sound_setup (preferences); preferences_sound_load (preferences); if (empathy_spell_supported ()) { GtkWidget *page; page = gtk_notebook_get_nth_page (GTK_NOTEBOOK (preferences->notebook), 2); gtk_widget_show (page); } page = gtk_notebook_get_nth_page (GTK_NOTEBOOK (preferences->notebook), 3); #if HAVE_GEOCLUE gtk_widget_show (page); #else gtk_widget_hide (page); #endif if (parent) { gtk_window_set_transient_for (GTK_WINDOW (preferences->dialog), GTK_WINDOW (parent)); } gtk_widget_show (preferences->dialog); return preferences->dialog; }