/* -*- 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 <micke@imendio.com>
* Richard Hult <richard@imendio.com>
* Martyn Russell <martyn@imendio.com>
*/
#include <config.h>
#include <string.h>
#include <gtk/gtk.h>
#include <glib/gi18n.h>
#include <libempathy/empathy-utils.h>
#include <libempathy-gtk/empathy-conf.h>
#include <libempathy-gtk/empathy-ui-utils.h>
#include <libempathy-gtk/empathy-theme-manager.h>
#include <libempathy-gtk/empathy-spell.h>
#include <libempathy-gtk/empathy-contact-list-store.h>
#include <libempathy-gtk/empathy-gtk-enum-types.h>
#ifdef HAVE_WEBKIT
#include <libempathy-gtk/empathy-theme-adium.h>
#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 *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_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_theme_adium_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,
"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;
}