/**
* @file gtkprefs.c GTK+ Preferences
* @ingroup gtkui
*
* gaim
*
* Gaim is the legal property of its developers, whose names are too numerous
* to list here. Please refer to the COPYRIGHT file distributed with this
* source distribution.
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*/
#include "gtkinternal.h"
#include "debug.h"
#include "notify.h"
#include "prefs.h"
#include "proxy.h"
#include "prpl.h"
#include "request.h"
#include "sound.h"
#include "util.h"
#include "network.h"
#include "gtkblist.h"
#include "gtkconv.h"
#include "gtkdebug.h"
#include "gtkimhtml.h"
#include "gtkimhtmltoolbar.h"
#include "gtkplugin.h"
#include "gtkpluginpref.h"
#include "gtkprefs.h"
#include "gtksound.h"
#include "gtkutils.h"
#include "stock.h"
#include "ui.h"
#define PROXYHOST 0
#define PROXYPORT 1
#define PROXYUSER 2
#define PROXYPASS 3
/* XXX This needs to be made static after we solve the away.c mess. */
GtkListStore *prefs_away_store = NULL;
GtkWidget *prefs_away_menu = NULL;
static GtkWidget *tree_v = NULL;
static int sound_row_sel = 0;
static GtkWidget *preflabel;
static GtkWidget *prefsnotebook;
static GtkTreeStore *prefstree;
static GtkWidget *sound_entry = NULL;
static GtkWidget *away_text = NULL;
static GtkListStore *smiley_theme_store = NULL;
static GtkWidget *prefs_proxy_frame = NULL;
static GtkWidget *prefs = NULL;
static GtkWidget *debugbutton = NULL;
static int notebook_page = 0;
static GtkTreeIter proto_iter, plugin_iter;
static guint browser_pref1_id = 0;
static guint browser_pref2_id = 0;
static guint proxy_pref_id = 0;
static guint sound_pref_id = 0;
static guint auto_resp_pref_id = 0;
static guint placement_pref_id = 0;
/*
* PROTOTYPES
*/
static GtkTreeIter *prefs_notebook_add_page(const char*, GdkPixbuf*,
GtkWidget*, GtkTreeIter*,
GtkTreeIter*, int);
static void delete_prefs(GtkWidget *, void *);
static void update_plugin_list(void *data);
static void set_default_away(GtkWidget *, gpointer);
static void
update_spin_value(GtkWidget *w, GtkWidget *spin)
{
const char *key = g_object_get_data(G_OBJECT(spin), "val");
int value;
value = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(spin));
gaim_prefs_set_int(key, value);
}
GtkWidget *
gaim_gtk_prefs_labeled_spin_button(GtkWidget *box, const gchar *title,
char *key, int min, int max, GtkSizeGroup *sg)
{
GtkWidget *hbox;
GtkWidget *label;
GtkWidget *spin;
GtkObject *adjust;
int val;
val = gaim_prefs_get_int(key);
hbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 5);
gtk_widget_show(hbox);
label = gtk_label_new_with_mnemonic(title);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
gtk_widget_show(label);
adjust = gtk_adjustment_new(val, min, max, 1, 1, 1);
spin = gtk_spin_button_new(GTK_ADJUSTMENT(adjust), 1, 0);
g_object_set_data(G_OBJECT(spin), "val", key);
if (max < 10000)
gtk_widget_set_size_request(spin, 50, -1);
else
gtk_widget_set_size_request(spin, 60, -1);
gtk_box_pack_start(GTK_BOX(hbox), spin, FALSE, FALSE, 0);
g_signal_connect(G_OBJECT(adjust), "value-changed",
G_CALLBACK(update_spin_value), GTK_WIDGET(spin));
gtk_widget_show(spin);
gtk_label_set_mnemonic_widget(GTK_LABEL(label), spin);
if (sg) {
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
}
gaim_set_accessible_label (spin, label);
return hbox;
}
static void
entry_set(GtkEntry *entry, gpointer data) {
char *key = (char*)data;
gaim_prefs_set_string(key, gtk_entry_get_text(entry));
}
GtkWidget *
gaim_gtk_prefs_labeled_entry(GtkWidget *page, const gchar *title, char *key,
GtkSizeGroup *sg)
{
GtkWidget *hbox, *label, *entry;
const gchar *value;
value = gaim_prefs_get_string(key);
hbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(page), hbox, FALSE, FALSE, 5);
gtk_widget_show(hbox);
label = gtk_label_new_with_mnemonic(title);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
gtk_widget_show(label);
entry = gtk_entry_new();
gtk_entry_set_text(GTK_ENTRY(entry), value);
gtk_box_pack_start(GTK_BOX(hbox), entry, FALSE, FALSE, 0);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(entry_set), (char*)key);
gtk_widget_show(entry);
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
if(sg) {
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
}
gaim_set_accessible_label(entry, label);
return hbox;
}
static void
dropdown_set(GObject *w, const char *key)
{
const char *str_value;
int int_value;
GaimPrefType type;
type = GPOINTER_TO_INT(g_object_get_data(w, "type"));
if (type == GAIM_PREF_INT) {
int_value = GPOINTER_TO_INT(g_object_get_data(w, "value"));
gaim_prefs_set_int(key, int_value);
}
else if (type == GAIM_PREF_STRING) {
str_value = (const char *)g_object_get_data(w, "value");
gaim_prefs_set_string(key, str_value);
}
else if (type == GAIM_PREF_BOOLEAN) {
gaim_prefs_set_bool(key,
GPOINTER_TO_INT(g_object_get_data(w, "value")));
}
}
GtkWidget *
gaim_gtk_prefs_dropdown_from_list(GtkWidget *box, const gchar *title,
GaimPrefType type, const char *key, GList *menuitems)
{
GtkWidget *dropdown, *opt, *menu;
GtkWidget *label;
GtkWidget *hbox;
gchar *text;
const char *stored_str = NULL;
int stored_int = 0;
int int_value = 0;
const char *str_value = NULL;
int o = 0;
g_return_val_if_fail(menuitems != NULL, NULL);
hbox = gtk_hbox_new(FALSE, 5);
/*gtk_container_add (GTK_CONTAINER (box), hbox);*/
gtk_box_pack_start(GTK_BOX(box), hbox, FALSE, FALSE, 0);
gtk_widget_show(hbox);
label = gtk_label_new_with_mnemonic(title);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
gtk_widget_show(label);
#if 0 /* GTK_CHECK_VERSION(2,4,0) */
if(type == GAIM_PREF_INT)
model = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_INT);
else if(type == GAIM_PREF_STRING)
model = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_STRING);
dropdown = gtk_combo_box_new_with_model(model);
#else
dropdown = gtk_option_menu_new();
menu = gtk_menu_new();
#endif
gtk_label_set_mnemonic_widget(GTK_LABEL(label), dropdown);
if (type == GAIM_PREF_INT)
stored_int = gaim_prefs_get_int(key);
else if (type == GAIM_PREF_STRING)
stored_str = gaim_prefs_get_string(key);
while (menuitems != NULL && (text = (char *) menuitems->data) != NULL) {
menuitems = g_list_next(menuitems);
g_return_val_if_fail(menuitems != NULL, NULL);
opt = gtk_menu_item_new_with_label(text);
g_object_set_data(G_OBJECT(opt), "type", GINT_TO_POINTER(type));
if (type == GAIM_PREF_INT) {
int_value = GPOINTER_TO_INT(menuitems->data);
g_object_set_data(G_OBJECT(opt), "value",
GINT_TO_POINTER(int_value));
}
else if (type == GAIM_PREF_STRING) {
str_value = (const char *)menuitems->data;
g_object_set_data(G_OBJECT(opt), "value", (char *)str_value);
}
else if (type == GAIM_PREF_BOOLEAN) {
g_object_set_data(G_OBJECT(opt), "value",
menuitems->data);
}
g_signal_connect(G_OBJECT(opt), "activate",
G_CALLBACK(dropdown_set), (char *)key);
gtk_widget_show(opt);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), opt);
if ((type == GAIM_PREF_INT && stored_int == int_value) ||
(type == GAIM_PREF_STRING && stored_str != NULL &&
!strcmp(stored_str, str_value)) ||
(type == GAIM_PREF_BOOLEAN &&
(gaim_prefs_get_bool(key) == GPOINTER_TO_INT(menuitems->data)))) {
gtk_menu_set_active(GTK_MENU(menu), o);
}
menuitems = g_list_next(menuitems);
o++;
}
gtk_option_menu_set_menu(GTK_OPTION_MENU(dropdown), menu);
gtk_box_pack_start(GTK_BOX(hbox), dropdown, FALSE, FALSE, 0);
gtk_widget_show(dropdown);
gaim_set_accessible_label (dropdown, label);
return label;
}
GtkWidget *
gaim_gtk_prefs_dropdown(GtkWidget *box, const gchar *title, GaimPrefType type,
const char *key, ...)
{
va_list ap;
GList *menuitems = NULL;
GtkWidget *dropdown = NULL;
char *name;
int int_value;
const char *str_value;
g_return_val_if_fail(type == GAIM_PREF_BOOLEAN || type == GAIM_PREF_INT ||
type == GAIM_PREF_STRING, NULL);
va_start(ap, key);
while ((name = va_arg(ap, char *)) != NULL) {
menuitems = g_list_prepend(menuitems, name);
if (type == GAIM_PREF_INT || type == GAIM_PREF_BOOLEAN) {
int_value = va_arg(ap, int);
menuitems = g_list_prepend(menuitems, GINT_TO_POINTER(int_value));
}
else {
str_value = va_arg(ap, const char *);
menuitems = g_list_prepend(menuitems, (char *)str_value);
}
}
va_end(ap);
g_return_val_if_fail(menuitems != NULL, NULL);
menuitems = g_list_reverse(menuitems);
dropdown = gaim_gtk_prefs_dropdown_from_list(box, title, type, key,
menuitems);
g_list_free(menuitems);
return dropdown;
}
static void
delete_prefs(GtkWidget *asdf, void *gdsa)
{
GList *l;
GaimPlugin *plug;
/* Close any "select sound" request dialogs */
gaim_request_close_with_handle(prefs);
gaim_plugins_unregister_probe_notify_cb(update_plugin_list);
prefs = NULL;
tree_v = NULL;
sound_entry = NULL;
debugbutton = NULL;
prefs_away_menu = NULL;
notebook_page = 0;
smiley_theme_store = NULL;
g_object_unref(G_OBJECT(prefs_away_store));
prefs_away_store = NULL;
/* Unregister callbacks. */
gaim_prefs_disconnect_callback(browser_pref1_id);
gaim_prefs_disconnect_callback(browser_pref2_id);
gaim_prefs_disconnect_callback(proxy_pref_id);
gaim_prefs_disconnect_callback(sound_pref_id);
gaim_prefs_disconnect_callback(auto_resp_pref_id);
gaim_prefs_disconnect_callback(placement_pref_id);
for (l = gaim_plugins_get_loaded(); l != NULL; l = l->next) {
plug = l->data;
if (GAIM_IS_GTK_PLUGIN(plug)) {
GaimGtkPluginUiInfo *ui_info;
ui_info = GAIM_GTK_PLUGIN_UI_INFO(plug);
if (ui_info->iter != NULL) {
g_free(ui_info->iter);
ui_info->iter = NULL;
}
}
if(GAIM_PLUGIN_HAS_PREF_FRAME(plug)) {
GaimPluginUiInfo *prefs_info;
prefs_info = GAIM_PLUGIN_UI_INFO(plug);
if(prefs_info->frame != NULL) {
gaim_plugin_pref_frame_destroy(prefs_info->frame);
prefs_info->frame = NULL;
}
if(prefs_info->iter != NULL) {
g_free(prefs_info->iter);
prefs_info->iter = NULL;
}
}
}
}
static void pref_nb_select(GtkTreeSelection *sel, GtkNotebook *nb) {
GtkTreeIter iter;
char text[128];
GValue val = { 0, };
GtkTreeModel *model = GTK_TREE_MODEL(prefstree);
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (model, &iter, 1, &val);
g_snprintf(text, sizeof(text), "%s",
g_value_get_string(&val));
gtk_label_set_markup (GTK_LABEL(preflabel), text);
g_value_unset (&val);
gtk_tree_model_get_value (model, &iter, 2, &val);
gtk_notebook_set_current_page (GTK_NOTEBOOK (prefsnotebook), g_value_get_int (&val));
g_value_unset(&val);
}
/* These are the pages in the preferences notebook */
GtkWidget *interface_page() {
GtkWidget *ret;
GtkWidget *vbox;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame(ret, _("Interface Options"));
gaim_gtk_prefs_checkbox(_("D_isplay remote nicknames if no alias is set"),
"/core/buddies/use_server_alias", vbox);
gtk_widget_show_all(ret);
return ret;
}
static void smiley_sel (GtkTreeSelection *sel, GtkTreeModel *model) {
GtkTreeIter iter;
const char *filename;
GValue val = { 0, };
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (model, &iter, 2, &val);
filename = g_value_get_string(&val);
gaim_prefs_set_string("/gaim/gtk/smileys/theme", filename);
g_value_unset (&val);
}
GtkTreePath *theme_refresh_theme_list()
{
GdkPixbuf *pixbuf;
GSList *themes;
GtkTreeIter iter;
GtkTreePath *path = NULL;
int ind = 0;
smiley_theme_probe();
if (!smiley_themes)
return NULL;
themes = smiley_themes;
gtk_list_store_clear(smiley_theme_store);
while (themes) {
struct smiley_theme *theme = themes->data;
char *description = g_strdup_printf("%s - %s\n"
"%s",
theme->name, theme->author, theme->desc);
gtk_list_store_append (smiley_theme_store, &iter);
/*
* LEAK - Gentoo memprof thinks pixbuf is leaking here... but it
* looks like it should be ok to me. Anyone know what's up? --Mark
*/
pixbuf = (theme->icon ? gdk_pixbuf_new_from_file(theme->icon, NULL) : NULL);
gtk_list_store_set(smiley_theme_store, &iter,
0, pixbuf,
1, description,
2, theme->path,
3, theme->name,
-1);
if (pixbuf != NULL)
g_object_unref(G_OBJECT(pixbuf));
g_free(description);
themes = themes->next;
if (current_smiley_theme && !strcmp(theme->path, current_smiley_theme->path)) {
/* path = gtk_tree_path_new_from_indices(ind); */
char *iwishihadgtk2_2 = g_strdup_printf("%d", ind);
path = gtk_tree_path_new_from_string(iwishihadgtk2_2);
g_free(iwishihadgtk2_2);
}
ind++;
}
return path;
}
void theme_install_theme(char *path, char *extn) {
#ifndef _WIN32
gchar *command;
#endif
gchar *destdir;
gchar *tail;
GtkTreePath *themepath = NULL;
/* Just to be safe */
g_strchomp(path);
/* I dont know what you are, get out of here */
if (extn != NULL)
tail = extn;
else if ((tail = strrchr(path, '.')) == NULL)
return;
destdir = g_strconcat(gaim_user_dir(), G_DIR_SEPARATOR_S "smileys", NULL);
/* We'll check this just to make sure. This also lets us do something different on
* other platforms, if need be */
if (!g_ascii_strcasecmp(tail, ".gz") || !g_ascii_strcasecmp(tail, ".tgz")) {
#ifndef _WIN32
command = g_strdup_printf("tar > /dev/null xzf \"%s\" -C %s", path, destdir);
#else
if(!wgaim_gz_untar(path, destdir)) {
g_free(destdir);
return;
}
#endif
}
else {
g_free(destdir);
return;
}
#ifndef _WIN32
/* Fire! */
system(command);
g_free(command);
#endif
g_free(destdir);
themepath = theme_refresh_theme_list();
if (themepath != NULL)
gtk_tree_path_free(themepath);
}
static void
theme_got_url(void *data, const char *themedata, size_t len)
{
FILE *f;
gchar *path;
f = gaim_mkstemp(&path);
fwrite(themedata, len, 1, f);
fclose(f);
theme_install_theme(path, data);
unlink(path);
g_free(path);
}
void theme_dnd_recv(GtkWidget *widget, GdkDragContext *dc, guint x, guint y, GtkSelectionData *sd,
guint info, guint t, gpointer data) {
gchar *name = sd->data;
if ((sd->length >= 0) && (sd->format == 8)) {
/* Well, it looks like the drag event was cool.
* Let's do something with it */
if (!g_ascii_strncasecmp(name, "file://", 7)) {
GError *converr = NULL;
gchar *tmp;
/* It looks like we're dealing with a local file. Let's
* just untar it in the right place */
if(!(tmp = g_filename_from_uri(name, NULL, &converr))) {
gaim_debug(GAIM_DEBUG_ERROR, "theme dnd", "%s\n",
(converr ? converr->message :
"g_filename_from_uri error"));
return;
}
theme_install_theme(tmp, NULL);
g_free(tmp);
} else if (!g_ascii_strncasecmp(name, "http://", 7)) {
/* Oo, a web drag and drop. This is where things
* will start to get interesting */
gchar *tail;
if ((tail = strrchr(name, '.')) == NULL)
return;
/* We'll check this just to make sure. This also lets us do something different on
* other platforms, if need be */
gaim_url_fetch(name, TRUE, NULL, FALSE, theme_got_url, ".tgz");
}
gtk_drag_finish(dc, TRUE, FALSE, t);
}
gtk_drag_finish(dc, FALSE, FALSE, t);
}
/* Does same as normal sort, except "none" is sorted first */
gint gaim_sort_smileys (GtkTreeModel *model,
GtkTreeIter *a,
GtkTreeIter *b,
gpointer userdata)
{
gint ret = 0;
gchar *name1, *name2;
gtk_tree_model_get(model, a, 3, &name1, -1);
gtk_tree_model_get(model, b, 3, &name2, -1);
if (name1 == NULL || name2 == NULL) {
if (!(name1 == NULL && name2 == NULL))
ret = (name1 == NULL) ? -1: 1;
} else if (!g_ascii_strcasecmp(name1, "none")) {
/* Sort name1 first */
ret = -1;
} else if (!g_ascii_strcasecmp(name2, "none")) {
/* Sort name2 first */
ret = 1;
} else {
/* Neither string is "none", default to normal sort */
ret = g_utf8_collate(name1,name2);
}
g_free(name1);
g_free(name2);
return ret;
}
GtkWidget *theme_page() {
GtkWidget *ret;
GtkWidget *sw;
GtkWidget *view;
GtkCellRenderer *rend;
GtkTreeViewColumn *col;
GtkTreeSelection *sel;
GtkTreePath *path = NULL;
GtkWidget *label;
GtkTargetEntry te[3] = {{"text/plain", 0, 0},{"text/uri-list", 0, 1},{"STRING", 0, 2}};
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
label = gtk_label_new(_("Select a smiley theme that you would like to use from the list below. New themes can be installed by dragging and dropping them onto the theme list."));
gtk_label_set_line_wrap(GTK_LABEL(label), TRUE);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_LEFT);
gtk_box_pack_start(GTK_BOX(ret), label, FALSE, TRUE, 0);
gtk_widget_show(label);
sw = gtk_scrolled_window_new(NULL,NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
gtk_box_pack_start(GTK_BOX(ret), sw, TRUE, TRUE, 0);
smiley_theme_store = gtk_list_store_new (4, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
path = theme_refresh_theme_list();
view = gtk_tree_view_new_with_model (GTK_TREE_MODEL(smiley_theme_store));
gtk_drag_dest_set(view, GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP, te,
sizeof(te) / sizeof(GtkTargetEntry) , GDK_ACTION_COPY | GDK_ACTION_MOVE);
g_signal_connect(G_OBJECT(view), "drag_data_received", G_CALLBACK(theme_dnd_recv), smiley_theme_store);
rend = gtk_cell_renderer_pixbuf_new();
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
if(path) {
gtk_tree_selection_select_path(sel, path);
gtk_tree_path_free(path);
}
/* Custom sort so "none" theme is at top of list */
gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(smiley_theme_store),
3, gaim_sort_smileys, NULL, NULL);
gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(smiley_theme_store),
3, GTK_SORT_ASCENDING);
col = gtk_tree_view_column_new_with_attributes (_("Icon"),
rend,
"pixbuf", 0,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(view), col);
rend = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes (_("Description"),
rend,
"markup", 1,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(view), col);
g_object_unref(G_OBJECT(smiley_theme_store));
gtk_container_add(GTK_CONTAINER(sw), view);
g_signal_connect(G_OBJECT(sel), "changed", G_CALLBACK(smiley_sel), NULL);
gtk_widget_show_all(ret);
gaim_set_accessible_label (view, label);
return ret;
}
static void
formatting_reset_cb(GtkWidget *w, GtkWidget* imhtml) {
gboolean bold, italic, uline;
bold = italic = uline = FALSE;
gtk_imhtml_get_current_format(GTK_IMHTML(imhtml), &bold, &italic, &uline);
if (bold)
gtk_imhtml_toggle_bold(GTK_IMHTML(imhtml));
if (italic)
gtk_imhtml_toggle_italic(GTK_IMHTML(imhtml));
if (uline)
gtk_imhtml_toggle_underline(GTK_IMHTML(imhtml));
gtk_imhtml_font_set_size(GTK_IMHTML(imhtml), 3);
gtk_imhtml_toggle_forecolor(GTK_IMHTML(imhtml), NULL);
gtk_imhtml_toggle_backcolor(GTK_IMHTML(imhtml), NULL);
gtk_imhtml_toggle_fontface(GTK_IMHTML(imhtml), NULL);
}
static void
formatting_toggle_cb(GtkIMHtml *imhtml, GtkIMHtmlButtons buttons, void *bah)
{
gboolean bold, italic, uline;
bold = italic = uline = FALSE;
gtk_imhtml_get_current_format(GTK_IMHTML(imhtml),
&bold, &italic, &uline);
if (buttons & GTK_IMHTML_BOLD)
gaim_prefs_set_bool("/gaim/gtk/conversations/send_bold", bold);
if (buttons & GTK_IMHTML_ITALIC)
gaim_prefs_set_bool("/gaim/gtk/conversations/send_italic", italic);
if (buttons & GTK_IMHTML_UNDERLINE)
gaim_prefs_set_bool("/gaim/gtk/conversations/send_underline", uline);
if (buttons & GTK_IMHTML_GROW || buttons & GTK_IMHTML_SHRINK)
gaim_prefs_set_int("/gaim/gtk/conversations/font_size",
gtk_imhtml_get_current_fontsize(GTK_IMHTML(imhtml)));
if (buttons & GTK_IMHTML_FACE) {
char *face = gtk_imhtml_get_current_fontface(GTK_IMHTML(imhtml));
if (!face)
face = g_strdup("");
gaim_prefs_set_string("/gaim/gtk/conversations/font_face", face);
g_free(face);
}
if (buttons & GTK_IMHTML_FORECOLOR) {
char *color = gtk_imhtml_get_current_forecolor(GTK_IMHTML(imhtml));
if (!color)
color = g_strdup("");
gaim_prefs_set_string("/gaim/gtk/conversations/fgcolor", color);
g_free(color);
}
if (buttons & GTK_IMHTML_BACKCOLOR) {
char *color = gtk_imhtml_get_current_backcolor(GTK_IMHTML(imhtml));
if (!color)
color = g_strdup("");
gaim_prefs_set_string("/gaim/gtk/conversations/bgcolor", color);
g_free(color);
}
}
GtkWidget *messages_page() {
GtkWidget *ret;
GtkWidget *vbox, *fontvbox;
GtkWidget *imhtml;
GtkWidget *toolbar;
GtkWidget *sw;
GtkWidget *frame;
GtkWidget *option;
GtkWidget *button;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame (ret, _("Display"));
gaim_gtk_prefs_checkbox(_("Show _timestamp on messages"),
"/gaim/gtk/conversations/show_timestamps", vbox);
#ifdef USE_GTKSPELL
gaim_gtk_prefs_checkbox(_("_Highlight misspelled words"),
"/gaim/gtk/conversations/spellcheck", vbox);
#endif
vbox = gaim_gtk_make_frame (ret, _("Ignore"));
gaim_gtk_prefs_checkbox(_("Ignore c_olors"),
"/gaim/gtk/conversations/ignore_colors", vbox);
gaim_gtk_prefs_checkbox(_("Ignore font _faces"),
"/gaim/gtk/conversations/ignore_fonts", vbox);
gaim_gtk_prefs_checkbox(_("Ignore font si_zes"),
"/gaim/gtk/conversations/ignore_font_sizes", vbox);
vbox = gaim_gtk_make_frame (ret, _("Default Formatting"));
option = gaim_gtk_prefs_checkbox(_("_Send default formatting with outgoing messages"),
"/gaim/gtk/conversations/send_formatting", vbox);
frame = gtk_frame_new(NULL);
gtk_frame_set_shadow_type(GTK_FRAME(frame), GTK_SHADOW_IN);
gtk_box_pack_start(GTK_BOX(vbox), frame, FALSE, FALSE, 0);
fontvbox = gtk_vbox_new(FALSE, 0);
gtk_container_add(GTK_CONTAINER(frame), fontvbox);
toolbar = gtk_imhtmltoolbar_new();
gtk_box_pack_start(GTK_BOX(fontvbox), toolbar, FALSE, FALSE, 0);
sw = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw),
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_NONE);
gtk_box_pack_start(GTK_BOX(fontvbox), sw, TRUE, TRUE, 0);
imhtml = gtk_imhtml_new(NULL, NULL);
gtk_widget_set_name(imhtml, "gaim_gtkprefs_font_imhtml");
gtk_imhtml_set_editable(GTK_IMHTML(imhtml), TRUE);
gtk_imhtml_set_format_functions(GTK_IMHTML(imhtml), GTK_IMHTML_ALL ^ GTK_IMHTML_IMAGE);
gtk_imhtml_set_whole_buffer_formatting_only(GTK_IMHTML(imhtml), TRUE);
gtk_imhtml_smiley_shortcuts(GTK_IMHTML(imhtml),
gaim_prefs_get_bool("/gaim/gtk/conversations/smiley_shortcuts"));
gtk_imhtml_html_shortcuts(GTK_IMHTML(imhtml),
gaim_prefs_get_bool("/gaim/gtk/conversations/html_shortcuts"));
gtk_imhtmltoolbar_attach(GTK_IMHTMLTOOLBAR(toolbar), imhtml);
gtk_imhtmltoolbar_associate_smileys(GTK_IMHTMLTOOLBAR(toolbar), "default");
gaim_setup_imhtml(imhtml);
gtk_imhtml_append_text(GTK_IMHTML(imhtml), _("This is how your outgoing message text will appear when you use protocols that support formatting. :)"), 0);
gtk_container_add(GTK_CONTAINER(sw), imhtml);
button = gtk_button_new_with_mnemonic(_("_Clear Formatting"));
gtk_box_pack_start(GTK_BOX(fontvbox), button, FALSE, FALSE, 0);
if (!gaim_prefs_get_bool("/gaim/gtk/conversations/send_formatting"))
gtk_widget_set_sensitive(fontvbox, FALSE);
if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_bold"))
gtk_imhtml_toggle_bold(GTK_IMHTML(imhtml));
if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_italic"))
gtk_imhtml_toggle_italic(GTK_IMHTML(imhtml));
if (gaim_prefs_get_bool("/gaim/gtk/conversations/send_underline"))
gtk_imhtml_toggle_underline(GTK_IMHTML(imhtml));
gtk_imhtml_font_set_size(GTK_IMHTML(imhtml), gaim_prefs_get_int("/gaim/gtk/conversations/font_size"));
gtk_imhtml_toggle_forecolor(GTK_IMHTML(imhtml), gaim_prefs_get_string("/gaim/gtk/conversations/fgcolor"));
gtk_imhtml_toggle_backcolor(GTK_IMHTML(imhtml), gaim_prefs_get_string("/gaim/gtk/conversations/bgcolor"));
gtk_imhtml_toggle_fontface(GTK_IMHTML(imhtml), gaim_prefs_get_string("/gaim/gtk/conversations/font_face"));
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(formatting_reset_cb), imhtml);
g_signal_connect(G_OBJECT(option), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), fontvbox);
g_signal_connect(G_OBJECT(imhtml), "format_function_toggle",
G_CALLBACK(formatting_toggle_cb), NULL);
gtk_widget_show_all(ret);
return ret;
}
GtkWidget *hotkeys_page() {
GtkWidget *ret;
GtkWidget *vbox;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame(ret, _("Send Message"));
gaim_gtk_prefs_checkbox(_("Enter _sends message"),
"/gaim/gtk/conversations/enter_sends", vbox);
gaim_gtk_prefs_checkbox(_("C_ontrol-Enter sends message"),
"/gaim/gtk/conversations/ctrl_enter_sends", vbox);
vbox = gaim_gtk_make_frame (ret, _("Window Closing"));
gaim_gtk_prefs_checkbox(_("_Escape closes window"),
"/gaim/gtk/conversations/escape_closes", vbox);
vbox = gaim_gtk_make_frame(ret, _("Insertions"));
gaim_gtk_prefs_checkbox(_("Control-{B/I/U} changes _formatting"),
"/gaim/gtk/conversations/html_shortcuts", vbox);
gaim_gtk_prefs_checkbox(_("Control-(number) _inserts smileys"),
"/gaim/gtk/conversations/smiley_shortcuts", vbox);
gtk_widget_show_all(ret);
return ret;
}
GtkWidget *list_page() {
GtkWidget *ret;
GtkWidget *vbox;
GList *l= NULL;
GSList *sl;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame (ret, _("Buddy List Sorting"));
for (sl = gaim_gtk_blist_sort_methods; sl != NULL; sl = sl->next) {
struct gaim_gtk_blist_sort_method *method = sl->data;
l = g_list_append(l, method->name);
l = g_list_append(l, method->id);
}
gaim_gtk_prefs_dropdown_from_list(vbox, _("_Sorting:"), GAIM_PREF_STRING,
"/gaim/gtk/blist/sort_type", l);
g_list_free(l);
vbox = gaim_gtk_make_frame (ret, _("Buddy List Window"));
gaim_gtk_prefs_dropdown(vbox, _("Show _buttons as:"), GAIM_PREF_INT,
"/gaim/gtk/blist/button_style",
_("Pictures"), GAIM_BUTTON_IMAGE,
_("Text"), GAIM_BUTTON_TEXT,
_("Pictures and text"), GAIM_BUTTON_TEXT_IMAGE,
_("None"), GAIM_BUTTON_NONE,
NULL);
gaim_gtk_prefs_checkbox(_("_Raise window on events"),
"/gaim/gtk/blist/raise_on_events", vbox);
vbox = gaim_gtk_make_frame (ret, _("Buddy Display"));
gaim_gtk_prefs_checkbox(_("Show buddy _icons"),
"/gaim/gtk/blist/show_buddy_icons", vbox);
gaim_gtk_prefs_checkbox(_("Show _warning levels"),
"/gaim/gtk/blist/show_warning_level", vbox);
gaim_gtk_prefs_checkbox(_("Show idle _times"),
"/gaim/gtk/blist/show_idle_time", vbox);
gaim_gtk_prefs_checkbox(_("Dim i_dle buddies"),
"/gaim/gtk/blist/grey_idle_buddies", vbox);
gaim_gtk_prefs_checkbox(_("_Automatically expand contacts"),
"/gaim/gtk/blist/auto_expand_contacts", vbox);
gtk_widget_show_all(ret);
return ret;
}
static void
conversation_usetabs_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
gboolean usetabs = (gboolean)value;
if (usetabs)
gtk_widget_set_sensitive(GTK_WIDGET(data), TRUE);
else
gtk_widget_set_sensitive(GTK_WIDGET(data), FALSE);
}
GtkWidget *conv_page() {
GtkWidget *ret;
GtkWidget *vbox, *vbox2;
GtkWidget *label;
GtkSizeGroup *sg;
GList *names = NULL;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width(GTK_CONTAINER(ret), 12);
sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
vbox = gaim_gtk_make_frame(ret, _("Conversations"));
label = gaim_gtk_prefs_dropdown(vbox, _("Show _buttons as:"), GAIM_PREF_INT,
"/gaim/gtk/conversations/button_type",
_("Pictures"), GAIM_BUTTON_IMAGE,
_("Text"), GAIM_BUTTON_TEXT,
_("Pictures and text"), GAIM_BUTTON_TEXT_IMAGE,
_("None"), GAIM_BUTTON_NONE,
NULL);
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gaim_gtk_prefs_checkbox(_("Enable \"_slash\" commands"),
"/gaim/gtk/conversations/enable_commands", vbox);
gaim_gtk_prefs_checkbox(_("Show _formatting toolbar"),
"/gaim/gtk/conversations/show_formatting_toolbar", vbox);
gaim_gtk_prefs_checkbox(_("Show _aliases in tabs/titles"),
"/core/conversations/use_alias_for_title", vbox);
gaim_gtk_prefs_checkbox(_("Show buddy _icons"),
"/gaim/gtk/conversations/im/show_buddy_icons", vbox);
gaim_gtk_prefs_checkbox(_("Enable buddy ic_on animation"),
"/gaim/gtk/conversations/im/animate_buddy_icons", vbox);
gaim_gtk_prefs_checkbox(_("_Notify buddies that you are typing to them"),
"/core/conversations/im/send_typing", vbox);
gaim_gtk_prefs_checkbox(_("_Raise IM window on events"),
"/gaim/gtk/conversations/im/raise_on_events", vbox);
gaim_gtk_prefs_checkbox(_("Raise chat _window on events"),
"/gaim/gtk/conversations/chat/raise_on_events", vbox);
gaim_gtk_prefs_checkbox(_("Use _multi-colored screen names in chats"),
"/gaim/gtk/conversations/chat/color_nicks", vbox);
/* All the tab options! */
vbox = gaim_gtk_make_frame(ret, _("Tab Options"));
gaim_gtk_prefs_checkbox(_("Show IMs and chats in _tabbed windows"),
"/gaim/gtk/conversations/tabs", vbox);
/*
* Connect a signal to the above preference. When conversations are not
* shown in a tabbed window then all tabbing options should be disabled.
*/
vbox2 = gtk_vbox_new(FALSE, 0);
gtk_box_pack_start(GTK_BOX(vbox), vbox2, FALSE, FALSE, 0);
placement_pref_id = gaim_prefs_connect_callback("/gaim/gtk/conversations/tabs",
conversation_usetabs_cb,
vbox2);
if (!gaim_prefs_get_bool("/gaim/gtk/conversations/tabs"))
gtk_widget_set_sensitive(vbox2, FALSE);
gaim_gtk_prefs_checkbox(_("Show _close button on tabs"),
"/gaim/gtk/conversations/close_on_tabs", vbox2);
label = gaim_gtk_prefs_dropdown(vbox2, _("Tab p_lacement:"), GAIM_PREF_INT,
"/gaim/gtk/conversations/tab_side",
_("Top"), GTK_POS_TOP,
_("Bottom"), GTK_POS_BOTTOM,
_("Left"), GTK_POS_LEFT,
_("Right"), GTK_POS_RIGHT,
NULL);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_size_group_add_widget(sg, label);
names = gaim_conv_placement_get_options();
label = gaim_gtk_prefs_dropdown_from_list(vbox2, _("New conversation _placement:"),
GAIM_PREF_STRING, "/gaim/gtk/conversations/placement", names);
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
g_list_free(names);
gtk_widget_show_all(ret);
return ret;
}
static void network_ip_changed(GtkEntry *entry, gpointer data)
{
gaim_network_set_public_ip(gtk_entry_get_text(entry));
}
static void
proxy_changed_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
GtkWidget *frame = data;
const char *proxy = value;
if (strcmp(proxy, "none") && strcmp(proxy, "envvar"))
gtk_widget_set_sensitive(frame, TRUE);
else
gtk_widget_set_sensitive(frame, FALSE);
}
static void proxy_print_option(GtkEntry *entry, int entrynum)
{
if (entrynum == PROXYHOST)
gaim_prefs_set_string("/core/proxy/host", gtk_entry_get_text(entry));
else if (entrynum == PROXYPORT)
gaim_prefs_set_int("/core/proxy/port", atoi(gtk_entry_get_text(entry)));
else if (entrynum == PROXYUSER)
gaim_prefs_set_string("/core/proxy/username", gtk_entry_get_text(entry));
else if (entrynum == PROXYPASS)
gaim_prefs_set_string("/core/proxy/password", gtk_entry_get_text(entry));
}
GtkWidget *network_page() {
GtkWidget *ret;
GtkWidget *vbox, *hbox, *entry;
GtkWidget *table, *label, *auto_ip_checkbox, *ports_checkbox, *spin_button;
GtkSizeGroup *sg;
GaimProxyInfo *proxy_info;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame (ret, _("IP Address"));
auto_ip_checkbox = gaim_gtk_prefs_checkbox(_("_Autodetect IP Address"),
"/core/network/auto_ip", vbox);
table = gtk_table_new(2, 1, FALSE);
gtk_container_set_border_width(GTK_CONTAINER(table), 5);
gtk_table_set_col_spacings(GTK_TABLE(table), 5);
gtk_table_set_row_spacings(GTK_TABLE(table), 10);
gtk_container_add(GTK_CONTAINER(vbox), table);
label = gtk_label_new_with_mnemonic(_("Public _IP:"));
gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 0, 1, GTK_FILL, 0, 0, 0);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_table_attach(GTK_TABLE(table), entry, 1, 2, 0, 1, GTK_FILL, 0, 0, 0);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(network_ip_changed), NULL);
if (gaim_network_get_public_ip() != NULL)
gtk_entry_set_text(GTK_ENTRY(entry),
gaim_network_get_public_ip());
gaim_set_accessible_label (entry, label);
if (gaim_prefs_get_bool("/core/network/auto_ip")) {
gtk_widget_set_sensitive(GTK_WIDGET(table), FALSE);
}
g_signal_connect(G_OBJECT(auto_ip_checkbox), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), table);
vbox = gaim_gtk_make_frame (ret, _("Ports"));
sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
ports_checkbox = gaim_gtk_prefs_checkbox(_("_Manually specify range of ports to listen on"),
"/core/network/ports_range_use", vbox);
spin_button = gaim_gtk_prefs_labeled_spin_button(vbox, _("_Start Port:"),
"/core/network/ports_range_start", 0, 65535, sg);
if (!gaim_prefs_get_bool("/core/network/ports_range_use"))
gtk_widget_set_sensitive(GTK_WIDGET(spin_button), FALSE);
g_signal_connect(G_OBJECT(ports_checkbox), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), spin_button);
spin_button = gaim_gtk_prefs_labeled_spin_button(vbox, _("_End Port:"),
"/core/network/ports_range_end", 0, 65535, sg);
if (!gaim_prefs_get_bool("/core/network/ports_range_use"))
gtk_widget_set_sensitive(GTK_WIDGET(spin_button), FALSE);
g_signal_connect(G_OBJECT(ports_checkbox), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), spin_button);
vbox = gaim_gtk_make_frame(ret, _("Proxy Server"));
prefs_proxy_frame = gtk_vbox_new(FALSE, 0);
gaim_gtk_prefs_dropdown(vbox, _("Proxy _type:"), GAIM_PREF_STRING,
"/core/proxy/type",
_("No proxy"), "none",
"SOCKS 4", "socks4",
"SOCKS 5", "socks5",
"HTTP", "http",
_("Use Environmental Settings"), "envvar",
NULL);
gtk_box_pack_start(GTK_BOX(vbox), prefs_proxy_frame, 0, 0, 0);
proxy_info = gaim_global_proxy_get_info();
if (proxy_info == NULL ||
gaim_proxy_info_get_type(proxy_info) == GAIM_PROXY_NONE ||
gaim_proxy_info_get_type(proxy_info) == GAIM_PROXY_USE_ENVVAR) {
gtk_widget_set_sensitive(GTK_WIDGET(prefs_proxy_frame), FALSE);
}
proxy_pref_id = gaim_prefs_connect_callback("/core/proxy/type",
proxy_changed_cb, prefs_proxy_frame);
table = gtk_table_new(4, 2, FALSE);
gtk_container_set_border_width(GTK_CONTAINER(table), 5);
gtk_table_set_col_spacings(GTK_TABLE(table), 5);
gtk_table_set_row_spacings(GTK_TABLE(table), 10);
gtk_container_add(GTK_CONTAINER(prefs_proxy_frame), table);
label = gtk_label_new_with_mnemonic(_("_Host:"));
gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 0, 1, GTK_FILL, 0, 0, 0);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_table_attach(GTK_TABLE(table), entry, 1, 2, 0, 1, GTK_FILL, 0, 0, 0);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(proxy_print_option), (void *)PROXYHOST);
if (proxy_info != NULL && gaim_proxy_info_get_host(proxy_info))
gtk_entry_set_text(GTK_ENTRY(entry),
gaim_proxy_info_get_host(proxy_info));
hbox = gtk_hbox_new(TRUE, 5);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
gaim_set_accessible_label (entry, label);
label = gtk_label_new_with_mnemonic(_("_Port:"));
gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 2, 3, 0, 1, GTK_FILL, 0, 0, 0);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_table_attach(GTK_TABLE(table), entry, 3, 4, 0, 1, GTK_FILL, 0, 0, 0);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(proxy_print_option), (void *)PROXYPORT);
if (proxy_info != NULL && gaim_proxy_info_get_port(proxy_info) != 0) {
char buf[128];
g_snprintf(buf, sizeof(buf), "%d",
gaim_proxy_info_get_port(proxy_info));
gtk_entry_set_text(GTK_ENTRY(entry), buf);
}
gaim_set_accessible_label (entry, label);
label = gtk_label_new_with_mnemonic(_("_User:"));
gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 0, 1, 1, 2, GTK_FILL, 0, 0, 0);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_table_attach(GTK_TABLE(table), entry, 1, 2, 1, 2, GTK_FILL, 0, 0, 0);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(proxy_print_option), (void *)PROXYUSER);
if (proxy_info != NULL && gaim_proxy_info_get_username(proxy_info) != NULL)
gtk_entry_set_text(GTK_ENTRY(entry),
gaim_proxy_info_get_username(proxy_info));
hbox = gtk_hbox_new(TRUE, 5);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
gaim_set_accessible_label (entry, label);
label = gtk_label_new_with_mnemonic(_("Pa_ssword:"));
gtk_misc_set_alignment(GTK_MISC(label), 1.0, 0.5);
gtk_table_attach(GTK_TABLE(table), label, 2, 3, 1, 2, GTK_FILL, 0, 0, 0);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_table_attach(GTK_TABLE(table), entry, 3, 4, 1, 2, GTK_FILL , 0, 0, 0);
gtk_entry_set_visibility(GTK_ENTRY(entry), FALSE);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(proxy_print_option), (void *)PROXYPASS);
if (proxy_info != NULL && gaim_proxy_info_get_password(proxy_info) != NULL)
gtk_entry_set_text(GTK_ENTRY(entry),
gaim_proxy_info_get_password(proxy_info));
gaim_set_accessible_label (entry, label);
gtk_widget_show_all(ret);
return ret;
}
#ifndef _WIN32
static gboolean manual_browser_set(GtkWidget *entry, GdkEventFocus *event, gpointer data) {
const char *program = gtk_entry_get_text(GTK_ENTRY(entry));
gaim_prefs_set_string("/gaim/gtk/browsers/command", program);
/* carry on normally */
return FALSE;
}
static GList *get_available_browsers()
{
struct browser {
char *name;
char *command;
};
static struct browser possible_browsers[] = {
{N_("Epiphany"), "epiphany"},
{N_("Firebird"), "mozilla-firebird"},
{N_("Firefox"), "firefox"},
{N_("Galeon"), "galeon"},
{N_("Gnome Default"), "gnome-open"},
{N_("Konqueror"), "kfmclient"},
{N_("Mozilla"), "mozilla"},
{N_("Netscape"), "netscape"},
{N_("Opera"), "opera"}
};
static const int num_possible_browsers = 9;
GList *browsers = NULL;
int i = 0;
char *browser_setting = (char *)gaim_prefs_get_string("/gaim/gtk/browsers/browser");
browsers = g_list_prepend(browsers, "custom");
browsers = g_list_prepend(browsers, _("Manual"));
for (i = 0; i < num_possible_browsers; i++) {
if (gaim_program_is_valid(possible_browsers[i].command)) {
browsers = g_list_prepend(browsers,
possible_browsers[i].command);
browsers = g_list_prepend(browsers, _(possible_browsers[i].name));
if(browser_setting && !strcmp(possible_browsers[i].command, browser_setting))
browser_setting = NULL;
}
}
if(browser_setting)
gaim_prefs_set_string("/gaim/gtk/browsers/browser", "custom");
return browsers;
}
static void
browser_changed1_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
GtkWidget *hbox = data;
const char *browser = value;
gtk_widget_set_sensitive(hbox, strcmp(browser, "custom"));
}
static void
browser_changed2_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
GtkWidget *hbox = data;
const char *browser = value;
gtk_widget_set_sensitive(hbox, !strcmp(browser, "custom"));
}
GtkWidget *browser_page() {
GtkWidget *ret;
GtkWidget *vbox;
GtkWidget *hbox;
GtkWidget *label;
GtkWidget *entry;
GtkSizeGroup *sg;
GList *browsers = NULL;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
vbox = gaim_gtk_make_frame (ret, _("Browser Selection"));
browsers = get_available_browsers();
if (browsers != NULL) {
label = gaim_gtk_prefs_dropdown_from_list(vbox,_("_Browser:"), GAIM_PREF_STRING,
"/gaim/gtk/browsers/browser",
browsers);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_size_group_add_widget(sg, label);
hbox = gtk_hbox_new(FALSE, 0);
label = gaim_gtk_prefs_dropdown(hbox, _("_Open link in:"), GAIM_PREF_INT,
"/gaim/gtk/browsers/place",
_("Browser default"), GAIM_BROWSER_DEFAULT,
_("Existing window"), GAIM_BROWSER_CURRENT,
_("New window"), GAIM_BROWSER_NEW_WINDOW,
_("New tab"), GAIM_BROWSER_NEW_TAB,
NULL);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_size_group_add_widget(sg, label);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
if (!strcmp(gaim_prefs_get_string("/gaim/gtk/browsers/browser"), "custom"))
gtk_widget_set_sensitive(hbox, FALSE);
browser_pref1_id = gaim_prefs_connect_callback("/gaim/gtk/browsers/browser",
browser_changed1_cb, hbox);
}
hbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
label = gtk_label_new_with_mnemonic(_("_Manual:\n(%s for URL)"));
gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_size_group_add_widget(sg, label);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
if (strcmp(gaim_prefs_get_string("/gaim/gtk/browsers/browser"), "custom"))
gtk_widget_set_sensitive(hbox, FALSE);
browser_pref2_id = gaim_prefs_connect_callback("/gaim/gtk/browsers/browser",
browser_changed2_cb, hbox);
gtk_box_pack_start (GTK_BOX (hbox), entry, FALSE, FALSE, 0);
gtk_entry_set_text(GTK_ENTRY(entry),
gaim_prefs_get_string("/gaim/gtk/browsers/command"));
g_signal_connect(G_OBJECT(entry), "focus-out-event",
G_CALLBACK(manual_browser_set), NULL);
gaim_set_accessible_label (entry, label);
gtk_widget_show_all(ret);
return ret;
}
#endif /*_WIN32*/
GtkWidget *logging_page() {
GtkWidget *ret;
GtkWidget *vbox;
GList *names;
GtkWidget *sys_box;
GtkWidget *box;
int syslog_enabled = gaim_prefs_get_bool("/core/logging/log_system");
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
vbox = gaim_gtk_make_frame (ret, _("Message Logs"));
names = gaim_log_logger_get_options();
gaim_gtk_prefs_dropdown_from_list(vbox, _("Log _Format:"), GAIM_PREF_STRING,
"/core/logging/format", names);
gaim_gtk_prefs_checkbox(_("_Log all instant messages"),
"/core/logging/log_ims", vbox);
gaim_gtk_prefs_checkbox(_("Log all c_hats"),
"/core/logging/log_chats", vbox);
vbox = gaim_gtk_make_frame (ret, _("System Logs"));
sys_box = gaim_gtk_prefs_checkbox(_("_Enable system log"),
"/core/logging/log_system", vbox);
box = gaim_gtk_prefs_checkbox(_("Log when buddies _sign on/sign off"),
"/core/logging/log_signon_signoff", vbox);
g_signal_connect(G_OBJECT(sys_box), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), box);
gtk_widget_set_sensitive(box, syslog_enabled);
box = gaim_gtk_prefs_checkbox(_("Log when buddies become _idle/un-idle"),
"/core/logging/log_idle_state", vbox);
g_signal_connect(G_OBJECT(sys_box), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), box);
gtk_widget_set_sensitive(box, syslog_enabled);
box = gaim_gtk_prefs_checkbox(_("Log when buddies go away/come _back"),
"/core/logging/log_away_state", vbox);
g_signal_connect(G_OBJECT(sys_box), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), box);
gtk_widget_set_sensitive(box, syslog_enabled);
box = gaim_gtk_prefs_checkbox(_("Log your _own signons/idleness/awayness"),
"/core/logging/log_own_states", vbox);
g_signal_connect(G_OBJECT(sys_box), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), box);
gtk_widget_set_sensitive(box, syslog_enabled);
gtk_widget_show_all(ret);
return ret;
}
#ifndef _WIN32
static gint sound_cmd_yeah(GtkEntry *entry, gpointer d)
{
gaim_prefs_set_string("/gaim/gtk/sound/command",
gtk_entry_get_text(GTK_ENTRY(entry)));
return TRUE;
}
static void
sound_changed_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
GtkWidget *hbox = data;
const char *method = value;
gtk_widget_set_sensitive(hbox, !strcmp(method, "custom"));
}
#endif
static void
event_toggled(GtkCellRendererToggle *cell, gchar *pth, gpointer data)
{
GtkTreeModel *model = (GtkTreeModel *)data;
GtkTreeIter iter;
GtkTreePath *path = gtk_tree_path_new_from_string(pth);
const char *pref;
gtk_tree_model_get_iter (model, &iter, path);
gtk_tree_model_get (model, &iter,
2, &pref,
-1);
gaim_prefs_set_bool(pref, !gtk_cell_renderer_toggle_get_active(cell));
gtk_list_store_set(GTK_LIST_STORE (model), &iter,
0, !gtk_cell_renderer_toggle_get_active(cell),
-1);
gtk_tree_path_free(path);
}
static void
test_sound(GtkWidget *button, gpointer i_am_NULL)
{
char *pref;
gboolean temp_value1, temp_value2;
pref = g_strdup_printf("/gaim/gtk/sound/enabled/%s",
gaim_gtk_sound_get_event_option(sound_row_sel));
temp_value1 = gaim_prefs_get_bool("/core/sound/while_away");
temp_value2 = gaim_prefs_get_bool(pref);
if (!temp_value1) gaim_prefs_set_bool("/core/sound/while_away", TRUE);
if (!temp_value2) gaim_prefs_set_bool(pref, TRUE);
gaim_sound_play_event(sound_row_sel);
if (!temp_value1) gaim_prefs_set_bool("/core/sound/while_away", FALSE);
if (!temp_value2) gaim_prefs_set_bool(pref, FALSE);
g_free(pref);
}
/*
* Resets a sound file back to default.
*/
static void
reset_sound(GtkWidget *button, gpointer i_am_also_NULL)
{
gchar *pref;
pref = g_strdup_printf("/gaim/gtk/sound/file/%s",
gaim_gtk_sound_get_event_option(sound_row_sel));
gaim_prefs_set_string(pref, "");
g_free(pref);
gtk_entry_set_text(GTK_ENTRY(sound_entry), "(default)");
}
static void
sound_chosen_cb(void *user_data, const char *filename)
{
gchar *pref;
int sound;
sound = GPOINTER_TO_INT(user_data);
/* Set it -- and forget it */
pref = g_strdup_printf("/gaim/gtk/sound/file/%s",
gaim_gtk_sound_get_event_option(sound));
gaim_prefs_set_string(pref, filename);
g_free(pref);
/*
* If the sound we just changed is still the currently selected
* sound, then update the box showing the file name.
*/
if (sound == sound_row_sel)
gtk_entry_set_text(GTK_ENTRY(sound_entry), filename);
}
static void select_sound(GtkWidget *button, gpointer being_NULL_is_fun)
{
gchar *pref;
const char *filename;
pref = g_strdup_printf("/gaim/gtk/sound/file/%s",
gaim_gtk_sound_get_event_option(sound_row_sel));
filename = gaim_prefs_get_string(pref);
g_free(pref);
if (*filename == '\0')
filename = NULL;
gaim_request_file(prefs, _("Sound Selection"), filename, FALSE,
G_CALLBACK(sound_chosen_cb), NULL, GINT_TO_POINTER(sound_row_sel));
}
static void prefs_sound_sel(GtkTreeSelection *sel, GtkTreeModel *model) {
GtkTreeIter iter;
GValue val = { 0, };
const char *file;
char *pref;
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (model, &iter, 3, &val);
sound_row_sel = g_value_get_uint(&val);
pref = g_strdup_printf("/gaim/gtk/sound/file/%s",
gaim_gtk_sound_get_event_option(sound_row_sel));
file = gaim_prefs_get_string(pref);
g_free(pref);
if (sound_entry)
gtk_entry_set_text(GTK_ENTRY(sound_entry), (file && *file != '\0') ? file : "(default)");
g_value_unset (&val);
}
GtkWidget *sound_page() {
GtkWidget *ret;
GtkWidget *vbox, *sw, *button;
GtkSizeGroup *sg;
GtkTreeIter iter;
GtkWidget *event_view;
GtkListStore *event_store;
GtkCellRenderer *rend;
GtkTreeViewColumn *col;
GtkTreeSelection *sel;
GtkTreePath *path;
GtkWidget *hbox;
int j;
const char *file;
char *pref;
#ifndef _WIN32
GtkWidget *dd;
GtkWidget *label;
GtkWidget *entry;
const char *cmd;
#endif
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
vbox = gaim_gtk_make_frame (ret, _("Sound Options"));
gaim_gtk_prefs_checkbox(_("Sounds when conversation has _focus"),
"/gaim/gtk/sound/conv_focus", vbox);
gaim_gtk_prefs_checkbox(_("_Sounds while away"),
"/core/sound/while_away", vbox);
#ifndef _WIN32
vbox = gaim_gtk_make_frame (ret, _("Sound Method"));
dd = gaim_gtk_prefs_dropdown(vbox, _("_Method:"), GAIM_PREF_STRING,
"/gaim/gtk/sound/method",
_("Console beep"), "beep",
#ifdef USE_AO
_("Automatic"), "automatic",
"ESD", "esd",
"Arts", "arts",
#endif
#ifdef USE_NAS_AUDIO
"NAS", "nas",
#endif
_("Command"), "custom",
NULL);
gtk_size_group_add_widget(sg, dd);
gtk_misc_set_alignment(GTK_MISC(dd), 0, 0);
hbox = gtk_hbox_new(FALSE, 5);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 5);
label = gtk_label_new_with_mnemonic(_("Sound c_ommand:\n(%s for filename)"));
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 5);
entry = gtk_entry_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), entry);
gtk_editable_set_editable(GTK_EDITABLE(entry), TRUE);
cmd = gaim_prefs_get_string("/gaim/gtk/sound/command");
if(cmd)
gtk_entry_set_text(GTK_ENTRY(entry), cmd);
gtk_widget_set_size_request(entry, 75, -1);
gtk_box_pack_start(GTK_BOX(hbox), entry, TRUE, TRUE, 5);
g_signal_connect(G_OBJECT(entry), "changed",
G_CALLBACK(sound_cmd_yeah), NULL);
gtk_widget_set_sensitive(hbox,
!strcmp(gaim_prefs_get_string("/gaim/gtk/sound/method"),
"custom"));
sound_pref_id = gaim_prefs_connect_callback("/gaim/gtk/sound/method",
sound_changed_cb, hbox);
gaim_set_accessible_label (entry, label);
#endif /* _WIN32 */
vbox = gaim_gtk_make_frame(ret, _("Sound Events"));
/* The following is an ugly hack to make the frame expand so the
* sound events list is big enough to be usable */
gtk_box_set_child_packing(GTK_BOX(vbox->parent), vbox, TRUE, TRUE, 0,
GTK_PACK_START);
gtk_box_set_child_packing(GTK_BOX(vbox->parent->parent), vbox->parent, TRUE,
TRUE, 0, GTK_PACK_START);
gtk_box_set_child_packing(GTK_BOX(vbox->parent->parent->parent),
vbox->parent->parent, TRUE, TRUE, 0, GTK_PACK_START);
sw = gtk_scrolled_window_new(NULL,NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
gtk_box_pack_start(GTK_BOX(vbox), sw, TRUE, TRUE, 0);
event_store = gtk_list_store_new (4, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_UINT);
for (j=0; j < GAIM_NUM_SOUNDS; j++) {
char *pref = g_strdup_printf("/gaim/gtk/sound/enabled/%s",
gaim_gtk_sound_get_event_option(j));
const char *label = gaim_gtk_sound_get_event_label(j);
if (label == NULL) {
g_free(pref);
continue;
}
gtk_list_store_append (event_store, &iter);
gtk_list_store_set(event_store, &iter,
0, gaim_prefs_get_bool(pref),
1, _(label),
2, pref,
3, j,
-1);
g_free(pref);
}
event_view = gtk_tree_view_new_with_model (GTK_TREE_MODEL(event_store));
rend = gtk_cell_renderer_toggle_new();
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (event_view));
g_signal_connect (G_OBJECT (sel), "changed",
G_CALLBACK (prefs_sound_sel),
NULL);
g_signal_connect (G_OBJECT(rend), "toggled",
G_CALLBACK(event_toggled), event_store);
path = gtk_tree_path_new_first();
gtk_tree_selection_select_path(sel, path);
gtk_tree_path_free(path);
col = gtk_tree_view_column_new_with_attributes (_("Play"),
rend,
"active", 0,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
rend = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes (_("Event"),
rend,
"text", 1,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
g_object_unref(G_OBJECT(event_store));
gtk_container_add(GTK_CONTAINER(sw), event_view);
hbox = gtk_hbox_new(FALSE, 6);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
sound_entry = gtk_entry_new();
pref = g_strdup_printf("/gaim/gtk/sound/file/%s",
gaim_gtk_sound_get_event_option(0));
file = gaim_prefs_get_string(pref);
g_free(pref);
gtk_entry_set_text(GTK_ENTRY(sound_entry), (file && *file != '\0') ? file : "(default)");
gtk_editable_set_editable(GTK_EDITABLE(sound_entry), FALSE);
gtk_box_pack_start(GTK_BOX(hbox), sound_entry, FALSE, FALSE, 5);
button = gtk_button_new_with_label(_("Test"));
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(test_sound), NULL);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 1);
button = gtk_button_new_with_label(_("Reset"));
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(reset_sound), NULL);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 1);
button = gtk_button_new_with_label(_("Choose..."));
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(select_sound), NULL);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 1);
gtk_widget_show_all(ret);
return ret;
}
GtkWidget *away_page() {
GtkWidget *ret;
GtkWidget *vbox;
GtkWidget *hbox;
GtkWidget *label;
GtkWidget *button;
GtkWidget *select;
GtkWidget *dd;
GtkSizeGroup *sg;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
sg = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
vbox = gaim_gtk_make_frame (ret, _("Away"));
gaim_gtk_prefs_checkbox(_("_Queue new messages when away"),
"/gaim/gtk/away/queue_messages", vbox);
label = gaim_gtk_prefs_dropdown(vbox, _("_Auto-reply:"),
GAIM_PREF_STRING, "/core/away/auto_reply",
_("Never"), "never",
_("Only when away"), "away",
_("Only when away and idle"), "awayidle",
NULL);
vbox = gaim_gtk_make_frame (ret, _("Idle"));
dd = gaim_gtk_prefs_dropdown(vbox, _("Idle _time reporting:"),
GAIM_PREF_STRING, "/gaim/gtk/idle/reporting_method",
_("None"), "none",
_("Gaim usage"), "gaim",
#ifdef USE_SCREENSAVER
#ifndef _WIN32
_("X usage"), "system",
#else
_("Windows usage"), "system",
#endif
#endif
NULL);
gtk_size_group_add_widget(sg, dd);
gtk_misc_set_alignment(GTK_MISC(dd), 0, 0);
vbox = gaim_gtk_make_frame (ret, _("Auto-away"));
button = gaim_gtk_prefs_checkbox(_("Set away _when idle"),
"/core/away/away_when_idle", vbox);
select = gaim_gtk_prefs_labeled_spin_button(vbox,
_("_Minutes before setting away:"), "/core/away/mins_before_away",
1, 24 * 60, sg);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), select);
hbox = gtk_hbox_new(FALSE, 0);
gtk_container_add(GTK_CONTAINER(vbox), hbox);
label = gtk_label_new_with_mnemonic(_("Away m_essage:"));
gtk_size_group_add_widget(sg, label);
gtk_misc_set_alignment(GTK_MISC(label), 0, 0);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), label);
gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
prefs_away_menu = gtk_option_menu_new();
gtk_label_set_mnemonic_widget(GTK_LABEL(label), prefs_away_menu);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(gaim_gtk_toggle_sensitive), prefs_away_menu);
default_away_menu_init(prefs_away_menu);
gtk_widget_show(prefs_away_menu);
gtk_box_pack_start(GTK_BOX(hbox), prefs_away_menu, FALSE, FALSE, 0);
gaim_set_accessible_label (prefs_away_menu, label);
if (!gaim_prefs_get_bool("/core/away/away_when_idle")) {
gtk_widget_set_sensitive(GTK_WIDGET(select), FALSE);
gtk_widget_set_sensitive(GTK_WIDGET(label), FALSE);
gtk_widget_set_sensitive(GTK_WIDGET(prefs_away_menu), FALSE);
}
gtk_widget_show_all(ret);
return ret;
}
static GtkWidget *plugin_description=NULL, *plugin_details=NULL;
static void prefs_plugin_sel (GtkTreeSelection *sel, GtkTreeModel *model)
{
gchar *buf, *pname, *perr, *pdesc, *pauth, *pweb;
GtkTreeIter iter;
GValue val = { 0, };
GaimPlugin *plug;
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (model, &iter, 3, &val);
plug = g_value_get_pointer(&val);
pname = g_markup_escape_text(_(plug->info->name), -1);
pdesc = (plug->info->description) ?
g_markup_escape_text(_(plug->info->description), -1) : NULL;
pauth = (plug->info->author) ?
g_markup_escape_text(_(plug->info->author), -1) : NULL;
pweb = (plug->info->homepage) ?
g_markup_escape_text(_(plug->info->homepage), -1) : NULL;
if (plug->error != NULL) {
perr = g_markup_escape_text(_(plug->error), -1);
buf = g_strdup_printf(
"%s %s\n\n"
"%s\n\n"
"%s",
pname, plug->info->version, perr, pdesc);
g_free(perr);
}
else {
buf = g_strdup_printf(
"%s %s\n\n%s",
pname, plug->info->version, pdesc);
}
gtk_label_set_markup(GTK_LABEL(plugin_description), buf);
g_free(buf);
buf = g_strdup_printf(
#ifndef _WIN32
_("%s %s\n\n"
"Written by:\t%s\n"
"Web site:\t\t%s\n"
"File name:\t%s"),
#else
_("%s %s\n\n"
"Written by: %s\n"
"URL: %s\n"
"File name: %s"),
#endif
pname, plug->info->version, pauth, pweb, plug->path);
gtk_label_set_markup(GTK_LABEL(plugin_details), buf);
g_value_unset(&val);
g_free(buf);
g_free(pname);
g_free(pdesc);
g_free(pauth);
g_free(pweb);
}
static void plugin_load (GtkCellRendererToggle *cell, gchar *pth, gpointer data)
{
GtkTreeModel *model = (GtkTreeModel *)data;
GtkTreeIter iter;
GtkTreePath *path = gtk_tree_path_new_from_string(pth);
GaimPlugin *plug;
gchar buf[1024];
gchar *name = NULL, *description = NULL;
GdkCursor *wait = gdk_cursor_new (GDK_WATCH);
gdk_window_set_cursor(prefs->window, wait);
gdk_cursor_unref(wait);
gtk_tree_model_get_iter (model, &iter, path);
gtk_tree_model_get (model, &iter, 3, &plug, -1);
if (!gaim_plugin_is_loaded(plug)) {
gaim_plugin_load(plug);
/*
* NOTE: This is basically the same check as before
* (plug->type == plugin), but now there aren't plugin types.
* Not yet, anyway. I want to do a V2 of the plugin API.
* The thing is, we should have a flag specifying the UI type,
* or just whether it's a general plugin or a UI-specific
* plugin. We should only load this if it's UI-specific.
*
* -- ChipX86
*/
if (GAIM_IS_GTK_PLUGIN(plug))
{
GtkWidget *config_frame;
GaimGtkPluginUiInfo *ui_info;
ui_info = GAIM_GTK_PLUGIN_UI_INFO(plug);
config_frame = gaim_gtk_plugin_get_config_frame(plug);
if (config_frame != NULL) {
ui_info->iter = g_new0(GtkTreeIter, 1);
prefs_notebook_add_page(_(plug->info->name), NULL,
config_frame, ui_info->iter,
&plugin_iter, notebook_page++);
if (gtk_tree_model_iter_n_children(GTK_TREE_MODEL(prefstree),
&plugin_iter) == 1) {
/* Expand the tree for the first plugin added */
GtkTreePath *path2;
path2 = gtk_tree_model_get_path(GTK_TREE_MODEL(prefstree),
&plugin_iter);
gtk_tree_view_expand_row(GTK_TREE_VIEW(tree_v),
path2, TRUE);
gtk_tree_path_free(path2);
}
}
}
if(GAIM_PLUGIN_HAS_PREF_FRAME(plug)) {
GtkTreeIter iter;
GtkWidget *pref_frame;
GaimPluginUiInfo *prefs_info;
if(plug->info->type == GAIM_PLUGIN_PROTOCOL)
iter = proto_iter;
else
iter = plugin_iter;
prefs_info = GAIM_PLUGIN_UI_INFO(plug);
prefs_info->frame = prefs_info->get_plugin_pref_frame(plug);
pref_frame = gaim_gtk_plugin_pref_create_frame(prefs_info->frame);
if(pref_frame != NULL) {
prefs_info->iter = g_new0(GtkTreeIter, 1);
prefs_notebook_add_page(_(plug->info->name), NULL,
pref_frame, prefs_info->iter,
&iter, notebook_page++);
if(gtk_tree_model_iter_n_children(GTK_TREE_MODEL(prefstree), &iter) == 1)
{
GtkTreePath *path2;
path2 = gtk_tree_model_get_path(GTK_TREE_MODEL(prefstree), &iter);
gtk_tree_view_expand_row(GTK_TREE_VIEW(tree_v), path2, TRUE);
gtk_tree_path_free(path2);
}
}
}
}
else {
if (GAIM_IS_GTK_PLUGIN(plug)) {
GaimGtkPluginUiInfo *ui_info;
ui_info = GAIM_GTK_PLUGIN_UI_INFO(plug);
if (ui_info != NULL && ui_info->iter != NULL) {
gtk_tree_store_remove(GTK_TREE_STORE(prefstree), ui_info->iter);
g_free(ui_info->iter);
ui_info->iter = NULL;
}
}
if (GAIM_PLUGIN_HAS_PREF_FRAME(plug)) {
GaimPluginUiInfo *prefs_info;
prefs_info = GAIM_PLUGIN_UI_INFO(plug);
if(prefs_info != NULL) {
if(prefs_info->frame != NULL) {
gaim_plugin_pref_frame_destroy(prefs_info->frame);
prefs_info->frame = NULL;
}
if(prefs_info->iter != NULL) {
gtk_tree_store_remove(GTK_TREE_STORE(prefstree), prefs_info->iter);
g_free(prefs_info->iter);
prefs_info->iter = NULL;
}
}
}
gaim_plugin_unload(plug);
}
gdk_window_set_cursor(prefs->window, NULL);
name = g_markup_escape_text(_(plug->info->name), -1);
description = g_markup_escape_text(_(plug->info->description), -1);
if (plug->error != NULL) {
gchar *error = g_markup_escape_text(plug->error, -1);
g_snprintf(buf, sizeof(buf),
"%s %s\n\n"
"%s\n\n"
"%s",
name, plug->info->version, error, description);
g_free(error);
} else {
g_snprintf(buf, sizeof(buf),
"%s %s\n\n%s",
name, plug->info->version, description);
}
g_free(name);
g_free(description);
gtk_label_set_markup(GTK_LABEL(plugin_description), buf);
gtk_list_store_set (GTK_LIST_STORE (model), &iter, 0,
gaim_plugin_is_loaded(plug), -1);
gtk_label_set_markup(GTK_LABEL(plugin_description), buf);
gtk_tree_path_free(path);
gaim_gtk_plugins_save();
}
static void
update_plugin_list(void *data)
{
GtkListStore *ls = GTK_LIST_STORE(data);
GtkTreeIter iter;
GList *probes;
GaimPlugin *plug;
gtk_list_store_clear(ls);
for (probes = gaim_plugins_get_all();
probes != NULL;
probes = probes->next)
{
plug = probes->data;
if (plug->info->type != GAIM_PLUGIN_STANDARD ||
(plug->info->flags & GAIM_PLUGIN_FLAG_INVISIBLE))
{
continue;
}
gtk_list_store_append (ls, &iter);
gtk_list_store_set(ls, &iter,
0, gaim_plugin_is_loaded(plug),
1, plug->info->name ? _(plug->info->name) : plug->path,
2, _(plug->info->summary),
3, plug, -1);
}
}
static GtkWidget *plugin_page ()
{
GtkWidget *ret;
GtkWidget *sw, *vp;
GtkWidget *event_view;
GtkListStore *ls;
GtkCellRenderer *rend, *rendt;
GtkTreeViewColumn *col;
GtkTreeSelection *sel;
GtkTreePath *path;
GtkWidget *nb;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
sw = gtk_scrolled_window_new(NULL,NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
gtk_box_pack_start(GTK_BOX(ret), sw, TRUE, TRUE, 0);
ls = gtk_list_store_new (4, G_TYPE_BOOLEAN, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_POINTER);
gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(ls),
1, GTK_SORT_ASCENDING);
update_plugin_list(ls);
event_view = gtk_tree_view_new_with_model (GTK_TREE_MODEL(ls));
rend = gtk_cell_renderer_toggle_new();
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (event_view));
col = gtk_tree_view_column_new_with_attributes (_("Load"),
rend,
"active", 0,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
rendt = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes (_("Name"),
rendt,
"text", 1,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
rendt = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes(_("Summary"),
rendt,
"text", 2,
NULL);
gtk_tree_view_append_column(GTK_TREE_VIEW(event_view), col);
g_object_unref(G_OBJECT(ls));
gtk_container_add(GTK_CONTAINER(sw), event_view);
nb = gtk_notebook_new();
gtk_notebook_set_tab_pos (GTK_NOTEBOOK(nb), GTK_POS_BOTTOM);
gtk_notebook_popup_disable(GTK_NOTEBOOK(nb));
/* Description */
sw = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
plugin_description = gtk_label_new(NULL);
vp = gtk_viewport_new(NULL, NULL);
gtk_viewport_set_shadow_type(GTK_VIEWPORT(vp), GTK_SHADOW_NONE);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_NONE);
gtk_container_add(GTK_CONTAINER(vp), plugin_description);
gtk_container_add(GTK_CONTAINER(sw), vp);
gtk_label_set_selectable(GTK_LABEL(plugin_description), TRUE);
gtk_label_set_line_wrap(GTK_LABEL(plugin_description), TRUE);
gtk_misc_set_alignment(GTK_MISC(plugin_description), 0, 0);
gtk_misc_set_padding(GTK_MISC(plugin_description), 6, 6);
gtk_notebook_append_page(GTK_NOTEBOOK(nb), sw, gtk_label_new(_("Description")));
/* Details */
sw = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
plugin_details = gtk_label_new(NULL);
vp = gtk_viewport_new(NULL, NULL);
gtk_viewport_set_shadow_type(GTK_VIEWPORT(vp), GTK_SHADOW_NONE);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_NONE);
gtk_container_add(GTK_CONTAINER(vp), plugin_details);
gtk_container_add(GTK_CONTAINER(sw), vp);
gtk_label_set_selectable(GTK_LABEL(plugin_details), TRUE);
gtk_label_set_line_wrap(GTK_LABEL(plugin_details), TRUE);
gtk_misc_set_alignment(GTK_MISC(plugin_details), 0, 0);
gtk_misc_set_padding(GTK_MISC(plugin_details), 6, 6);
gtk_notebook_append_page(GTK_NOTEBOOK(nb), sw, gtk_label_new(_("Details")));
gtk_box_pack_start(GTK_BOX(ret), nb, TRUE, TRUE, 0);
g_signal_connect (G_OBJECT (sel), "changed",
G_CALLBACK (prefs_plugin_sel),
NULL);
g_signal_connect (G_OBJECT(rend), "toggled",
G_CALLBACK(plugin_load), ls);
path = gtk_tree_path_new_first();
gtk_tree_selection_select_path(sel, path);
gtk_tree_path_free(path);
gaim_plugins_register_probe_notify_cb(update_plugin_list, ls);
gtk_widget_show_all(ret);
return ret;
}
static void away_message_sel_cb(GtkTreeSelection *sel, GtkTreeModel *model)
{
GtkTreeIter iter;
GValue val = { 0, };
gchar buffer[BUF_LONG];
char *tmp;
struct away_message *am;
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (model, &iter, 1, &val);
am = g_value_get_pointer(&val);
gtk_imhtml_clear(GTK_IMHTML(away_text));
strncpy(buffer, am->message, BUF_LONG);
tmp = stylize(buffer, BUF_LONG);
gtk_imhtml_append_text(GTK_IMHTML(away_text), tmp, GTK_IMHTML_NO_TITLE |
GTK_IMHTML_NO_COMMENTS | GTK_IMHTML_NO_SCROLL);
gtk_imhtml_append_text(GTK_IMHTML(away_text), "
", GTK_IMHTML_NO_TITLE |
GTK_IMHTML_NO_COMMENTS | GTK_IMHTML_NO_SCROLL);
g_free(tmp);
g_value_unset (&val);
}
static gboolean away_message_click_cb(GtkWidget *tv, GdkEventButton *event, gpointer null)
{
/* Only respond to double click on button 1 */
if ((event->button != 1) || (event->type != GDK_2BUTTON_PRESS))
return FALSE;
/* Show the edit away message dialog */
create_away_mess(NULL, tv);
return FALSE;
}
void remove_away_message(GtkWidget *widget, GtkTreeView *tv) {
struct away_message *am;
GtkTreeIter iter;
GtkTreeSelection *sel = gtk_tree_view_get_selection(tv);
GtkTreeModel *model = GTK_TREE_MODEL(prefs_away_store);
GValue val = { 0, };
if (! gtk_tree_selection_get_selected (sel, &model, &iter))
return;
gtk_tree_model_get_value (GTK_TREE_MODEL(prefs_away_store), &iter, 1, &val);
am = g_value_get_pointer (&val);
gtk_imhtml_clear(GTK_IMHTML(away_text));
rem_away_mess(NULL, am);
}
GtkWidget *away_message_page() {
GtkWidget *ret;
GtkWidget *hbox;
GtkWidget *button;
GtkWidget *sw;
GtkTreeIter iter;
GtkWidget *event_view;
GtkCellRenderer *rend;
GtkTreeViewColumn *col;
GtkTreeSelection *sel;
GSList *awy = away_messages;
struct away_message *a;
GtkSizeGroup *sg;
ret = gtk_vbox_new(FALSE, 18);
gtk_container_set_border_width (GTK_CONTAINER (ret), 12);
sg = gtk_size_group_new(GTK_SIZE_GROUP_BOTH);
sw = gtk_scrolled_window_new(NULL,NULL);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw), GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
gtk_box_pack_start(GTK_BOX(ret), sw, TRUE, TRUE, 0);
prefs_away_store = gtk_list_store_new(2, G_TYPE_STRING, G_TYPE_POINTER);
while (awy) {
a = (struct away_message *)awy->data;
gtk_list_store_append (prefs_away_store, &iter);
gtk_list_store_set(prefs_away_store, &iter,
0, a->name,
1, a, -1);
awy = awy->next;
}
event_view = gtk_tree_view_new_with_model(GTK_TREE_MODEL(prefs_away_store));
rend = gtk_cell_renderer_text_new();
col = gtk_tree_view_column_new_with_attributes ("NULL",
rend,
"text", 0,
NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW(event_view), col);
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(event_view), FALSE);
gtk_widget_show(event_view);
gtk_container_add(GTK_CONTAINER(sw), event_view);
sw = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(sw), GTK_SHADOW_IN);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(sw),
GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
gtk_box_pack_start(GTK_BOX(ret), sw, TRUE, TRUE, 0);
away_text = gtk_imhtml_new(NULL, NULL);
gtk_container_add(GTK_CONTAINER(sw), away_text);
gaim_setup_imhtml(away_text);
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (event_view));
g_signal_connect(G_OBJECT(sel), "changed",
G_CALLBACK(away_message_sel_cb), NULL);
g_signal_connect(G_OBJECT(event_view), "button-press-event",
G_CALLBACK(away_message_click_cb), NULL);
hbox = gtk_hbox_new(TRUE, 5);
gtk_box_pack_start(GTK_BOX(ret), hbox, FALSE, FALSE, 0);
button = gtk_button_new_from_stock (GTK_STOCK_ADD);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 0);
gtk_size_group_add_widget(sg, button);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(create_away_mess), NULL);
button = gtk_button_new_from_stock (GTK_STOCK_REMOVE);
gtk_size_group_add_widget(sg, button);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(remove_away_message), event_view);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 0);
button = gaim_pixbuf_button_from_stock(_("_Edit"), GAIM_STOCK_EDIT, GAIM_BUTTON_HORIZONTAL);
gtk_size_group_add_widget(sg, button);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(create_away_mess), event_view);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, FALSE, 0);
gtk_widget_show_all(ret);
return ret;
}
GtkTreeIter *prefs_notebook_add_page(const char *text,
GdkPixbuf *pixbuf,
GtkWidget *page,
GtkTreeIter *iter,
GtkTreeIter *parent,
int ind) {
GdkPixbuf *icon = NULL;
if (pixbuf)
icon = gdk_pixbuf_scale_simple (pixbuf, 18, 18, GDK_INTERP_BILINEAR);
gtk_tree_store_append (prefstree, iter, parent);
gtk_tree_store_set (prefstree, iter, 0, icon, 1, text, 2, ind, -1);
if (pixbuf)
g_object_unref(pixbuf);
if (icon)
g_object_unref(icon);
gtk_notebook_append_page(GTK_NOTEBOOK(prefsnotebook), page, gtk_label_new(text));
return iter;
}
void prefs_notebook_init() {
GtkTreeIter p, c, c2;
GList *l;
GaimPlugin *plug;
prefs_notebook_add_page(_("Interface"), NULL, interface_page(), &p, NULL, notebook_page++);
prefs_notebook_add_page(_("Buddy List"), NULL, list_page(), &c, &p, notebook_page++);
prefs_notebook_add_page(_("Conversations"), NULL, conv_page(), &c, &p, notebook_page++);
prefs_notebook_add_page(_("Message Text"), NULL, messages_page(), &c2, &c, notebook_page++);
prefs_notebook_add_page(_("Shortcuts"), NULL, hotkeys_page(), &c2, &c, notebook_page++);
prefs_notebook_add_page(_("Smiley Themes"), NULL, theme_page(), &c2, &c, notebook_page++);
prefs_notebook_add_page(_("Sounds"), NULL, sound_page(), &c, &p, notebook_page++);
prefs_notebook_add_page(_("Network"), NULL, network_page(), &p, NULL, notebook_page++);
#ifndef _WIN32
/* We use the registered default browser in windows */
prefs_notebook_add_page(_("Browser"), NULL, browser_page(), &p, NULL, notebook_page++);
#endif
prefs_notebook_add_page(_("Logging"), NULL, logging_page(), &p, NULL, notebook_page++);
prefs_notebook_add_page(_("Away / Idle"), NULL, away_page(), &p, NULL, notebook_page++);
prefs_notebook_add_page(_("Away Messages"), NULL, away_message_page(), &c, &p, notebook_page++);
if (gaim_plugins_enabled()) {
prefs_notebook_add_page(_("Plugins"), NULL, plugin_page(), &plugin_iter, NULL, notebook_page++);
for (l = gaim_plugins_get_loaded(); l != NULL; l = l->next) {
plug = (GaimPlugin *)l->data;
if (GAIM_IS_GTK_PLUGIN(plug)) {
GtkWidget *config_frame;
GaimGtkPluginUiInfo *ui_info;
ui_info = GAIM_GTK_PLUGIN_UI_INFO(plug);
config_frame = gaim_gtk_plugin_get_config_frame(plug);
if (config_frame != NULL) {
ui_info->iter = g_new0(GtkTreeIter, 1);
prefs_notebook_add_page(_(plug->info->name), NULL,
config_frame, ui_info->iter,
&plugin_iter, notebook_page++);
}
}
if(GAIM_PLUGIN_HAS_PREF_FRAME(plug)) {
GtkWidget *gtk_frame;
GaimPluginUiInfo *prefs_info;
prefs_info = GAIM_PLUGIN_UI_INFO(plug);
prefs_info->frame = prefs_info->get_plugin_pref_frame(plug);
gtk_frame = gaim_gtk_plugin_pref_create_frame(prefs_info->frame);
if(GTK_IS_WIDGET(gtk_frame)) {
prefs_info->iter = g_new0(GtkTreeIter, 1);
prefs_notebook_add_page(_(plug->info->name), NULL,
gtk_frame, prefs_info->iter,
(plug->info->type == GAIM_PLUGIN_PROTOCOL) ? NULL : &plugin_iter,
notebook_page++);
} else if(prefs_info->frame) {
/* in the event that there is a pref frame and we can
* not make a widget out of it, we free the
* pluginpref frame --Gary
*/
gaim_plugin_pref_frame_destroy(prefs_info->frame);
}
}
}
}
}
void gaim_gtk_prefs_show(void)
{
GtkWidget *vbox, *vbox2;
GtkWidget *hbox;
GtkWidget *bbox;
GtkWidget *frame;
GtkWidget *scrolled_window;
GtkTreeViewColumn *column;
GtkCellRenderer *cell;
GtkTreeSelection *sel;
GtkWidget *notebook;
GtkWidget *sep;
GtkWidget *button;
if (prefs) {
gtk_window_present(GTK_WINDOW(prefs));
return;
}
/* copy the preferences to tmp values...
* I liked "take affect immediately" Oh well :-( */
/* (that should have been "effect," right?) */
/* Back to instant-apply! I win! BU-HAHAHA! */
/* Create the window */
prefs = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_role(GTK_WINDOW(prefs), "preferences");
gtk_window_set_title(GTK_WINDOW(prefs), _("Preferences"));
gtk_window_set_resizable (GTK_WINDOW(prefs), FALSE);
gtk_container_set_border_width(GTK_CONTAINER(prefs), 12);
g_signal_connect(G_OBJECT(prefs), "destroy",
G_CALLBACK(delete_prefs), NULL);
vbox = gtk_vbox_new(FALSE, 12);
gtk_container_add(GTK_CONTAINER(prefs), vbox);
gtk_widget_show(vbox);
hbox = gtk_hbox_new (FALSE, 6);
gtk_container_add (GTK_CONTAINER(vbox), hbox);
gtk_widget_show (hbox);
frame = gtk_frame_new (NULL);
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
gtk_box_pack_start (GTK_BOX (hbox), frame, FALSE, FALSE, 0);
gtk_widget_show (frame);
scrolled_window = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_window),
GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);
gtk_container_add(GTK_CONTAINER(frame), scrolled_window);
gtk_widget_show(scrolled_window);
/* The tree -- much inspired by the Gimp */
prefstree = gtk_tree_store_new (3, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_INT);
tree_v = gtk_tree_view_new_with_model (GTK_TREE_MODEL (prefstree));
gtk_container_add(GTK_CONTAINER(scrolled_window), tree_v);
gtk_tree_view_set_headers_visible (GTK_TREE_VIEW (tree_v), FALSE);
gtk_widget_show(tree_v);
/* icons */
/* XXX: to be used at a later date
cell = gtk_cell_renderer_pixbuf_new ();
column = gtk_tree_view_column_new_with_attributes ("icons", cell, "pixbuf", 0, NULL);
*/
/* text */
cell = gtk_cell_renderer_text_new ();
column = gtk_tree_view_column_new_with_attributes ("text", cell, "text", 1, NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW (tree_v), column);
/* The right side */
frame = gtk_frame_new (NULL);
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_IN);
gtk_box_pack_start (GTK_BOX (hbox), frame, TRUE, TRUE, 0);
gtk_widget_show (frame);
vbox2 = gtk_vbox_new (FALSE, 4);
gtk_container_add (GTK_CONTAINER (frame), vbox2);
gtk_widget_show (vbox2);
frame = gtk_frame_new (NULL);
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_OUT);
gtk_box_pack_start (GTK_BOX (vbox2), frame, FALSE, TRUE, 0);
gtk_widget_show (frame);
hbox = gtk_hbox_new (FALSE, 4);
gtk_container_set_border_width (GTK_CONTAINER (hbox), 4);
gtk_container_add (GTK_CONTAINER (frame), hbox);
gtk_widget_show (hbox);
preflabel = gtk_label_new(NULL);
gtk_box_pack_end (GTK_BOX (hbox), preflabel, FALSE, FALSE, 0);
gtk_widget_show (preflabel);
/* The notebook */
prefsnotebook = notebook = gtk_notebook_new ();
gtk_notebook_set_show_tabs (GTK_NOTEBOOK (notebook), FALSE);
gtk_notebook_set_show_border (GTK_NOTEBOOK (notebook), FALSE);
gtk_box_pack_start (GTK_BOX (vbox2), notebook, FALSE, FALSE, 0);
sel = gtk_tree_view_get_selection (GTK_TREE_VIEW (tree_v));
g_signal_connect (G_OBJECT (sel), "changed",
G_CALLBACK (pref_nb_select),
notebook);
gtk_widget_show(notebook);
sep = gtk_hseparator_new();
gtk_widget_show(sep);
gtk_box_pack_start (GTK_BOX (vbox), sep, FALSE, FALSE, 0);
/* The buttons to press! */
bbox = gtk_hbutton_box_new();
gtk_box_set_spacing(GTK_BOX(bbox), 6);
gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
gtk_box_pack_start(GTK_BOX(vbox), bbox, FALSE, FALSE, 0);
gtk_widget_show (bbox);
button = gtk_button_new_from_stock (GTK_STOCK_CLOSE);
g_signal_connect_swapped(G_OBJECT(button), "clicked",
G_CALLBACK(gtk_widget_destroy), prefs);
gtk_box_pack_start(GTK_BOX(bbox), button, FALSE, FALSE, 0);
gtk_widget_show(button);
prefs_notebook_init();
/* Show everything. */
gtk_tree_view_expand_all (GTK_TREE_VIEW(tree_v));
gtk_widget_show(prefs);
}
static void
set_bool_pref(GtkWidget *w, const char *key)
{
gaim_prefs_set_bool(key,
gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w)));
}
GtkWidget *
gaim_gtk_prefs_checkbox(const char *text, const char *key, GtkWidget *page)
{
GtkWidget *button;
button = gtk_check_button_new_with_mnemonic(text);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button),
gaim_prefs_get_bool(key));
gtk_box_pack_start(GTK_BOX(page), button, FALSE, FALSE, 0);
g_signal_connect(G_OBJECT(button), "clicked",
G_CALLBACK(set_bool_pref), (char *)key);
gtk_widget_show(button);
return button;
}
void default_away_menu_init(GtkWidget *omenu)
{
GtkWidget *menu, *opt;
int index = 0, default_index = 0;
GSList *awy = away_messages;
struct away_message *a;
const char *default_name;
menu = gtk_menu_new();
default_name = gaim_prefs_get_string("/core/away/default_message");
while (awy) {
a = (struct away_message *)awy->data;
opt = gtk_menu_item_new_with_label(a->name);
g_signal_connect(G_OBJECT(opt), "activate",
G_CALLBACK(set_default_away), GINT_TO_POINTER(index));
gtk_widget_show(opt);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), opt);
if(!strcmp(default_name, a->name))
default_index = index;
awy = awy->next;
index++;
}
gtk_option_menu_remove_menu(GTK_OPTION_MENU(omenu));
gtk_option_menu_set_menu(GTK_OPTION_MENU(omenu), menu);
gtk_option_menu_set_history(GTK_OPTION_MENU(omenu), default_index);
}
void set_default_away(GtkWidget *w, gpointer data)
{
struct away_message *default_away = NULL;
int length = g_slist_length(away_messages);
int i = GPOINTER_TO_INT(data);
if (away_messages == NULL)
default_away = NULL;
else if (i >= length)
default_away = g_slist_nth_data(away_messages, length - 1);
else
default_away = g_slist_nth_data(away_messages, i);
if(default_away)
gaim_prefs_set_string("/core/away/default_message", default_away->name);
else
gaim_prefs_set_string("/core/away/default_message", "");
}
static void
smiley_theme_pref_cb(const char *name, GaimPrefType type, gpointer value,
gpointer data)
{
if (!strcmp(name, "/gaim/gtk/smileys/theme"))
load_smiley_theme((const char *)value, TRUE);
}
void
gaim_gtk_prefs_init(void)
{
gaim_prefs_add_none("/gaim");
gaim_prefs_add_none("/gaim/gtk");
gaim_prefs_add_none("/plugins/gtk");
/* XXX Move this! HACK! :( Aww... */
gaim_prefs_add_none("/plugins/gtk/docklet");
gaim_prefs_add_bool("/plugins/gtk/docklet/queue_messages", FALSE);
/* Accounts Dialog */
gaim_prefs_add_none("/gaim/gtk/accounts");
gaim_prefs_add_none("/gaim/gtk/accounts/dialog");
gaim_prefs_add_int("/gaim/gtk/accounts/dialog/width", 550);
gaim_prefs_add_int("/gaim/gtk/accounts/dialog/height", 250);
/* Away Queueing */
gaim_prefs_add_none("/gaim/gtk/away");
gaim_prefs_add_bool("/gaim/gtk/away/queue_messages", FALSE);
#ifndef _WIN32
/* Browsers */
gaim_prefs_add_none("/gaim/gtk/browsers");
gaim_prefs_add_int("/gaim/gtk/browsers/place", GAIM_BROWSER_DEFAULT);
gaim_prefs_add_string("/gaim/gtk/browsers/command", "");
gaim_prefs_add_string("/gaim/gtk/browsers/browser", "mozilla");
#endif
/* Idle */
gaim_prefs_add_none("/gaim/gtk/idle");
gaim_prefs_add_string("/gaim/gtk/idle/reporting_method", "system");
/* Plugins */
gaim_prefs_add_none("/gaim/gtk/plugins");
gaim_prefs_add_string_list("/gaim/gtk/plugins/loaded", NULL);
/* Smiley Themes */
gaim_prefs_add_none("/gaim/gtk/smileys");
gaim_prefs_add_string("/gaim/gtk/smileys/theme", "");
/* Smiley Callbacks */
gaim_prefs_connect_callback("/gaim/gtk/smileys/theme",
smiley_theme_pref_cb, NULL);
}
void gaim_gtk_prefs_update_old() {
/* Rename some old prefs */
gaim_prefs_rename("/gaim/gtk/logging/log_ims", "/core/logging/log_ims");
gaim_prefs_rename("/gaim/gtk/logging/log_chats", "/core/logging/log_chats");
gaim_prefs_rename("/core/conversations/placement",
"/gaim/gtk/conversations/placement");
gaim_prefs_rename("/gaim/gtk/conversations/use_custom_font",
"/gaim/gtk/conversations/send_formatting");
gaim_prefs_rename("/gaim/gtk/conversations/im/button_type",
"/gaim/gtk/conversations/button_type");
/* Remove some no-longer-used prefs */
gaim_prefs_remove("/gaim/gtk/blist/show_group_count");
gaim_prefs_remove("/gaim/gtk/conversations/icons_on_tabs");
gaim_prefs_remove("/gaim/gtk/conversations/show_urls_as_links");
gaim_prefs_remove("/gaim/gtk/conversations/show_smileys");
gaim_prefs_remove("/gaim/gtk/conversations/chat/tab_completion");
gaim_prefs_remove("/gaim/gtk/conversations/chat/old_tab_complete");
gaim_prefs_remove("/gaim/gtk/sound/signon");
gaim_prefs_remove("/gaim/gtk/sound/silent_signon");
gaim_prefs_remove("/gaim/gtk/logging/individual_logs");
gaim_prefs_remove("/gaim/gtk/conversations/use_custom_bgcolor");
gaim_prefs_remove("/gaim/gtk/conversations/use_custom_fgcolor");
gaim_prefs_remove("/gaim/gtk/conversations/use_custom_size");
gaim_prefs_remove("/gaim/gtk/conversations/im/hide_on_send");
}