3 * Pidgin is the legal property of its developers, whose names are too numerous
4 * to list here. Please refer to the COPYRIGHT file distributed with this
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111-1301 USA
25 #include "glibcompat.h"
36 #include "savedstatuses.h"
38 #include "sound-theme.h"
40 #include "theme-manager.h"
48 #include "gtkdialogs.h"
50 #include "gtksavedstatuses.h"
51 #include "gtksmiley-theme.h"
53 #include "gtkstatus-icon-theme.h"
55 #include "pidgindebug.h"
56 #include "pidginstock.h"
58 #include "media-gst.h"
59 #include <gst/video/videooverlay.h>
60 #ifdef GDK_WINDOWING_WIN32
61 #include <gdk/gdkwin32.h>
63 #ifdef GDK_WINDOWING_X11
66 #ifdef GDK_WINDOWING_QUARTZ
67 #include <gdk/gdkquartz.h>
71 #include "gtk3compat.h"
73 #define PREFS_OPTIMAL_ICON_SIZE 32
76 #define PREFS_MAX_DOWNLOADED_THEME_SIZE 26214400
84 typedef struct _PidginPrefCombo PidginPrefCombo
;
86 typedef void (*PidginPrefsBindDropdownCallback
)(GtkComboBox
*combo_box
,
87 PidginPrefCombo
*combo
);
89 struct _PidginPrefCombo
{
98 gint previously_active
;
100 PidginPrefsBindDropdownCallback cb
;
103 struct _PidginPrefsWindow
{
111 PidginPrefCombo docklet
;
113 PidginPrefCombo hide_new
;
116 GtkWidget
*minimize_new_convs
;
120 GtkWidget
*tabs_vbox
;
121 GtkWidget
*close_on_tabs
;
122 PidginPrefCombo tab_side
;
123 PidginPrefCombo placement
;
132 GtkWidget
*gnome_not_found
;
133 GtkWidget
*gnome_program
;
134 gchar
*gnome_program_path
;
135 /* Non-GNOME version */
136 PidginPrefCombo browser
;
137 GtkWidget
*place_hbox
;
138 PidginPrefCombo place
;
139 GtkWidget
*manual_command_hbox
;
140 GtkWidget
*manual_command
;
143 /* Conversations page */
145 PidginPrefCombo notification_chat
;
146 GtkWidget
*show_incoming_formatting
;
148 GtkWidget
*close_immediately
;
149 GtkWidget
*show_buddy_icons
;
150 GtkWidget
*animate_buddy_icons
;
151 GtkWidget
*send_typing
;
153 GtkWidget
*spellcheck
;
154 GtkWidget
*use_smooth_scrolling
;
158 GtkWidget
*resize_custom_smileys
;
159 GtkWidget
*custom_smileys_size
;
160 GtkWidget
*minimum_entry_lines
;
161 GtkTextBuffer
*format_buffer
;
162 GtkWidget
*format_view
;
163 /* Win32 specific frame */
164 GtkWidget
*font_frame
;
165 GtkWidget
*use_theme_font
;
166 GtkWidget
*custom_font_hbox
;
167 GtkWidget
*custom_font
;
172 PidginPrefCombo format
;
174 GtkWidget
*log_chats
;
175 GtkWidget
*log_system
;
180 GtkWidget
*stun_server
;
182 GtkWidget
*public_ip
;
183 GtkWidget
*public_ip_hbox
;
184 GtkWidget
*map_ports
;
185 GtkWidget
*ports_range_use
;
186 GtkWidget
*ports_range_hbox
;
187 GtkWidget
*ports_range_start
;
188 GtkWidget
*ports_range_end
;
189 GtkWidget
*turn_server
;
190 GtkWidget
*turn_port_udp
;
191 GtkWidget
*turn_port_tcp
;
192 GtkWidget
*turn_username
;
193 GtkWidget
*turn_password
;
200 GtkWidget
*gnome_not_found
;
201 GtkWidget
*gnome_program
;
202 gchar
*gnome_program_path
;
203 /* Non-GNOME version */
204 GtkWidget
*socks4_remotedns
;
205 PidginPrefCombo type
;
215 PidginPrefCombo active
;
217 PurpleRequestFields
*settings
;
218 GtkWidget
*settings_box
;
224 PidginPrefCombo method
;
225 GtkWidget
*method_vbox
;
227 GtkWidget
*command_hbox
;
229 GtkWidget
*conv_focus
;
230 PidginPrefCombo while_status
;
240 PidginPrefCombo idle_reporting
;
241 GtkWidget
*mins_before_away
;
242 GtkWidget
*idle_hbox
;
243 GtkWidget
*away_when_idle
;
244 PidginPrefCombo auto_reply
;
245 GtkWidget
*startup_current_status
;
246 GtkWidget
*startup_hbox
;
247 GtkWidget
*startup_label
;
259 /* Voice/Video page */
262 PidginPrefCombo input
;
263 PidginPrefCombo output
;
265 GtkWidget
*threshold
;
268 GstElement
*pipeline
;
272 PidginPrefCombo input
;
273 PidginPrefCombo output
;
274 GtkWidget
*drawing_area
;
276 GstElement
*pipeline
;
283 static PidginPrefsWindow
*prefs
= NULL
;
286 static GtkWidget
*prefs_sound_themes_combo_box
;
287 static GtkWidget
*prefs_blist_themes_combo_box
;
288 static GtkWidget
*prefs_status_themes_combo_box
;
289 static GtkWidget
*prefs_smiley_themes_combo_box
;
290 static PurpleHttpConnection
*prefs_themes_running_request
= NULL
;
292 /* Sound theme specific */
293 static int sound_row_sel
= 0;
294 static gboolean prefs_sound_themes_loading
;
296 /* These exist outside the lifetime of the prefs dialog */
297 static GtkListStore
*prefs_sound_themes
;
298 static GtkListStore
*prefs_blist_themes
;
299 static GtkListStore
*prefs_status_icon_themes
;
300 static GtkListStore
*prefs_smiley_themes
;
305 G_DEFINE_TYPE(PidginPrefsWindow
, pidgin_prefs_window
, GTK_TYPE_DIALOG
);
306 static void delete_prefs(GtkWidget
*, void *);
309 update_spin_value(GtkWidget
*w
, GtkWidget
*spin
)
311 const char *key
= g_object_get_data(G_OBJECT(spin
), "val");
314 value
= gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(spin
));
316 purple_prefs_set_int(key
, value
);
320 pidgin_prefs_labeled_spin_button(GtkWidget
*box
, const gchar
*title
,
321 const char *key
, int min
, int max
, GtkSizeGroup
*sg
)
324 GtkAdjustment
*adjust
;
327 val
= purple_prefs_get_int(key
);
329 adjust
= GTK_ADJUSTMENT(gtk_adjustment_new(val
, min
, max
, 1, 1, 0));
330 spin
= gtk_spin_button_new(adjust
, 1, 0);
331 g_object_set_data(G_OBJECT(spin
), "val", (char *)key
);
333 gtk_widget_set_size_request(spin
, 50, -1);
335 gtk_widget_set_size_request(spin
, 60, -1);
336 g_signal_connect(G_OBJECT(adjust
), "value-changed",
337 G_CALLBACK(update_spin_value
), GTK_WIDGET(spin
));
338 gtk_widget_show(spin
);
340 return pidgin_add_widget_to_vbox(GTK_BOX(box
), title
, sg
, spin
, FALSE
, NULL
);
344 pidgin_prefs_bind_spin_button(const char *key
, GtkWidget
*spin
)
346 GtkAdjustment
*adjust
;
349 val
= purple_prefs_get_int(key
);
351 adjust
= gtk_spin_button_get_adjustment(GTK_SPIN_BUTTON(spin
));
352 gtk_adjustment_set_value(adjust
, val
);
353 g_object_set_data(G_OBJECT(spin
), "val", (char *)key
);
354 g_signal_connect(G_OBJECT(adjust
), "value-changed",
355 G_CALLBACK(update_spin_value
), GTK_WIDGET(spin
));
359 entry_set(GtkEntry
*entry
, gpointer data
)
361 const char *key
= (const char*)data
;
363 purple_prefs_set_string(key
, gtk_entry_get_text(entry
));
367 pidgin_prefs_labeled_entry(GtkWidget
*page
, const gchar
*title
,
368 const char *key
, GtkSizeGroup
*sg
)
373 value
= purple_prefs_get_string(key
);
375 entry
= gtk_entry_new();
376 gtk_entry_set_text(GTK_ENTRY(entry
), value
);
377 g_signal_connect(G_OBJECT(entry
), "changed",
378 G_CALLBACK(entry_set
), (char*)key
);
379 gtk_widget_show(entry
);
381 return pidgin_add_widget_to_vbox(GTK_BOX(page
), title
, sg
, entry
, TRUE
, NULL
);
385 pidgin_prefs_bind_entry(const char *key
, GtkWidget
*entry
)
389 value
= purple_prefs_get_string(key
);
391 gtk_entry_set_text(GTK_ENTRY(entry
), value
);
392 g_signal_connect(G_OBJECT(entry
), "changed", G_CALLBACK(entry_set
),
397 pidgin_prefs_labeled_password(GtkWidget
*page
, const gchar
*title
,
398 const char *key
, GtkSizeGroup
*sg
)
403 value
= purple_prefs_get_string(key
);
405 entry
= gtk_entry_new();
406 gtk_entry_set_visibility(GTK_ENTRY(entry
), FALSE
);
407 gtk_entry_set_text(GTK_ENTRY(entry
), value
);
408 g_signal_connect(G_OBJECT(entry
), "changed",
409 G_CALLBACK(entry_set
), (char*)key
);
410 gtk_widget_show(entry
);
412 return pidgin_add_widget_to_vbox(GTK_BOX(page
), title
, sg
, entry
, TRUE
, NULL
);
415 /* TODO: Maybe move this up somewheres... */
432 typedef void (*PidginPrefsDropdownCallback
)(GtkComboBox
*combo_box
,
433 PidginPrefValue value
);
436 dropdown_set(GtkComboBox
*combo_box
, gpointer _cb
)
438 PidginPrefsDropdownCallback cb
= _cb
;
440 GtkTreeModel
*tree_model
;
441 PidginPrefValue active
;
443 tree_model
= gtk_combo_box_get_model(combo_box
);
444 if (!gtk_combo_box_get_active_iter(combo_box
, &iter
))
446 active
.type
= GPOINTER_TO_INT(g_object_get_data(G_OBJECT(combo_box
),
449 g_object_set_data(G_OBJECT(combo_box
), "previously_active",
450 g_object_get_data(G_OBJECT(combo_box
), "current_active"));
451 g_object_set_data(G_OBJECT(combo_box
), "current_active",
452 GINT_TO_POINTER(gtk_combo_box_get_active(combo_box
)));
454 if (active
.type
== PURPLE_PREF_INT
) {
455 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
456 &active
.value
.integer
, -1);
458 else if (active
.type
== PURPLE_PREF_STRING
) {
459 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
460 &active
.value
.string
, -1);
462 else if (active
.type
== PURPLE_PREF_BOOLEAN
) {
463 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
464 &active
.value
.boolean
, -1);
467 cb(combo_box
, active
);
471 pidgin_prefs_bind_dropdown_revert_active(PidginPrefCombo
*combo
)
473 g_return_if_fail(combo
!= NULL
);
475 combo
->current_active
= combo
->previously_active
;
477 gtk_combo_box_set_active(GTK_COMBO_BOX(combo
->combo
),
478 combo
->previously_active
);
482 pidgin_prefs_dropdown_from_list_with_cb(GtkWidget
*box
, const gchar
*title
,
483 GtkComboBox
**dropdown_out
, GList
*menuitems
,
484 PidginPrefValue initial
, PidginPrefsDropdownCallback cb
)
487 GtkWidget
*label
= NULL
;
489 GtkListStore
*store
= NULL
;
492 GtkCellRenderer
*renderer
;
493 gpointer current_active
;
495 g_return_val_if_fail(menuitems
!= NULL
, NULL
);
497 if (initial
.type
== PURPLE_PREF_INT
) {
498 store
= gtk_list_store_new(PREF_DROPDOWN_COUNT
, G_TYPE_STRING
, G_TYPE_INT
);
499 } else if (initial
.type
== PURPLE_PREF_STRING
) {
500 store
= gtk_list_store_new(PREF_DROPDOWN_COUNT
, G_TYPE_STRING
, G_TYPE_STRING
);
501 } else if (initial
.type
== PURPLE_PREF_BOOLEAN
) {
502 store
= gtk_list_store_new(PREF_DROPDOWN_COUNT
, G_TYPE_STRING
, G_TYPE_BOOLEAN
);
508 dropdown
= gtk_combo_box_new_with_model(GTK_TREE_MODEL(store
));
509 if (dropdown_out
!= NULL
)
510 *dropdown_out
= GTK_COMBO_BOX(dropdown
);
511 g_object_set_data(G_OBJECT(dropdown
), "type", GINT_TO_POINTER(initial
.type
));
513 while (menuitems
!= NULL
&& (text
= (char *)menuitems
->data
) != NULL
) {
515 const char *str_value
= NULL
;
516 gboolean bool_value
= FALSE
;
518 menuitems
= g_list_next(menuitems
);
519 g_return_val_if_fail(menuitems
!= NULL
, NULL
);
521 gtk_list_store_append(store
, &iter
);
522 gtk_list_store_set(store
, &iter
,
523 PREF_DROPDOWN_TEXT
, text
,
526 if (initial
.type
== PURPLE_PREF_INT
) {
527 int_value
= GPOINTER_TO_INT(menuitems
->data
);
528 gtk_list_store_set(store
, &iter
,
529 PREF_DROPDOWN_VALUE
, int_value
,
532 else if (initial
.type
== PURPLE_PREF_STRING
) {
533 str_value
= (const char *)menuitems
->data
;
534 gtk_list_store_set(store
, &iter
,
535 PREF_DROPDOWN_VALUE
, str_value
,
538 else if (initial
.type
== PURPLE_PREF_BOOLEAN
) {
539 bool_value
= (gboolean
)GPOINTER_TO_INT(menuitems
->data
);
540 gtk_list_store_set(store
, &iter
,
541 PREF_DROPDOWN_VALUE
, bool_value
,
545 if ((initial
.type
== PURPLE_PREF_INT
&&
546 initial
.value
.integer
== int_value
) ||
547 (initial
.type
== PURPLE_PREF_STRING
&&
548 purple_strequal(initial
.value
.string
, str_value
)) ||
549 (initial
.type
== PURPLE_PREF_BOOLEAN
&&
550 (initial
.value
.boolean
== bool_value
))) {
555 menuitems
= g_list_next(menuitems
);
558 renderer
= gtk_cell_renderer_text_new();
559 gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(dropdown
), renderer
, TRUE
);
560 gtk_cell_layout_set_attributes(GTK_CELL_LAYOUT(dropdown
), renderer
,
564 gtk_combo_box_set_active_iter(GTK_COMBO_BOX(dropdown
), &active
);
565 current_active
= GINT_TO_POINTER(gtk_combo_box_get_active(GTK_COMBO_BOX(
567 g_object_set_data(G_OBJECT(dropdown
), "current_active", current_active
);
568 g_object_set_data(G_OBJECT(dropdown
), "previously_active", current_active
);
570 g_signal_connect(G_OBJECT(dropdown
), "changed",
571 G_CALLBACK(dropdown_set
), cb
);
573 pidgin_add_widget_to_vbox(GTK_BOX(box
), title
, NULL
, dropdown
, FALSE
, &label
);
579 pidgin_prefs_dropdown_from_list_cb(GtkComboBox
*combo_box
,
580 PidginPrefValue value
)
584 key
= g_object_get_data(G_OBJECT(combo_box
), "key");
586 if (value
.type
== PURPLE_PREF_INT
) {
587 purple_prefs_set_int(key
, value
.value
.integer
);
588 } else if (value
.type
== PURPLE_PREF_STRING
) {
589 purple_prefs_set_string(key
, value
.value
.string
);
590 } else if (value
.type
== PURPLE_PREF_BOOLEAN
) {
591 purple_prefs_set_bool(key
, value
.value
.boolean
);
593 g_return_if_reached();
598 pidgin_prefs_dropdown_from_list(GtkWidget
*box
, const gchar
*title
,
599 PurplePrefType type
, const char *key
, GList
*menuitems
)
601 PidginPrefValue initial
;
602 GtkComboBox
*dropdown
= NULL
;
606 if (type
== PURPLE_PREF_INT
) {
607 initial
.value
.integer
= purple_prefs_get_int(key
);
608 } else if (type
== PURPLE_PREF_STRING
) {
609 initial
.value
.string
= purple_prefs_get_string(key
);
610 } else if (type
== PURPLE_PREF_BOOLEAN
) {
611 initial
.value
.boolean
= purple_prefs_get_bool(key
);
613 g_return_val_if_reached(NULL
);
616 label
= pidgin_prefs_dropdown_from_list_with_cb(box
, title
, &dropdown
,
617 menuitems
, initial
, pidgin_prefs_dropdown_from_list_cb
);
619 g_object_set_data(G_OBJECT(dropdown
), "key", (gpointer
)key
);
625 pidgin_prefs_dropdown(GtkWidget
*box
, const gchar
*title
, PurplePrefType type
,
626 const char *key
, ...)
629 GList
*menuitems
= NULL
;
630 GtkWidget
*dropdown
= NULL
;
633 const char *str_value
;
635 g_return_val_if_fail(type
== PURPLE_PREF_BOOLEAN
|| type
== PURPLE_PREF_INT
||
636 type
== PURPLE_PREF_STRING
, NULL
);
639 while ((name
= va_arg(ap
, char *)) != NULL
) {
641 menuitems
= g_list_prepend(menuitems
, name
);
643 if (type
== PURPLE_PREF_INT
|| type
== PURPLE_PREF_BOOLEAN
) {
644 int_value
= va_arg(ap
, int);
645 menuitems
= g_list_prepend(menuitems
, GINT_TO_POINTER(int_value
));
648 str_value
= va_arg(ap
, const char *);
649 menuitems
= g_list_prepend(menuitems
, (char *)str_value
);
654 g_return_val_if_fail(menuitems
!= NULL
, NULL
);
656 menuitems
= g_list_reverse(menuitems
);
658 dropdown
= pidgin_prefs_dropdown_from_list(box
, title
, type
, key
,
661 g_list_free(menuitems
);
667 pidgin_prefs_bind_dropdown_from_list_cb(GtkComboBox
*combo_box
,
668 PidginPrefCombo
*combo
)
670 if (combo
->type
== PURPLE_PREF_INT
) {
671 purple_prefs_set_int(combo
->key
, combo
->value
.integer
);
672 } else if (combo
->type
== PURPLE_PREF_STRING
) {
673 purple_prefs_set_string(combo
->key
, combo
->value
.string
);
674 } else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
675 purple_prefs_set_bool(combo
->key
, combo
->value
.boolean
);
677 g_return_if_reached();
682 bind_dropdown_set(GtkComboBox
*combo_box
, gpointer data
)
684 PidginPrefCombo
*combo
= data
;
686 GtkTreeModel
*tree_model
;
688 tree_model
= gtk_combo_box_get_model(combo_box
);
689 if (!gtk_combo_box_get_active_iter(combo_box
, &iter
))
692 combo
->previously_active
= combo
->current_active
;
693 combo
->current_active
= gtk_combo_box_get_active(combo_box
);
695 if (combo
->type
== PURPLE_PREF_INT
) {
696 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
697 &combo
->value
.integer
, -1);
699 else if (combo
->type
== PURPLE_PREF_STRING
) {
700 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
701 &combo
->value
.string
, -1);
703 else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
704 gtk_tree_model_get(tree_model
, &iter
, PREF_DROPDOWN_VALUE
,
705 &combo
->value
.boolean
, -1);
708 combo
->cb(combo_box
, combo
);
712 pidgin_prefs_bind_dropdown_from_list(PidginPrefCombo
*combo
, GList
*menuitems
)
715 GtkListStore
*store
= NULL
;
719 g_return_if_fail(menuitems
!= NULL
);
721 if (combo
->type
== PURPLE_PREF_INT
) {
722 combo
->value
.integer
= purple_prefs_get_int(combo
->key
);
723 } else if (combo
->type
== PURPLE_PREF_STRING
) {
724 combo
->value
.string
= purple_prefs_get_string(combo
->key
);
725 } else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
726 combo
->value
.boolean
= purple_prefs_get_bool(combo
->key
);
728 g_return_if_reached();
731 store
= GTK_LIST_STORE(
732 gtk_combo_box_get_model(GTK_COMBO_BOX(combo
->combo
)));
734 while (menuitems
!= NULL
&& (text
= (char *)menuitems
->data
) != NULL
) {
736 const char *str_value
= NULL
;
737 gboolean bool_value
= FALSE
;
739 menuitems
= g_list_next(menuitems
);
740 g_return_if_fail(menuitems
!= NULL
);
742 gtk_list_store_append(store
, &iter
);
743 gtk_list_store_set(store
, &iter
,
744 PREF_DROPDOWN_TEXT
, text
,
747 if (combo
->type
== PURPLE_PREF_INT
) {
748 int_value
= GPOINTER_TO_INT(menuitems
->data
);
749 gtk_list_store_set(store
, &iter
,
750 PREF_DROPDOWN_VALUE
, int_value
,
753 else if (combo
->type
== PURPLE_PREF_STRING
) {
754 str_value
= (const char *)menuitems
->data
;
755 gtk_list_store_set(store
, &iter
,
756 PREF_DROPDOWN_VALUE
, str_value
,
759 else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
760 bool_value
= (gboolean
)GPOINTER_TO_INT(menuitems
->data
);
761 gtk_list_store_set(store
, &iter
,
762 PREF_DROPDOWN_VALUE
, bool_value
,
766 if ((combo
->type
== PURPLE_PREF_INT
&&
767 combo
->value
.integer
== int_value
) ||
768 (combo
->type
== PURPLE_PREF_STRING
&&
769 purple_strequal(combo
->value
.string
, str_value
)) ||
770 (combo
->type
== PURPLE_PREF_BOOLEAN
&&
771 (combo
->value
.boolean
== bool_value
))) {
776 menuitems
= g_list_next(menuitems
);
779 gtk_combo_box_set_active_iter(GTK_COMBO_BOX(combo
->combo
), &active
);
780 combo
->current_active
= gtk_combo_box_get_active(
781 GTK_COMBO_BOX(combo
->combo
));
782 combo
->previously_active
= combo
->current_active
;
784 combo
->cb
= pidgin_prefs_bind_dropdown_from_list_cb
;
785 g_signal_connect(G_OBJECT(combo
->combo
), "changed",
786 G_CALLBACK(bind_dropdown_set
), combo
);
790 pidgin_prefs_bind_dropdown(PidginPrefCombo
*combo
)
792 GtkTreeModel
*store
= NULL
;
796 if (combo
->type
== PURPLE_PREF_INT
) {
797 combo
->value
.integer
= purple_prefs_get_int(combo
->key
);
798 } else if (combo
->type
== PURPLE_PREF_STRING
) {
799 combo
->value
.string
= purple_prefs_get_string(combo
->key
);
800 } else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
801 combo
->value
.boolean
= purple_prefs_get_bool(combo
->key
);
803 g_return_if_reached();
806 store
= gtk_combo_box_get_model(GTK_COMBO_BOX(combo
->combo
));
808 if (!gtk_tree_model_get_iter_first(store
, &iter
)) {
809 g_return_if_reached();
814 const char *str_value
= NULL
;
815 gboolean bool_value
= FALSE
;
817 if (combo
->type
== PURPLE_PREF_INT
) {
818 gtk_tree_model_get(store
, &iter
,
819 PREF_DROPDOWN_VALUE
, &int_value
,
821 if (combo
->value
.integer
== int_value
) {
826 else if (combo
->type
== PURPLE_PREF_STRING
) {
827 gtk_tree_model_get(store
, &iter
,
828 PREF_DROPDOWN_VALUE
, &str_value
,
830 if (purple_strequal(combo
->value
.string
, str_value
)) {
835 else if (combo
->type
== PURPLE_PREF_BOOLEAN
) {
836 gtk_tree_model_get(store
, &iter
,
837 PREF_DROPDOWN_VALUE
, &bool_value
,
839 if (combo
->value
.boolean
== bool_value
) {
844 } while (gtk_tree_model_iter_next(store
, &iter
));
846 gtk_combo_box_set_active_iter(GTK_COMBO_BOX(combo
->combo
), &active
);
848 combo
->current_active
= gtk_combo_box_get_active(
849 GTK_COMBO_BOX(combo
->combo
));
850 combo
->previously_active
= combo
->current_active
;
852 combo
->cb
= pidgin_prefs_bind_dropdown_from_list_cb
;
853 g_signal_connect(G_OBJECT(combo
->combo
), "changed",
854 G_CALLBACK(bind_dropdown_set
), combo
);
858 set_bool_pref(GtkWidget
*w
, const char *key
)
860 purple_prefs_set_bool(key
,
861 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w
)));
865 pidgin_prefs_checkbox(const char *text
, const char *key
, GtkWidget
*page
)
869 button
= gtk_check_button_new_with_mnemonic(text
);
870 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button
),
871 purple_prefs_get_bool(key
));
873 gtk_box_pack_start(GTK_BOX(page
), button
, FALSE
, FALSE
, 0);
875 g_signal_connect(G_OBJECT(button
), "clicked",
876 G_CALLBACK(set_bool_pref
), (char *)key
);
878 gtk_widget_show(button
);
884 pidgin_prefs_bind_checkbox(const char *key
, GtkWidget
*button
)
886 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button
),
887 purple_prefs_get_bool(key
));
888 g_signal_connect(G_OBJECT(button
), "toggled",
889 G_CALLBACK(set_bool_pref
), (char *)key
);
892 static void keyring_page_cleanup(PidginPrefsWindow
*win
);
895 delete_prefs(GtkWidget
*asdf
, void *gdsa
)
897 /* Close any "select sound" request dialogs */
898 purple_request_close_with_handle(prefs
);
900 purple_notify_close_with_handle(prefs
);
902 /* Unregister callbacks. */
903 purple_prefs_disconnect_by_handle(prefs
);
905 /* NULL-ify globals */
907 prefs_sound_themes_loading
= FALSE
;
909 prefs_sound_themes_combo_box
= NULL
;
910 prefs_blist_themes_combo_box
= NULL
;
911 prefs_status_themes_combo_box
= NULL
;
912 prefs_smiley_themes_combo_box
= NULL
;
914 keyring_page_cleanup(prefs
);
916 g_free(prefs
->proxy
.gnome_program_path
);
917 g_free(prefs
->browser
.gnome_program_path
);
922 get_theme_markup(const char *name
, gboolean custom
, const char *author
,
923 const char *description
)
926 return g_strdup_printf("<b>%s</b>%s%s%s%s\n<span foreground='dim grey'>%s</span>",
927 name
, custom
? " " : "", custom
? _("(Custom)") : "",
928 author
!= NULL
? " - " : "", author
!= NULL
? author
: "",
929 description
!= NULL
? description
: "");
933 smileys_refresh_theme_list(void)
939 description
= get_theme_markup(_("none"), FALSE
, _("Penguin Pimps"),
940 _("Selecting this disables graphical emoticons."));
941 gtk_list_store_append(prefs_smiley_themes
, &iter
);
942 gtk_list_store_set(prefs_smiley_themes
, &iter
,
943 0, NULL
, 1, description
, 2, "none", -1);
946 for (it
= pidgin_smiley_theme_get_all(); it
; it
= g_list_next(it
)) {
947 PidginSmileyTheme
*theme
= it
->data
;
949 description
= get_theme_markup(
950 _(pidgin_smiley_theme_get_name(theme
)), FALSE
,
951 _(pidgin_smiley_theme_get_author(theme
)),
952 _(pidgin_smiley_theme_get_description(theme
)));
954 gtk_list_store_append(prefs_smiley_themes
, &iter
);
955 gtk_list_store_set(prefs_smiley_themes
, &iter
,
956 0, pidgin_smiley_theme_get_icon(theme
),
958 2, pidgin_smiley_theme_get_name(theme
),
965 /* Rebuild the markup for the sound theme selection for "(Custom)" themes */
967 pref_sound_generate_markup(void)
969 gboolean print_custom
, customized
;
970 const gchar
*author
, *description
, *current_theme
;
971 gchar
*name
, *markup
;
972 PurpleSoundTheme
*theme
;
975 customized
= pidgin_sound_is_customized();
976 current_theme
= purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/sound/theme");
978 if (gtk_tree_model_get_iter_first(GTK_TREE_MODEL(prefs_sound_themes
), &iter
)) {
980 gtk_tree_model_get(GTK_TREE_MODEL(prefs_sound_themes
), &iter
, 2, &name
, -1);
982 print_custom
= customized
&& name
&& purple_strequal(current_theme
, name
);
984 if (!name
|| *name
== '\0') {
986 name
= g_strdup(_("Default"));
987 author
= _("Penguin Pimps");
988 description
= _("The default Pidgin sound theme");
990 theme
= PURPLE_SOUND_THEME(purple_theme_manager_find_theme(name
, "sound"));
991 author
= purple_theme_get_author(PURPLE_THEME(theme
));
992 description
= purple_theme_get_description(PURPLE_THEME(theme
));
995 markup
= get_theme_markup(name
, print_custom
, author
, description
);
997 gtk_list_store_set(prefs_sound_themes
, &iter
, 1, markup
, -1);
1002 } while (gtk_tree_model_iter_next(GTK_TREE_MODEL(prefs_sound_themes
), &iter
));
1006 /* adds the themes to the theme list from the manager so they can be displayed in prefs */
1008 prefs_themes_sort(PurpleTheme
*theme
)
1010 GdkPixbuf
*pixbuf
= NULL
;
1012 gchar
*image_full
= NULL
, *markup
;
1013 const gchar
*name
, *author
, *description
;
1015 if (PURPLE_IS_SOUND_THEME(theme
)){
1017 image_full
= purple_theme_get_image_full(theme
);
1018 if (image_full
!= NULL
){
1019 pixbuf
= pidgin_pixbuf_new_from_file_at_scale(image_full
, PREFS_OPTIMAL_ICON_SIZE
, PREFS_OPTIMAL_ICON_SIZE
, TRUE
);
1024 gtk_list_store_append(prefs_sound_themes
, &iter
);
1025 gtk_list_store_set(prefs_sound_themes
, &iter
, 0, pixbuf
, 2, purple_theme_get_name(theme
), -1);
1028 g_object_unref(G_OBJECT(pixbuf
));
1030 } else if (PIDGIN_IS_BLIST_THEME(theme
) || PIDGIN_IS_STATUS_ICON_THEME(theme
)){
1031 GtkListStore
*store
;
1033 if (PIDGIN_IS_BLIST_THEME(theme
))
1034 store
= prefs_blist_themes
;
1036 store
= prefs_status_icon_themes
;
1038 image_full
= purple_theme_get_image_full(theme
);
1039 if (image_full
!= NULL
){
1040 pixbuf
= pidgin_pixbuf_new_from_file_at_scale(image_full
, PREFS_OPTIMAL_ICON_SIZE
, PREFS_OPTIMAL_ICON_SIZE
, TRUE
);
1045 name
= purple_theme_get_name(theme
);
1046 author
= purple_theme_get_author(theme
);
1047 description
= purple_theme_get_description(theme
);
1049 markup
= get_theme_markup(name
, FALSE
, author
, description
);
1051 gtk_list_store_append(store
, &iter
);
1052 gtk_list_store_set(store
, &iter
, 0, pixbuf
, 1, markup
, 2, name
, -1);
1056 g_object_unref(G_OBJECT(pixbuf
));
1062 prefs_set_active_theme_combo(GtkWidget
*combo_box
, GtkListStore
*store
, const gchar
*current_theme
)
1065 gchar
*theme
= NULL
;
1066 gboolean unset
= TRUE
;
1068 if (current_theme
&& *current_theme
&& gtk_tree_model_get_iter_first(GTK_TREE_MODEL(store
), &iter
)) {
1070 gtk_tree_model_get(GTK_TREE_MODEL(store
), &iter
, 2, &theme
, -1);
1072 if (purple_strequal(current_theme
, theme
)) {
1073 gtk_combo_box_set_active_iter(GTK_COMBO_BOX(combo_box
), &iter
);
1078 } while (gtk_tree_model_iter_next(GTK_TREE_MODEL(store
), &iter
));
1082 gtk_combo_box_set_active(GTK_COMBO_BOX(combo_box
), 0);
1086 prefs_themes_refresh(void)
1088 GdkPixbuf
*pixbuf
= NULL
;
1092 prefs_sound_themes_loading
= TRUE
;
1093 /* refresh the list of themes in the manager */
1094 purple_theme_manager_refresh();
1096 tmp
= g_build_filename(PURPLE_DATADIR
, "icons", "hicolor", "32x32",
1097 "apps", "im.pidgin.Pidgin3.png", NULL
);
1098 pixbuf
= pidgin_pixbuf_new_from_file_at_scale(tmp
, PREFS_OPTIMAL_ICON_SIZE
, PREFS_OPTIMAL_ICON_SIZE
, TRUE
);
1102 gtk_list_store_clear(prefs_sound_themes
);
1103 gtk_list_store_append(prefs_sound_themes
, &iter
);
1104 gtk_list_store_set(prefs_sound_themes
, &iter
, 0, pixbuf
, 2, "", -1);
1107 gtk_list_store_clear(prefs_blist_themes
);
1108 gtk_list_store_append(prefs_blist_themes
, &iter
);
1109 tmp
= get_theme_markup(_("Default"), FALSE
, _("Penguin Pimps"),
1110 _("The default Pidgin buddy list theme"));
1111 gtk_list_store_set(prefs_blist_themes
, &iter
, 0, pixbuf
, 1, tmp
, 2, "", -1);
1114 /* status icon themes */
1115 gtk_list_store_clear(prefs_status_icon_themes
);
1116 gtk_list_store_append(prefs_status_icon_themes
, &iter
);
1117 tmp
= get_theme_markup(_("Default"), FALSE
, _("Penguin Pimps"),
1118 _("The default Pidgin status icon theme"));
1119 gtk_list_store_set(prefs_status_icon_themes
, &iter
, 0, pixbuf
, 1, tmp
, 2, "", -1);
1122 g_object_unref(G_OBJECT(pixbuf
));
1125 gtk_list_store_clear(prefs_smiley_themes
);
1127 purple_theme_manager_for_each_theme(prefs_themes_sort
);
1128 pref_sound_generate_markup();
1129 smileys_refresh_theme_list();
1132 prefs_set_active_theme_combo(prefs_sound_themes_combo_box
, prefs_sound_themes
, purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/sound/theme"));
1133 prefs_set_active_theme_combo(prefs_blist_themes_combo_box
, prefs_blist_themes
, purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/blist/theme"));
1134 prefs_set_active_theme_combo(prefs_status_themes_combo_box
, prefs_status_icon_themes
, purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/status/icon-theme"));
1135 prefs_set_active_theme_combo(prefs_smiley_themes_combo_box
, prefs_smiley_themes
, purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/smileys/theme"));
1136 prefs_sound_themes_loading
= FALSE
;
1139 /* init all the theme variables so that the themes can be sorted later and used by pref pages */
1141 prefs_themes_init(void)
1143 prefs_sound_themes
= gtk_list_store_new(3, GDK_TYPE_PIXBUF
, G_TYPE_STRING
, G_TYPE_STRING
);
1145 prefs_blist_themes
= gtk_list_store_new(3, GDK_TYPE_PIXBUF
, G_TYPE_STRING
, G_TYPE_STRING
);
1147 prefs_status_icon_themes
= gtk_list_store_new(3, GDK_TYPE_PIXBUF
, G_TYPE_STRING
, G_TYPE_STRING
);
1149 prefs_smiley_themes
= gtk_list_store_new(3, GDK_TYPE_PIXBUF
, G_TYPE_STRING
, G_TYPE_STRING
);
1153 * prefs_theme_find_theme:
1154 * @path: A directory containing a theme. The theme could be at the
1155 * top level of this directory or in any subdirectory thereof.
1156 * @type: The type of theme to load. The loader for this theme type
1157 * will be used and this loader will determine what constitutes a
1160 * Attempt to load the given directory as a theme. If we are unable to
1161 * open the path as a theme then we recurse into path and attempt to
1162 * load each subdirectory that we encounter.
1164 * Returns: A new reference to a #PurpleTheme.
1166 static PurpleTheme
*
1167 prefs_theme_find_theme(const gchar
*path
, const gchar
*type
)
1169 PurpleTheme
*theme
= purple_theme_manager_load_theme(path
, type
);
1170 GDir
*dir
= g_dir_open(path
, 0, NULL
);
1173 while (!PURPLE_IS_THEME(theme
) && (next
= g_dir_read_name(dir
))) {
1174 gchar
*next_path
= g_build_filename(path
, next
, NULL
);
1176 if (g_file_test(next_path
, G_FILE_TEST_IS_DIR
))
1177 theme
= prefs_theme_find_theme(next_path
, type
);
1187 /* Eww. Seriously ewww. But thanks, grim! This is taken from guifications2 */
1189 purple_theme_file_copy(const gchar
*source
, const gchar
*destination
)
1194 if(!(src
= g_fopen(source
, "rb")))
1196 if(!(dest
= g_fopen(destination
, "wb"))) {
1201 while((chr
= fgetc(src
)) != EOF
) {
1212 free_theme_info(struct theme_info
*info
)
1216 g_free(info
->extension
);
1217 g_free(info
->original_name
);
1222 /* installs a theme, info is freed by function */
1224 theme_install_theme(char *path
, struct theme_info
*info
)
1228 gboolean is_smiley_theme
, is_archive
;
1229 PurpleTheme
*theme
= NULL
;
1234 /* check the extension */
1235 tail
= info
->extension
? info
->extension
: strrchr(path
, '.');
1238 free_theme_info(info
);
1242 is_archive
= !g_ascii_strcasecmp(tail
, ".gz") || !g_ascii_strcasecmp(tail
, ".tgz");
1244 /* Just to be safe */
1247 if ((is_smiley_theme
= purple_strequal(info
->type
, "smiley")))
1248 destdir
= g_build_filename(purple_user_dir(), "smileys", NULL
);
1250 destdir
= g_build_filename(purple_user_dir(), "themes", "temp", NULL
);
1252 /* We'll check this just to make sure. This also lets us do something different on
1253 * other platforms, if need be */
1256 gchar
*path_escaped
= g_shell_quote(path
);
1257 gchar
*destdir_escaped
= g_shell_quote(destdir
);
1260 if (!g_file_test(destdir
, G_FILE_TEST_IS_DIR
))
1261 purple_build_dir(destdir
, S_IRUSR
| S_IWUSR
| S_IXUSR
);
1263 command
= g_strdup_printf("tar > /dev/null xzf %s -C %s", path_escaped
, destdir_escaped
);
1264 g_free(path_escaped
);
1265 g_free(destdir_escaped
);
1268 if (system(command
)) {
1269 purple_notify_error(NULL
, NULL
, _("Theme failed to unpack."), NULL
, NULL
);
1272 free_theme_info(info
);
1277 if (!winpidgin_gz_untar(path
, destdir
)) {
1278 purple_notify_error(NULL
, NULL
, _("Theme failed to unpack."), NULL
, NULL
);
1280 free_theme_info(info
);
1286 if (is_smiley_theme
) {
1287 /* just extract the folder to the smiley directory */
1288 prefs_themes_refresh();
1290 } else if (is_archive
) {
1291 theme
= prefs_theme_find_theme(destdir
, info
->type
);
1293 if (PURPLE_IS_THEME(theme
)) {
1294 /* create the location for the theme */
1295 gchar
*theme_dest
= g_build_filename(purple_user_dir(), "themes",
1296 purple_theme_get_name(theme
),
1297 "purple", info
->type
, NULL
);
1299 if (!g_file_test(theme_dest
, G_FILE_TEST_IS_DIR
))
1300 purple_build_dir(theme_dest
, S_IRUSR
| S_IWUSR
| S_IXUSR
);
1303 theme_dest
= g_build_filename(purple_user_dir(), "themes",
1304 purple_theme_get_name(theme
),
1305 "purple", info
->type
, NULL
);
1307 /* move the entire directory to new location */
1308 if (g_rename(purple_theme_get_dir(theme
), theme_dest
)) {
1309 purple_debug_error("gtkprefs", "Error renaming %s to %s: "
1310 "%s\n", purple_theme_get_dir(theme
), theme_dest
,
1315 if (g_remove(destdir
) != 0) {
1316 purple_debug_error("gtkprefs",
1317 "couldn't remove temp (dest) path\n");
1319 g_object_unref(theme
);
1321 prefs_themes_refresh();
1324 /* something was wrong with the theme archive */
1326 purple_notify_error(NULL
, NULL
, _("Theme failed to load."), NULL
, NULL
);
1329 } else { /* just a single file so copy it to a new temp directory and attempt to load it*/
1330 gchar
*temp_path
, *temp_file
;
1332 temp_path
= g_build_filename(purple_user_dir(), "themes", "temp", "sub_folder", NULL
);
1334 if (info
->original_name
!= NULL
) {
1335 /* name was changed from the original (probably a dnd) change it back before loading */
1336 temp_file
= g_build_filename(temp_path
, info
->original_name
, NULL
);
1339 gchar
*source_name
= g_path_get_basename(path
);
1340 temp_file
= g_build_filename(temp_path
, source_name
, NULL
);
1341 g_free(source_name
);
1344 if (!g_file_test(temp_path
, G_FILE_TEST_IS_DIR
))
1345 purple_build_dir(temp_path
, S_IRUSR
| S_IWUSR
| S_IXUSR
);
1347 if (purple_theme_file_copy(path
, temp_file
)) {
1348 /* find the theme, could be in subfolder */
1349 theme
= prefs_theme_find_theme(temp_path
, info
->type
);
1351 if (PURPLE_IS_THEME(theme
)) {
1352 gchar
*theme_dest
= g_build_filename(purple_user_dir(), "themes",
1353 purple_theme_get_name(theme
),
1354 "purple", info
->type
, NULL
);
1356 if(!g_file_test(theme_dest
, G_FILE_TEST_IS_DIR
))
1357 purple_build_dir(theme_dest
, S_IRUSR
| S_IWUSR
| S_IXUSR
);
1359 if (g_rename(purple_theme_get_dir(theme
), theme_dest
)) {
1360 purple_debug_error("gtkprefs", "Error renaming %s to %s: "
1361 "%s\n", purple_theme_get_dir(theme
), theme_dest
,
1366 g_object_unref(theme
);
1368 prefs_themes_refresh();
1370 if (g_remove(temp_path
) != 0) {
1371 purple_debug_error("gtkprefs",
1372 "couldn't remove temp path");
1374 purple_notify_error(NULL
, NULL
, _("Theme failed to load."), NULL
, NULL
);
1377 purple_notify_error(NULL
, NULL
, _("Theme failed to copy."), NULL
, NULL
);
1385 free_theme_info(info
);
1389 theme_got_url(PurpleHttpConnection
*http_conn
, PurpleHttpResponse
*response
,
1392 struct theme_info
*info
= _info
;
1393 const gchar
*themedata
;
1399 g_assert(http_conn
== prefs_themes_running_request
);
1400 prefs_themes_running_request
= NULL
;
1402 if (!purple_http_response_is_successful(response
)) {
1403 free_theme_info(info
);
1407 themedata
= purple_http_response_get_data(response
, &len
);
1409 f
= purple_mkstemp(&path
, TRUE
);
1410 wc
= fwrite(themedata
, len
, 1, f
);
1412 purple_debug_warning("theme_got_url", "Unable to write theme data.\n");
1416 free_theme_info(info
);
1421 theme_install_theme(path
, info
);
1428 theme_dnd_recv(GtkWidget
*widget
, GdkDragContext
*dc
, guint x
, guint y
,
1429 GtkSelectionData
*sd
, guint info
, guint t
, gpointer user_data
)
1431 gchar
*name
= g_strchomp((gchar
*)gtk_selection_data_get_data(sd
));
1433 if ((gtk_selection_data_get_length(sd
) >= 0)
1434 && (gtk_selection_data_get_format(sd
) == 8)) {
1435 /* Well, it looks like the drag event was cool.
1436 * Let's do something with it */
1438 struct theme_info
*info
= g_new0(struct theme_info
, 1);
1439 info
->type
= g_strdup((gchar
*)user_data
);
1440 info
->extension
= g_strdup(g_strrstr(name
,"."));
1441 temp
= g_strrstr(name
, "/");
1442 info
->original_name
= temp
? g_strdup(++temp
) : NULL
;
1444 if (!g_ascii_strncasecmp(name
, "file://", 7)) {
1445 GError
*converr
= NULL
;
1447 /* It looks like we're dealing with a local file. Let's
1448 * just untar it in the right place */
1449 if(!(tmp
= g_filename_from_uri(name
, NULL
, &converr
))) {
1450 purple_debug(PURPLE_DEBUG_ERROR
, "theme dnd", "%s\n",
1451 (converr
? converr
->message
:
1452 "g_filename_from_uri error"));
1453 free_theme_info(info
);
1456 theme_install_theme(tmp
, info
);
1458 } else if (!g_ascii_strncasecmp(name
, "http://", 7) ||
1459 !g_ascii_strncasecmp(name
, "https://", 8)) {
1460 /* Oo, a web drag and drop. This is where things
1461 * will start to get interesting */
1462 PurpleHttpRequest
*hr
;
1463 purple_http_conn_cancel(prefs_themes_running_request
);
1465 hr
= purple_http_request_new(name
);
1466 purple_http_request_set_max_len(hr
,
1467 PREFS_MAX_DOWNLOADED_THEME_SIZE
);
1468 prefs_themes_running_request
= purple_http_request(
1469 NULL
, hr
, theme_got_url
, info
);
1470 purple_http_request_unref(hr
);
1472 free_theme_info(info
);
1474 gtk_drag_finish(dc
, TRUE
, FALSE
, t
);
1477 gtk_drag_finish(dc
, FALSE
, FALSE
, t
);
1480 /* builds a theme combo box from a list store with colums: icon preview, markup, theme name */
1482 prefs_build_theme_combo_box(GtkWidget
*combo_box
, GtkListStore
*store
,
1483 const char *current_theme
, const char *type
)
1485 GtkTargetEntry te
[3] = {
1486 {"text/plain", 0, 0},
1487 {"text/uri-list", 0, 1},
1491 g_return_if_fail(store
!= NULL
&& current_theme
!= NULL
);
1493 gtk_combo_box_set_model(GTK_COMBO_BOX(combo_box
),
1494 GTK_TREE_MODEL(store
));
1496 gtk_drag_dest_set(combo_box
, GTK_DEST_DEFAULT_MOTION
| GTK_DEST_DEFAULT_HIGHLIGHT
| GTK_DEST_DEFAULT_DROP
, te
,
1497 sizeof(te
) / sizeof(GtkTargetEntry
) , GDK_ACTION_COPY
| GDK_ACTION_MOVE
);
1499 g_signal_connect(G_OBJECT(combo_box
), "drag_data_received", G_CALLBACK(theme_dnd_recv
), (gpointer
) type
);
1502 /* sets the current sound theme */
1504 prefs_set_sound_theme_cb(GtkComboBox
*combo_box
, gpointer user_data
)
1506 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(user_data
);
1510 GtkTreeIter new_iter
;
1512 if(gtk_combo_box_get_active_iter(combo_box
, &new_iter
) && !prefs_sound_themes_loading
) {
1514 gtk_tree_model_get(GTK_TREE_MODEL(prefs_sound_themes
), &new_iter
, 2, &new_theme
, -1);
1516 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/sound/theme", new_theme
);
1518 /* New theme removes all customization */
1519 for(i
= 0; i
< PURPLE_NUM_SOUNDS
; i
++){
1520 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
1521 pidgin_sound_get_event_option(i
));
1522 purple_prefs_set_path(pref
, "");
1526 /* gets rid of the "(Custom)" from the last selection */
1527 pref_sound_generate_markup();
1529 gtk_entry_set_text(GTK_ENTRY(win
->sound
.entry
), _("(default)"));
1535 /* sets the current smiley theme */
1537 prefs_set_smiley_theme_cb(GtkComboBox
*combo_box
, gpointer user_data
)
1540 GtkTreeIter new_iter
;
1542 if (gtk_combo_box_get_active_iter(combo_box
, &new_iter
)) {
1544 gtk_tree_model_get(GTK_TREE_MODEL(prefs_smiley_themes
), &new_iter
, 2, &new_theme
, -1);
1546 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/smileys/theme", new_theme
);
1553 /* Does same as normal sort, except "none" is sorted first */
1554 static gint
pidgin_sort_smileys (GtkTreeModel
*model
,
1560 gchar
*name1
= NULL
, *name2
= NULL
;
1562 gtk_tree_model_get(model
, a
, 2, &name1
, -1);
1563 gtk_tree_model_get(model
, b
, 2, &name2
, -1);
1565 if (name1
== NULL
|| name2
== NULL
) {
1566 if (!(name1
== NULL
&& name2
== NULL
))
1567 ret
= (name1
== NULL
) ? -1: 1;
1568 } else if (!g_ascii_strcasecmp(name1
, "none")) {
1569 if (!g_utf8_collate(name1
, name2
))
1572 /* Sort name1 first */
1574 } else if (!g_ascii_strcasecmp(name2
, "none")) {
1575 /* Sort name2 first */
1578 /* Neither string is "none", default to normal sort */
1579 ret
= purple_utf8_strcasecmp(name1
, name2
);
1588 /* sets the current buddy list theme */
1590 prefs_set_blist_theme_cb(GtkComboBox
*combo_box
, gpointer user_data
)
1592 PidginBlistTheme
*theme
= NULL
;
1596 if(gtk_combo_box_get_active_iter(combo_box
, &iter
)) {
1598 gtk_tree_model_get(GTK_TREE_MODEL(prefs_blist_themes
), &iter
, 2, &name
, -1);
1601 theme
= PIDGIN_BLIST_THEME(purple_theme_manager_find_theme(name
, "blist"));
1605 pidgin_blist_set_theme(theme
);
1609 /* sets the current icon theme */
1611 prefs_set_status_icon_theme_cb(GtkComboBox
*combo_box
, gpointer user_data
)
1613 PidginStatusIconTheme
*theme
= NULL
;
1617 if(gtk_combo_box_get_active_iter(combo_box
, &iter
)) {
1619 gtk_tree_model_get(GTK_TREE_MODEL(prefs_status_icon_themes
), &iter
, 2, &name
, -1);
1622 theme
= PIDGIN_STATUS_ICON_THEME(purple_theme_manager_find_theme(name
, "status-icon"));
1626 pidgin_stock_load_status_icon_theme(theme
);
1627 pidgin_blist_refresh(purple_blist_get_default());
1632 bind_theme_page(PidginPrefsWindow
*win
)
1634 /* Buddy List Themes */
1635 prefs_build_theme_combo_box(win
->theme
.blist
, prefs_blist_themes
,
1636 PIDGIN_PREFS_ROOT
"/blist/theme", "blist");
1637 prefs_blist_themes_combo_box
= win
->theme
.blist
;
1639 /* Status Icon Themes */
1640 prefs_build_theme_combo_box(win
->theme
.status
, prefs_status_icon_themes
,
1641 PIDGIN_PREFS_ROOT
"/status/icon-theme",
1643 prefs_status_themes_combo_box
= win
->theme
.status
;
1646 prefs_build_theme_combo_box(win
->theme
.sound
, prefs_sound_themes
,
1647 PIDGIN_PREFS_ROOT
"/sound/theme", "sound");
1648 prefs_sound_themes_combo_box
= win
->theme
.sound
;
1651 prefs_build_theme_combo_box(win
->theme
.smiley
, prefs_smiley_themes
,
1652 PIDGIN_PREFS_ROOT
"/smileys/theme",
1654 prefs_smiley_themes_combo_box
= win
->theme
.smiley
;
1656 /* Custom sort so "none" theme is at top of list */
1657 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(prefs_smiley_themes
),
1658 2, pidgin_sort_smileys
, NULL
, NULL
);
1659 gtk_tree_sortable_set_sort_column_id(GTK_TREE_SORTABLE(prefs_smiley_themes
),
1660 2, GTK_SORT_ASCENDING
);
1664 formatting_toggle_cb(TalkatuActionGroup
*ag
, GAction
*action
, const gchar
*name
, gpointer data
)
1666 gboolean activated
= talkatu_action_group_get_action_activated(ag
, name
);
1667 if(g_ascii_strcasecmp(TALKATU_ACTION_FORMAT_BOLD
, name
) != 0) {
1668 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/conversations/send_bold",
1670 } else if(g_ascii_strcasecmp(TALKATU_ACTION_FORMAT_ITALIC
, name
) != 0) {
1671 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/conversations/send_italic",
1673 } else if(g_ascii_strcasecmp(TALKATU_ACTION_FORMAT_UNDERLINE
, name
) != 0) {
1674 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/conversations/send_underline",
1676 } else if(g_ascii_strcasecmp(TALKATU_ACTION_FORMAT_STRIKETHROUGH
, name
) != 0) {
1677 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/conversations/send_strike",
1683 bind_interface_page(PidginPrefsWindow
*win
)
1685 GList
*names
= NULL
;
1688 win
->iface
.docklet
.type
= PURPLE_PREF_STRING
;
1689 win
->iface
.docklet
.key
= PIDGIN_PREFS_ROOT
"/docklet/show";
1690 pidgin_prefs_bind_dropdown(&win
->iface
.docklet
);
1692 win
->iface
.im
.hide_new
.type
= PURPLE_PREF_STRING
;
1693 win
->iface
.im
.hide_new
.key
= PIDGIN_PREFS_ROOT
"/conversations/im/hide_new";
1694 pidgin_prefs_bind_dropdown(&win
->iface
.im
.hide_new
);
1697 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/win32/minimize_new_convs",
1698 win
->iface
.win32
.minimize_new_convs
);
1700 gtk_widget_hide(win
->iface
.win32
.minimize_new_convs
);
1703 /* All the tab options! */
1704 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/tabs",
1705 win
->iface
.conversations
.tabs
);
1708 * Connect a signal to the above preference. When conversations are not
1709 * shown in a tabbed window then all tabbing options should be disabled.
1711 g_object_bind_property(win
->iface
.conversations
.tabs
, "active",
1712 win
->iface
.conversations
.tabs_vbox
, "sensitive",
1713 G_BINDING_SYNC_CREATE
);
1715 pidgin_prefs_bind_checkbox(
1716 PIDGIN_PREFS_ROOT
"/conversations/close_on_tabs",
1717 win
->iface
.conversations
.close_on_tabs
);
1719 win
->iface
.conversations
.tab_side
.type
= PURPLE_PREF_INT
;
1720 win
->iface
.conversations
.tab_side
.key
= PIDGIN_PREFS_ROOT
"/conversations/tab_side";
1721 pidgin_prefs_bind_dropdown(&win
->iface
.conversations
.tab_side
);
1723 win
->iface
.conversations
.placement
.type
= PURPLE_PREF_STRING
;
1724 win
->iface
.conversations
.placement
.key
= PIDGIN_PREFS_ROOT
"/conversations/placement";
1725 names
= pidgin_conv_placement_get_options();
1726 pidgin_prefs_bind_dropdown_from_list(
1727 &win
->iface
.conversations
.placement
,
1732 /* This is also Win32-specific, but must be visible for Glade binding. */
1734 apply_custom_font(GtkWidget
*unused
, PidginPrefsWindow
*win
)
1736 PangoFontDescription
*desc
= NULL
;
1737 if (!purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/conversations/use_theme_font")) {
1738 const char *font
= purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/conversations/custom_font");
1739 desc
= pango_font_description_from_string(font
);
1742 gtk_widget_override_font(win
->conversations
.format_view
, desc
);
1744 pango_font_description_free(desc
);
1749 pidgin_custom_font_set(GtkWidget
*font_button
, PidginPrefsWindow
*win
)
1752 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/conversations/custom_font",
1753 gtk_font_chooser_get_font(GTK_FONT_CHOOSER(font_button
)));
1755 apply_custom_font(font_button
, win
);
1759 bind_conv_page(PidginPrefsWindow
*win
)
1761 GSimpleActionGroup
*ag
= NULL
;
1763 win
->conversations
.notification_chat
.type
= PURPLE_PREF_INT
;
1764 win
->conversations
.notification_chat
.key
= PIDGIN_PREFS_ROOT
"/conversations/notification_chat";
1765 pidgin_prefs_bind_dropdown(&win
->conversations
.notification_chat
);
1767 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/show_incoming_formatting",
1768 win
->conversations
.show_incoming_formatting
);
1769 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/im/close_immediately",
1770 win
->conversations
.im
.close_immediately
);
1772 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/im/show_buddy_icons",
1773 win
->conversations
.im
.show_buddy_icons
);
1774 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/im/animate_buddy_icons",
1775 win
->conversations
.im
.animate_buddy_icons
);
1776 g_object_bind_property(win
->conversations
.im
.show_buddy_icons
, "active",
1777 win
->conversations
.im
.animate_buddy_icons
, "sensitive",
1778 G_BINDING_SYNC_CREATE
);
1780 pidgin_prefs_bind_checkbox("/purple/conversations/im/send_typing",
1781 win
->conversations
.im
.send_typing
);
1782 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/spellcheck",
1783 win
->conversations
.spellcheck
);
1785 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/conversations/use_smooth_scrolling",
1786 win
->conversations
.use_smooth_scrolling
);
1789 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/win32/blink_im",
1790 win
->conversations
.win32
.blink_im
);
1792 gtk_widget_hide(win
->conversations
.win32
.blink_im
);
1796 /* TODO: it's not implemented */
1797 pidgin_prefs_bind_checkbox(
1798 PIDGIN_PREFS_ROOT
"/conversations/resize_custom_smileys",
1799 win
->conversations
.resize_custom_smileys
);
1801 pidgin_prefs_bind_spin_button(
1802 PIDGIN_PREFS_ROOT
"/conversations/custom_smileys_size",
1803 win
->conversations
.custom_smileys_size
);
1805 g_object_bind_property(win
->conversations
.resize_custom_smileys
, "active",
1806 win
->conversations
.custom_smileys_size
, "sensitive",
1807 G_BINDING_SYNC_CREATE
);
1810 pidgin_prefs_bind_spin_button(
1811 PIDGIN_PREFS_ROOT
"/conversations/minimum_entry_lines",
1812 win
->conversations
.minimum_entry_lines
);
1816 const char *font_name
;
1817 gtk_widget_show(win
->conversations
.font_frame
);
1819 pidgin_prefs_bind_checkbox(
1820 PIDGIN_PREFS_ROOT
"/conversations/use_theme_font",
1821 win
->conversations
.use_theme_font
);
1823 font_name
= purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/conversations/custom_font");
1824 if (font_name
!= NULL
&& *font_name
!= '\0') {
1825 gtk_font_chooser_set_font(
1826 GTK_FONT_CHOOSER(win
->conversations
.custom_font
),
1830 g_object_bind_property(win
->conversations
.use_theme_font
, "active",
1831 win
->conversations
.custom_font_hbox
, "sensitive",
1832 G_BINDING_SYNC_CREATE
|G_BINDING_INVERT_BOOLEAN
);
1836 ag
= talkatu_buffer_get_action_group(TALKATU_BUFFER(win
->conversations
.format_buffer
));
1837 g_signal_connect_after(G_OBJECT(ag
), "action-activated",
1838 G_CALLBACK(formatting_toggle_cb
), NULL
);
1842 network_ip_changed(GtkEntry
*entry
, gpointer data
)
1844 const gchar
*text
= gtk_entry_get_text(entry
);
1845 GtkStyleContext
*context
= gtk_widget_get_style_context(GTK_WIDGET(entry
));
1847 if (text
&& *text
) {
1848 if (purple_ip_address_is_valid(text
)) {
1849 purple_network_set_public_ip(text
);
1850 gtk_style_context_add_class(context
, "good-ip");
1851 gtk_style_context_remove_class(context
, "bad-ip");
1853 gtk_style_context_add_class(context
, "bad-ip");
1854 gtk_style_context_remove_class(context
, "good-ip");
1858 purple_network_set_public_ip("");
1859 gtk_style_context_remove_class(context
, "bad-ip");
1860 gtk_style_context_remove_class(context
, "good-ip");
1865 network_stun_server_changed_cb(GtkWidget
*widget
,
1866 GdkEventFocus
*event
, gpointer data
)
1868 GtkEntry
*entry
= GTK_ENTRY(widget
);
1869 purple_prefs_set_string("/purple/network/stun_server",
1870 gtk_entry_get_text(entry
));
1871 purple_network_set_stun_server(gtk_entry_get_text(entry
));
1877 network_turn_server_changed_cb(GtkWidget
*widget
,
1878 GdkEventFocus
*event
, gpointer data
)
1880 GtkEntry
*entry
= GTK_ENTRY(widget
);
1881 purple_prefs_set_string("/purple/network/turn_server",
1882 gtk_entry_get_text(entry
));
1883 purple_network_set_turn_server(gtk_entry_get_text(entry
));
1889 proxy_changed_cb(const char *name
, PurplePrefType type
,
1890 gconstpointer value
, gpointer data
)
1892 PidginPrefsWindow
*win
= data
;
1893 const char *proxy
= value
;
1895 if (!purple_strequal(proxy
, "none") && !purple_strequal(proxy
, "envvar"))
1896 gtk_widget_show_all(win
->proxy
.options
);
1898 gtk_widget_hide(win
->proxy
.options
);
1902 proxy_print_option(GtkWidget
*entry
, PidginPrefsWindow
*win
)
1904 if (entry
== win
->proxy
.host
) {
1905 purple_prefs_set_string("/purple/proxy/host",
1906 gtk_entry_get_text(GTK_ENTRY(entry
)));
1907 } else if (entry
== win
->proxy
.port
) {
1908 purple_prefs_set_int("/purple/proxy/port",
1909 gtk_spin_button_get_value_as_int(
1910 GTK_SPIN_BUTTON(entry
)));
1911 } else if (entry
== win
->proxy
.username
) {
1912 purple_prefs_set_string("/purple/proxy/username",
1913 gtk_entry_get_text(GTK_ENTRY(entry
)));
1914 } else if (entry
== win
->proxy
.password
) {
1915 purple_prefs_set_string("/purple/proxy/password",
1916 gtk_entry_get_text(GTK_ENTRY(entry
)));
1921 proxy_button_clicked_cb(GtkWidget
*button
, PidginPrefsWindow
*win
)
1925 if (g_spawn_command_line_async(win
->proxy
.gnome_program_path
, &err
))
1928 purple_notify_error(NULL
, NULL
, _("Cannot start proxy configuration program."), err
->message
, NULL
);
1933 browser_button_clicked_cb(GtkWidget
*button
, PidginPrefsWindow
*win
)
1937 if (g_spawn_command_line_async(win
->browser
.gnome_program_path
, &err
))
1940 purple_notify_error(NULL
, NULL
, _("Cannot start browser configuration program."), err
->message
, NULL
);
1945 auto_ip_button_clicked_cb(GtkWidget
*button
, gpointer null
)
1948 PurpleStunNatDiscovery
*stun
;
1951 /* purple_network_get_my_ip will return the IP that was set by the user with
1952 purple_network_set_public_ip, so make a lookup for the auto-detected IP
1955 if (purple_prefs_get_bool("/purple/network/auto_ip")) {
1956 /* Check if STUN discovery was already done */
1957 stun
= purple_stun_discover(NULL
);
1958 if ((stun
!= NULL
) && (stun
->status
== PURPLE_STUN_STATUS_DISCOVERED
)) {
1959 ip
= stun
->publicip
;
1961 /* Attempt to get the IP from a NAT device using UPnP */
1962 ip
= purple_upnp_get_public_ip();
1964 /* Attempt to get the IP from a NAT device using NAT-PMP */
1965 ip
= purple_pmp_get_public_ip();
1967 /* Just fetch the IP of the local system */
1968 ip
= purple_network_get_local_system_ip(-1);
1976 auto_ip_text
= g_strdup_printf(_("Use _automatically detected IP address: %s"), ip
);
1977 gtk_button_set_label(GTK_BUTTON(button
), auto_ip_text
);
1978 g_free(auto_ip_text
);
1982 bind_network_page(PidginPrefsWindow
*win
)
1984 GtkStyleContext
*context
;
1985 GtkCssProvider
*ip_css
;
1986 const gchar ip_style
[] =
1988 "color: @error_fg_color;"
1989 "text-shadow: 0 1px @error_text_shadow;"
1990 "background-image: none;"
1991 "background-color: @error_bg_color;"
1994 "color: @question_fg_color;"
1995 "text-shadow: 0 1px @question_text_shadow;"
1996 "background-image: none;"
1997 "background-color: @success_color;"
2000 gtk_entry_set_text(GTK_ENTRY(win
->network
.stun_server
),
2001 purple_prefs_get_string("/purple/network/stun_server"));
2003 pidgin_prefs_bind_checkbox("/purple/network/auto_ip",
2004 win
->network
.auto_ip
);
2005 auto_ip_button_clicked_cb(win
->network
.auto_ip
, NULL
); /* Update label */
2007 gtk_entry_set_text(GTK_ENTRY(win
->network
.public_ip
),
2008 purple_network_get_public_ip());
2010 ip_css
= gtk_css_provider_new();
2011 gtk_css_provider_load_from_data(ip_css
, ip_style
, -1, NULL
);
2012 context
= gtk_widget_get_style_context(win
->network
.public_ip
);
2013 gtk_style_context_add_provider(context
,
2014 GTK_STYLE_PROVIDER(ip_css
),
2015 GTK_STYLE_PROVIDER_PRIORITY_APPLICATION
);
2017 g_object_bind_property(win
->network
.auto_ip
, "active",
2018 win
->network
.public_ip_hbox
, "sensitive",
2019 G_BINDING_SYNC_CREATE
|G_BINDING_INVERT_BOOLEAN
);
2021 pidgin_prefs_bind_checkbox("/purple/network/map_ports",
2022 win
->network
.map_ports
);
2024 pidgin_prefs_bind_checkbox("/purple/network/ports_range_use",
2025 win
->network
.ports_range_use
);
2026 g_object_bind_property(win
->network
.ports_range_use
, "active",
2027 win
->network
.ports_range_hbox
, "sensitive",
2028 G_BINDING_SYNC_CREATE
);
2030 pidgin_prefs_bind_spin_button("/purple/network/ports_range_start",
2031 win
->network
.ports_range_start
);
2032 pidgin_prefs_bind_spin_button("/purple/network/ports_range_end",
2033 win
->network
.ports_range_end
);
2036 gtk_entry_set_text(GTK_ENTRY(win
->network
.turn_server
),
2037 purple_prefs_get_string("/purple/network/turn_server"));
2039 pidgin_prefs_bind_spin_button("/purple/network/turn_port",
2040 win
->network
.turn_port_udp
);
2042 pidgin_prefs_bind_spin_button("/purple/network/turn_port_tcp",
2043 win
->network
.turn_port_tcp
);
2045 pidgin_prefs_bind_entry("/purple/network/turn_username",
2046 win
->network
.turn_username
);
2047 pidgin_prefs_bind_entry("/purple/network/turn_password",
2048 win
->network
.turn_password
);
2052 manual_browser_set(GtkWidget
*entry
, GdkEventFocus
*event
, gpointer data
)
2054 const char *program
= gtk_entry_get_text(GTK_ENTRY(entry
));
2056 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/browsers/manual_command", program
);
2058 /* carry on normally */
2064 get_available_browsers(void)
2071 /* Sorted reverse alphabetically */
2072 static const struct browser possible_browsers
[] = {
2073 {N_("Seamonkey"), "seamonkey"},
2074 {N_("Opera"), "opera"},
2075 {N_("Mozilla"), "mozilla"},
2076 {N_("Konqueror"), "kfmclient"},
2077 {N_("Google Chrome"), "google-chrome"},
2078 /* Do not move the line below. Code below expects gnome-open to be in
2079 * this list immediately after xdg-open! */
2080 {N_("Desktop Default"), "xdg-open"},
2081 {N_("GNOME Default"), "gnome-open"},
2082 {N_("Galeon"), "galeon"},
2083 {N_("Firefox"), "firefox"},
2084 {N_("Firebird"), "mozilla-firebird"},
2085 {N_("Epiphany"), "epiphany"},
2086 /* Translators: please do not translate "chromium-browser" here! */
2087 {N_("Chromium (chromium-browser)"), "chromium-browser"},
2088 /* Translators: please do not translate "chrome" here! */
2089 {N_("Chromium (chrome)"), "chrome"}
2091 static const int num_possible_browsers
= G_N_ELEMENTS(possible_browsers
);
2093 GList
*browsers
= NULL
;
2095 char *browser_setting
= (char *)purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/browsers/browser");
2097 browsers
= g_list_prepend(browsers
, (gpointer
)"custom");
2098 browsers
= g_list_prepend(browsers
, (gpointer
)_("Manual"));
2100 for (i
= 0; i
< num_possible_browsers
; i
++) {
2101 if (purple_program_is_valid(possible_browsers
[i
].command
)) {
2102 browsers
= g_list_prepend(browsers
,
2103 possible_browsers
[i
].command
);
2104 browsers
= g_list_prepend(browsers
, (gpointer
)_(possible_browsers
[i
].name
));
2105 if(browser_setting
&& purple_strequal(possible_browsers
[i
].command
, browser_setting
))
2106 browser_setting
= NULL
;
2107 /* If xdg-open is valid, prefer it over gnome-open and skip forward */
2108 if(purple_strequal(possible_browsers
[i
].command
, "xdg-open")) {
2109 if (purple_strequal("gnome-open", browser_setting
)) {
2110 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/browsers/browser", possible_browsers
[i
].command
);
2111 browser_setting
= NULL
;
2119 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/browsers/browser", "custom");
2125 browser_changed1_cb(const char *name
, PurplePrefType type
,
2126 gconstpointer value
, gpointer data
)
2128 GtkWidget
*hbox
= data
;
2129 const char *browser
= value
;
2131 gtk_widget_set_sensitive(hbox
, !purple_strequal(browser
, "custom"));
2135 browser_changed2_cb(const char *name
, PurplePrefType type
,
2136 gconstpointer value
, gpointer data
)
2138 GtkWidget
*hbox
= data
;
2139 const char *browser
= value
;
2141 gtk_widget_set_sensitive(hbox
, purple_strequal(browser
, "custom"));
2146 bind_browser_page(PidginPrefsWindow
*win
)
2149 /* We use the registered default browser in windows */
2150 gtk_widget_hide(win
->browser
.page
);
2153 /* if the user is running Mac OS X, hide the browsers tab */
2154 if (purple_running_osx()) {
2155 gtk_widget_hide(win
->browser
.page
);
2156 } else if (purple_running_gnome()) {
2159 gtk_stack_set_visible_child_name(GTK_STACK(win
->browser
.stack
),
2162 path
= g_find_program_in_path("gnome-control-center");
2164 gchar
*tmp
= g_strdup_printf("%s info", path
);
2168 path
= g_find_program_in_path("gnome-default-applications-properties");
2171 win
->browser
.gnome_program_path
= path
;
2172 gtk_widget_set_visible(win
->browser
.gnome_not_found
,
2174 gtk_widget_set_visible(win
->browser
.gnome_program
,
2177 GList
*browsers
= NULL
;
2179 gtk_stack_set_visible_child_name(GTK_STACK(win
->browser
.stack
),
2182 win
->browser
.browser
.type
= PURPLE_PREF_STRING
;
2183 win
->browser
.browser
.key
= PIDGIN_PREFS_ROOT
"/browsers/browser";
2184 browsers
= get_available_browsers();
2185 pidgin_prefs_bind_dropdown_from_list(
2186 &win
->browser
.browser
,
2188 g_list_free(browsers
);
2190 win
->browser
.place
.type
= PURPLE_PREF_INT
;
2191 win
->browser
.place
.key
= PIDGIN_PREFS_ROOT
"/browsers/place";
2192 pidgin_prefs_bind_dropdown(&win
->browser
.place
);
2194 purple_prefs_connect_callback(prefs
,
2195 PIDGIN_PREFS_ROOT
"/browsers/browser",
2196 browser_changed1_cb
,
2197 win
->browser
.place_hbox
);
2199 gtk_entry_set_text(GTK_ENTRY(win
->browser
.manual_command
),
2200 purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/browsers/manual_command"));
2201 purple_prefs_connect_callback(prefs
,
2202 PIDGIN_PREFS_ROOT
"/browsers/browser",
2203 browser_changed2_cb
,
2204 win
->browser
.manual_command_hbox
);
2206 if (purple_strequal(
2207 purple_prefs_get_string(
2208 PIDGIN_PREFS_ROOT
"/browsers/browser"),
2210 gtk_widget_set_sensitive(win
->browser
.place_hbox
,
2213 gtk_widget_set_sensitive(win
->browser
.manual_command_hbox
,
2221 bind_proxy_page(PidginPrefsWindow
*win
)
2223 PurpleProxyInfo
*proxy_info
;
2225 if(purple_running_gnome()) {
2228 gtk_stack_set_visible_child_name(GTK_STACK(win
->proxy
.stack
),
2231 path
= g_find_program_in_path("gnome-network-properties");
2233 path
= g_find_program_in_path("gnome-network-preferences");
2235 path
= g_find_program_in_path("gnome-control-center");
2237 char *tmp
= g_strdup_printf("%s network", path
);
2243 win
->proxy
.gnome_program_path
= path
;
2244 gtk_widget_set_visible(win
->proxy
.gnome_not_found
,
2246 gtk_widget_set_visible(win
->proxy
.gnome_program
,
2249 gtk_stack_set_visible_child_name(GTK_STACK(win
->proxy
.stack
),
2252 /* This is a global option that affects SOCKS4 usage even with
2253 * account-specific proxy settings */
2254 pidgin_prefs_bind_checkbox("/purple/proxy/socks4_remotedns",
2255 win
->proxy
.socks4_remotedns
);
2257 win
->proxy
.type
.type
= PURPLE_PREF_STRING
;
2258 win
->proxy
.type
.key
= "/purple/proxy/type";
2259 pidgin_prefs_bind_dropdown(&win
->proxy
.type
);
2260 proxy_info
= purple_global_proxy_get_info();
2262 purple_prefs_connect_callback(prefs
, "/purple/proxy/type",
2263 proxy_changed_cb
, win
);
2265 if (proxy_info
!= NULL
) {
2266 if (purple_proxy_info_get_host(proxy_info
)) {
2267 gtk_entry_set_text(GTK_ENTRY(win
->proxy
.host
),
2268 purple_proxy_info_get_host(proxy_info
));
2271 if (purple_proxy_info_get_port(proxy_info
) != 0) {
2272 gtk_spin_button_set_value(
2273 GTK_SPIN_BUTTON(win
->proxy
.port
),
2274 purple_proxy_info_get_port(proxy_info
));
2277 if (purple_proxy_info_get_username(proxy_info
) != NULL
) {
2278 gtk_entry_set_text(GTK_ENTRY(win
->proxy
.username
),
2279 purple_proxy_info_get_username(proxy_info
));
2282 if (purple_proxy_info_get_password(proxy_info
) != NULL
) {
2283 gtk_entry_set_text(GTK_ENTRY(win
->proxy
.password
),
2284 purple_proxy_info_get_password(proxy_info
));
2288 proxy_changed_cb("/purple/proxy/type", PURPLE_PREF_STRING
,
2289 purple_prefs_get_string("/purple/proxy/type"),
2295 bind_logging_page(PidginPrefsWindow
*win
)
2299 win
->logging
.format
.type
= PURPLE_PREF_STRING
;
2300 win
->logging
.format
.key
= "/purple/logging/format";
2301 names
= purple_log_logger_get_options();
2302 pidgin_prefs_bind_dropdown_from_list(&win
->logging
.format
, names
);
2305 pidgin_prefs_bind_checkbox("/purple/logging/log_ims",
2306 win
->logging
.log_ims
);
2307 pidgin_prefs_bind_checkbox("/purple/logging/log_chats",
2308 win
->logging
.log_chats
);
2309 pidgin_prefs_bind_checkbox("/purple/logging/log_system",
2310 win
->logging
.log_system
);
2313 /*** keyring page *******************************************************/
2316 keyring_page_settings_changed(GtkWidget
*widget
, gpointer _setting
)
2318 PurpleRequestField
*setting
= _setting
;
2319 PurpleRequestFieldType field_type
;
2321 gtk_widget_set_sensitive(prefs
->keyring
.apply
, TRUE
);
2323 field_type
= purple_request_field_get_field_type(setting
);
2325 if (field_type
== PURPLE_REQUEST_FIELD_BOOLEAN
) {
2326 purple_request_field_bool_set_value(setting
,
2327 gtk_toggle_button_get_active(
2328 GTK_TOGGLE_BUTTON(widget
)));
2329 } else if (field_type
== PURPLE_REQUEST_FIELD_STRING
) {
2330 purple_request_field_string_set_value(setting
,
2331 gtk_entry_get_text(GTK_ENTRY(widget
)));
2332 } else if (field_type
== PURPLE_REQUEST_FIELD_INTEGER
) {
2333 purple_request_field_int_set_value(setting
,
2334 gtk_spin_button_get_value_as_int(
2335 GTK_SPIN_BUTTON(widget
)));
2337 g_return_if_reached();
2341 keyring_page_add_settings_field(GtkBox
*vbox
, PurpleRequestField
*setting
,
2345 PurpleRequestFieldType field_type
;
2348 label
= purple_request_field_get_label(setting
);
2350 field_type
= purple_request_field_get_field_type(setting
);
2351 if (field_type
== PURPLE_REQUEST_FIELD_BOOLEAN
) {
2352 widget
= gtk_check_button_new_with_label(label
);
2354 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(widget
),
2355 purple_request_field_bool_get_value(setting
));
2356 g_signal_connect(G_OBJECT(widget
), "toggled",
2357 G_CALLBACK(keyring_page_settings_changed
), setting
);
2358 } else if (field_type
== PURPLE_REQUEST_FIELD_STRING
) {
2359 widget
= gtk_entry_new();
2360 gtk_entry_set_text(GTK_ENTRY(widget
),
2361 purple_request_field_string_get_value(setting
));
2362 if (purple_request_field_string_is_masked(setting
))
2363 gtk_entry_set_visibility(GTK_ENTRY(widget
), FALSE
);
2364 g_signal_connect(G_OBJECT(widget
), "changed",
2365 G_CALLBACK(keyring_page_settings_changed
), setting
);
2366 } else if (field_type
== PURPLE_REQUEST_FIELD_INTEGER
) {
2367 widget
= gtk_spin_button_new_with_range(
2368 purple_request_field_int_get_lower_bound(setting
),
2369 purple_request_field_int_get_upper_bound(setting
), 1);
2370 gtk_spin_button_set_value(GTK_SPIN_BUTTON(widget
),
2371 purple_request_field_int_get_value(setting
));
2372 g_signal_connect(G_OBJECT(widget
), "value-changed",
2373 G_CALLBACK(keyring_page_settings_changed
), setting
);
2375 purple_debug_error("gtkprefs", "Unsupported field type\n");
2379 pidgin_add_widget_to_vbox(vbox
, label
, sg
, widget
, FALSE
, NULL
);
2382 /* XXX: it could be available for all plugins, not keyrings only */
2384 keyring_page_add_settings(PidginPrefsWindow
*win
)
2390 box
= gtk_box_new(GTK_ORIENTATION_VERTICAL
, PIDGIN_HIG_BOX_SPACE
);
2391 gtk_box_pack_start(GTK_BOX(win
->keyring
.vbox
), box
, FALSE
, FALSE
, 0);
2393 sg
= gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL
);
2395 groups
= purple_request_fields_get_groups(win
->keyring
.settings
);
2396 for (it
= g_list_first(groups
); it
!= NULL
; it
= g_list_next(it
)) {
2397 GList
*it2
, *fields
;
2399 PurpleRequestFieldGroup
*group
;
2400 const gchar
*group_title
;
2403 group_title
= purple_request_field_group_get_title(group
);
2405 vbox
= GTK_BOX(pidgin_make_frame(box
, group_title
));
2407 vbox
= GTK_BOX(box
);
2410 fields
= purple_request_field_group_get_fields(group
);
2411 for (it2
= g_list_first(fields
); it2
!= NULL
;
2412 it2
= g_list_next(it2
)) {
2413 keyring_page_add_settings_field(vbox
, it2
->data
, sg
);
2419 win
->keyring
.settings_box
= box
;
2423 keyring_page_settings_apply(GtkButton
*button
, gpointer data
)
2425 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
2427 if (!purple_keyring_apply_settings(win
, win
->keyring
.settings
)) {
2431 gtk_widget_set_sensitive(win
->keyring
.apply
, FALSE
);
2435 keyring_page_update_settings(PidginPrefsWindow
*win
)
2437 g_clear_pointer(&win
->keyring
.settings
, purple_request_fields_destroy
);
2438 win
->keyring
.settings
= purple_keyring_read_settings();
2439 if (!win
->keyring
.settings
) {
2443 keyring_page_add_settings(win
);
2445 win
->keyring
.apply
= gtk_button_new_with_mnemonic(_("_Apply"));
2446 gtk_box_pack_start(GTK_BOX(win
->keyring
.settings_box
),
2447 win
->keyring
.apply
, FALSE
, FALSE
, 1);
2448 gtk_widget_set_sensitive(win
->keyring
.apply
, FALSE
);
2449 g_signal_connect(G_OBJECT(win
->keyring
.apply
), "clicked",
2450 G_CALLBACK(keyring_page_settings_apply
), win
);
2452 gtk_widget_show_all(win
->keyring
.settings_box
);
2456 keyring_page_pref_set_inuse(GError
*error
, G_GNUC_UNUSED gpointer unused
)
2458 PurpleKeyring
*in_use
= purple_keyring_get_inuse();
2460 if (prefs
== NULL
) {
2461 purple_debug_info("gtkprefs", "pref window already closed\n");
2465 gtk_widget_set_sensitive(GTK_WIDGET(prefs
->keyring
.active
.combo
), TRUE
);
2467 if (error
!= NULL
) {
2468 pidgin_prefs_bind_dropdown_revert_active(
2469 &prefs
->keyring
.active
);
2470 purple_notify_error(NULL
, _("Keyring"),
2471 _("Failed to set new keyring"), error
->message
, NULL
);
2475 g_return_if_fail(in_use
!= NULL
);
2476 purple_prefs_set_string("/purple/keyring/active",
2477 purple_keyring_get_id(in_use
));
2479 keyring_page_update_settings(prefs
);
2483 keyring_page_pref_changed(GtkComboBox
*combo_box
, PidginPrefCombo
*combo
)
2485 const char *keyring_id
;
2486 PurpleKeyring
*keyring
;
2488 g_return_if_fail(combo_box
!= NULL
);
2490 keyring_id
= combo
->value
.string
;
2491 keyring
= purple_keyring_find_keyring_by_id(keyring_id
);
2492 if (keyring
== NULL
) {
2493 pidgin_prefs_bind_dropdown_revert_active(combo
);
2494 purple_notify_error(NULL
, _("Keyring"),
2495 _("Selected keyring is disabled"), NULL
, NULL
);
2499 gtk_widget_set_sensitive(GTK_WIDGET(combo_box
), FALSE
);
2501 g_clear_pointer(&prefs
->keyring
.settings_box
, gtk_widget_destroy
);
2502 g_clear_pointer(&prefs
->keyring
.settings
,
2503 purple_request_fields_destroy
);
2505 purple_keyring_set_inuse(keyring
, FALSE
, keyring_page_pref_set_inuse
,
2510 keyring_page_cleanup(PidginPrefsWindow
*win
)
2512 g_clear_pointer(&win
->keyring
.settings
, purple_request_fields_destroy
);
2516 bind_keyring_page(PidginPrefsWindow
*win
)
2520 /* Keyring selection */
2521 names
= purple_keyring_get_options();
2522 win
->keyring
.active
.type
= PURPLE_PREF_STRING
;
2523 win
->keyring
.active
.key
= "/purple/keyring/active";
2524 pidgin_prefs_bind_dropdown_from_list(&win
->keyring
.active
, names
);
2525 /* Override the usual callback to defer changing the pref. */
2526 win
->keyring
.active
.cb
= keyring_page_pref_changed
;
2529 keyring_page_update_settings(win
);
2532 /*** keyring page - end *************************************************/
2535 sound_method_filter(GtkTreeModel
*model
, GtkTreeIter
*iter
, gpointer data
)
2537 gboolean any
= FALSE
;
2538 gboolean gstreamer
= FALSE
;
2539 gboolean win32
= FALSE
;
2541 gtk_tree_model_get(model
, iter
, 2, &any
, 3, &gstreamer
, 4, &win32
, -1);
2548 #ifdef USE_GSTREAMER
2567 sound_cmd_yeah(GtkEntry
*entry
, gpointer d
)
2569 purple_prefs_set_path(PIDGIN_PREFS_ROOT
"/sound/command",
2570 gtk_entry_get_text(GTK_ENTRY(entry
)));
2575 sound_changed1_cb(const char *name
, PurplePrefType type
,
2576 gconstpointer value
, gpointer data
)
2578 GtkWidget
*hbox
= data
;
2579 const char *method
= value
;
2581 gtk_widget_set_sensitive(hbox
, purple_strequal(method
, "custom"));
2585 sound_changed2_cb(const char *name
, PurplePrefType type
,
2586 gconstpointer value
, gpointer data
)
2588 GtkWidget
*vbox
= data
;
2589 const char *method
= value
;
2591 gtk_widget_set_sensitive(vbox
, !purple_strequal(method
, "none"));
2596 event_toggled(GtkCellRendererToggle
*cell
, gchar
*pth
, gpointer data
)
2598 GtkTreeModel
*model
= (GtkTreeModel
*)data
;
2600 GtkTreePath
*path
= gtk_tree_path_new_from_string(pth
);
2603 gtk_tree_model_get_iter (model
, &iter
, path
);
2604 gtk_tree_model_get (model
, &iter
,
2608 purple_prefs_set_bool(pref
, !gtk_cell_renderer_toggle_get_active(cell
));
2611 gtk_list_store_set(GTK_LIST_STORE (model
), &iter
,
2612 0, !gtk_cell_renderer_toggle_get_active(cell
),
2615 gtk_tree_path_free(path
);
2619 test_sound(GtkWidget
*button
, gpointer i_am_NULL
)
2622 gboolean temp_enabled
;
2625 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/enabled/%s",
2626 pidgin_sound_get_event_option(sound_row_sel
));
2628 temp_enabled
= purple_prefs_get_bool(pref
);
2629 temp_mute
= purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/sound/mute");
2631 if (!temp_enabled
) purple_prefs_set_bool(pref
, TRUE
);
2632 if (temp_mute
) purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/sound/mute", FALSE
);
2634 purple_sound_play_event(sound_row_sel
, NULL
);
2636 if (!temp_enabled
) purple_prefs_set_bool(pref
, FALSE
);
2637 if (temp_mute
) purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/sound/mute", TRUE
);
2643 * Resets a sound file back to default.
2646 reset_sound(GtkWidget
*button
, gpointer data
)
2648 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
2651 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
2652 pidgin_sound_get_event_option(sound_row_sel
));
2653 purple_prefs_set_path(pref
, "");
2656 gtk_entry_set_text(GTK_ENTRY(win
->sound
.entry
), _("(default)"));
2658 pref_sound_generate_markup();
2662 sound_chosen_cb(void *user_data
, const char *filename
)
2667 sound
= GPOINTER_TO_INT(user_data
);
2669 /* Set it -- and forget it */
2670 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
2671 pidgin_sound_get_event_option(sound
));
2672 purple_prefs_set_path(pref
, filename
);
2676 * If the sound we just changed is still the currently selected
2677 * sound, then update the box showing the file name.
2679 if (sound
== sound_row_sel
)
2680 gtk_entry_set_text(GTK_ENTRY(prefs
->sound
.entry
), filename
);
2682 pref_sound_generate_markup();
2686 select_sound(GtkWidget
*button
, gpointer being_NULL_is_fun
)
2689 const char *filename
;
2691 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
2692 pidgin_sound_get_event_option(sound_row_sel
));
2693 filename
= purple_prefs_get_path(pref
);
2696 if (*filename
== '\0')
2699 purple_request_file(prefs
, _("Sound Selection"), filename
, FALSE
,
2700 G_CALLBACK(sound_chosen_cb
), NULL
, NULL
,
2701 GINT_TO_POINTER(sound_row_sel
));
2705 prefs_sound_sel(GtkTreeSelection
*sel
, gpointer data
)
2707 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
2708 GtkTreeModel
*model
;
2714 if (! gtk_tree_selection_get_selected (sel
, &model
, &iter
))
2718 gtk_tree_model_get_value (model
, &iter
, 3, &val
);
2719 sound_row_sel
= g_value_get_uint(&val
);
2721 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
2722 pidgin_sound_get_event_option(sound_row_sel
));
2723 file
= purple_prefs_get_path(pref
);
2725 if (win
->sound
.entry
) {
2726 gtk_entry_set_text(GTK_ENTRY(win
->sound
.entry
),
2727 (file
&& *file
!= '\0') ? file
2730 g_value_unset (&val
);
2732 pref_sound_generate_markup();
2736 bind_sound_page(PidginPrefsWindow
*win
)
2738 GtkTreeModel
*model
;
2739 GtkTreeSelection
*sel
;
2746 win
->sound
.method
.type
= PURPLE_PREF_STRING
;
2747 win
->sound
.method
.key
= PIDGIN_PREFS_ROOT
"/sound/method";
2748 pidgin_prefs_bind_dropdown(&win
->sound
.method
);
2749 model
= gtk_combo_box_get_model(GTK_COMBO_BOX(win
->sound
.method
.combo
));
2750 gtk_tree_model_filter_set_visible_func(GTK_TREE_MODEL_FILTER(model
), sound_method_filter
, NULL
, NULL
);
2751 gtk_tree_model_filter_refilter(GTK_TREE_MODEL_FILTER(model
));
2753 gtk_widget_set_sensitive(
2754 win
->sound
.method_vbox
,
2755 !purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
2758 purple_prefs_connect_callback(prefs
, PIDGIN_PREFS_ROOT
"/sound/method",
2760 win
->sound
.method_vbox
);
2762 gtk_widget_set_sensitive(
2763 win
->sound
.command_hbox
,
2764 purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
2767 purple_prefs_connect_callback(prefs
, PIDGIN_PREFS_ROOT
"/sound/method",
2769 win
->sound
.command_hbox
);
2771 cmd
= purple_prefs_get_path(PIDGIN_PREFS_ROOT
"/sound/command");
2773 gtk_entry_set_text(GTK_ENTRY(win
->sound
.command
), cmd
);
2776 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/sound/mute",
2779 pidgin_prefs_bind_checkbox(PIDGIN_PREFS_ROOT
"/sound/conv_focus",
2780 win
->sound
.conv_focus
);
2782 win
->sound
.while_status
.type
= PURPLE_PREF_INT
;
2783 win
->sound
.while_status
.key
= "/purple/sound/while_status";
2784 pidgin_prefs_bind_dropdown(&win
->sound
.while_status
);
2786 /* SOUND SELECTION */
2787 for (j
=0; j
< PURPLE_NUM_SOUNDS
; j
++) {
2788 char *pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/enabled/%s",
2789 pidgin_sound_get_event_option(j
));
2790 const char *label
= pidgin_sound_get_event_label(j
);
2793 if (label
== NULL
) {
2798 gtk_list_store_append(win
->sound
.event
.store
, &iter
);
2799 gtk_list_store_set(win
->sound
.event
.store
, &iter
,
2800 0, purple_prefs_get_bool(pref
),
2808 sel
= gtk_tree_view_get_selection(GTK_TREE_VIEW(win
->sound
.event
.view
));
2809 path
= gtk_tree_path_new_first();
2810 gtk_tree_selection_select_path(sel
, path
);
2811 gtk_tree_path_free(path
);
2813 pref
= g_strdup_printf(PIDGIN_PREFS_ROOT
"/sound/file/%s",
2814 pidgin_sound_get_event_option(0));
2815 file
= purple_prefs_get_path(pref
);
2817 gtk_entry_set_text(GTK_ENTRY(win
->sound
.entry
),
2818 (file
&& *file
!= '\0') ? file
: _("(default)"));
2823 set_idle_away(PurpleSavedStatus
*status
)
2825 purple_prefs_set_int("/purple/savedstatus/idleaway", purple_savedstatus_get_creation_time(status
));
2829 set_startupstatus(PurpleSavedStatus
*status
)
2831 purple_prefs_set_int("/purple/savedstatus/startup", purple_savedstatus_get_creation_time(status
));
2835 bind_away_page(PidginPrefsWindow
*win
)
2840 win
->away
.idle_reporting
.type
= PURPLE_PREF_STRING
;
2841 win
->away
.idle_reporting
.key
= "/purple/away/idle_reporting";
2842 pidgin_prefs_bind_dropdown(&win
->away
.idle_reporting
);
2844 pidgin_prefs_bind_spin_button("/purple/away/mins_before_away",
2845 win
->away
.mins_before_away
);
2847 pidgin_prefs_bind_checkbox("/purple/away/away_when_idle",
2848 win
->away
.away_when_idle
);
2850 /* TODO: Show something useful if we don't have any saved statuses. */
2851 menu
= pidgin_status_menu(purple_savedstatus_get_idleaway(), G_CALLBACK(set_idle_away
));
2852 gtk_widget_show_all(menu
);
2853 gtk_box_pack_start(GTK_BOX(win
->away
.idle_hbox
), menu
, FALSE
, FALSE
, 0);
2855 g_object_bind_property(win
->away
.away_when_idle
, "active",
2857 G_BINDING_SYNC_CREATE
);
2860 win
->away
.auto_reply
.type
= PURPLE_PREF_STRING
;
2861 win
->away
.auto_reply
.key
= "/purple/away/auto_reply";
2862 pidgin_prefs_bind_dropdown(&win
->away
.auto_reply
);
2864 /* Signon status stuff */
2865 pidgin_prefs_bind_checkbox("/purple/savedstatus/startup_current_status",
2866 win
->away
.startup_current_status
);
2868 /* TODO: Show something useful if we don't have any saved statuses. */
2869 menu
= pidgin_status_menu(purple_savedstatus_get_startup(), G_CALLBACK(set_startupstatus
));
2870 gtk_widget_show_all(menu
);
2871 gtk_box_pack_start(GTK_BOX(win
->away
.startup_hbox
), menu
, FALSE
, FALSE
, 0);
2872 gtk_label_set_mnemonic_widget(GTK_LABEL(win
->away
.startup_label
), menu
);
2873 pidgin_set_accessible_label(menu
, GTK_LABEL(win
->away
.startup_label
));
2874 g_object_bind_property(win
->away
.startup_current_status
, "active",
2875 win
->away
.startup_hbox
, "sensitive",
2876 G_BINDING_SYNC_CREATE
|G_BINDING_INVERT_BOOLEAN
);
2881 get_vv_device_menuitems(PurpleMediaElementType type
)
2883 GList
*result
= NULL
;
2886 i
= purple_media_manager_enumerate_elements(purple_media_manager_get(),
2888 for (; i
; i
= g_list_delete_link(i
, i
)) {
2889 PurpleMediaElementInfo
*info
= i
->data
;
2891 result
= g_list_append(result
,
2892 purple_media_element_info_get_name(info
));
2893 result
= g_list_append(result
,
2894 purple_media_element_info_get_id(info
));
2895 g_object_unref(info
);
2902 create_test_element(PurpleMediaElementType type
)
2904 PurpleMediaElementInfo
*element_info
;
2906 element_info
= purple_media_manager_get_active_element(purple_media_manager_get(), type
);
2908 g_return_val_if_fail(element_info
, NULL
);
2910 return purple_media_element_info_call_create(element_info
,
2915 vv_test_switch_page_cb(GtkStack
*stack
, G_GNUC_UNUSED GParamSpec
*pspec
,
2918 PidginPrefsWindow
*win
= data
;
2920 if (!g_str_equal(gtk_stack_get_visible_child_name(stack
), "vv")) {
2921 /* Disable any running test pipelines. */
2922 gtk_toggle_button_set_active(
2923 GTK_TOGGLE_BUTTON(win
->vv
.voice
.test
), FALSE
);
2924 gtk_toggle_button_set_active(
2925 GTK_TOGGLE_BUTTON(win
->vv
.video
.test
), FALSE
);
2930 create_voice_pipeline(void)
2932 GstElement
*pipeline
;
2933 GstElement
*src
, *sink
;
2938 pipeline
= gst_pipeline_new("voicetest");
2940 src
= create_test_element(PURPLE_MEDIA_ELEMENT_AUDIO
| PURPLE_MEDIA_ELEMENT_SRC
);
2941 sink
= create_test_element(PURPLE_MEDIA_ELEMENT_AUDIO
| PURPLE_MEDIA_ELEMENT_SINK
);
2942 volume
= gst_element_factory_make("volume", "volume");
2943 level
= gst_element_factory_make("level", "level");
2944 valve
= gst_element_factory_make("valve", "valve");
2946 gst_bin_add_many(GST_BIN(pipeline
), src
, volume
, level
, valve
, sink
, NULL
);
2947 gst_element_link_many(src
, volume
, level
, valve
, sink
, NULL
);
2949 purple_debug_info("gtkprefs", "create_voice_pipeline: setting pipeline "
2950 "state to GST_STATE_PLAYING - it may hang here on win32\n");
2951 gst_element_set_state(GST_ELEMENT(pipeline
), GST_STATE_PLAYING
);
2952 purple_debug_info("gtkprefs", "create_voice_pipeline: state is set\n");
2958 on_volume_change_cb(GtkWidget
*w
, gdouble value
, gpointer data
)
2960 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
2963 if (!win
->vv
.voice
.pipeline
) {
2967 volume
= gst_bin_get_by_name(GST_BIN(win
->vv
.voice
.pipeline
), "volume");
2968 g_object_set(volume
, "volume",
2969 gtk_scale_button_get_value(GTK_SCALE_BUTTON(w
)) * 10.0, NULL
);
2973 gst_msg_db_to_percent(GstMessage
*msg
, gchar
*value_name
)
2976 const GValue
*value
;
2980 list
= gst_structure_get_value(gst_message_get_structure(msg
), value_name
);
2981 G_GNUC_BEGIN_IGNORE_DEPRECATIONS
2982 value
= g_value_array_get_nth(g_value_get_boxed(list
), 0);
2983 G_GNUC_END_IGNORE_DEPRECATIONS
2984 value_db
= g_value_get_double(value
);
2985 percent
= pow(10, value_db
/ 20);
2986 return (percent
> 1.0) ? 1.0 : percent
;
2990 gst_bus_cb(GstBus
*bus
, GstMessage
*msg
, gpointer data
)
2992 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
2994 if (GST_MESSAGE_TYPE(msg
) == GST_MESSAGE_ELEMENT
&&
2995 gst_structure_has_name(gst_message_get_structure(msg
), "level")) {
2997 GstElement
*src
= GST_ELEMENT(GST_MESSAGE_SRC(msg
));
2998 gchar
*name
= gst_element_get_name(src
);
3000 if (purple_strequal(name
, "level")) {
3005 percent
= gst_msg_db_to_percent(msg
, "rms");
3006 gtk_progress_bar_set_fraction(
3007 GTK_PROGRESS_BAR(win
->vv
.voice
.level
), percent
);
3009 percent
= gst_msg_db_to_percent(msg
, "decay");
3010 threshold
= gtk_range_get_value(GTK_RANGE(
3011 win
->vv
.voice
.threshold
)) /
3013 valve
= gst_bin_get_by_name(GST_BIN(GST_ELEMENT_PARENT(src
)), "valve");
3014 g_object_set(valve
, "drop", (percent
< threshold
), NULL
);
3015 g_object_set(win
->vv
.voice
.level
, "text",
3016 (percent
< threshold
) ? _("DROP") : " ",
3027 voice_test_destroy_cb(GtkWidget
*w
, gpointer data
)
3029 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
3031 if (!win
->vv
.voice
.pipeline
) {
3035 gst_element_set_state(win
->vv
.voice
.pipeline
, GST_STATE_NULL
);
3036 g_clear_pointer(&win
->vv
.voice
.pipeline
, gst_object_unref
);
3040 enable_voice_test(PidginPrefsWindow
*win
)
3044 win
->vv
.voice
.pipeline
= create_voice_pipeline();
3045 bus
= gst_pipeline_get_bus(GST_PIPELINE(win
->vv
.voice
.pipeline
));
3046 gst_bus_add_signal_watch(bus
);
3047 g_signal_connect(bus
, "message", G_CALLBACK(gst_bus_cb
), win
);
3048 gst_object_unref(bus
);
3052 toggle_voice_test_cb(GtkToggleButton
*test
, gpointer data
)
3054 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
3056 if (gtk_toggle_button_get_active(test
)) {
3057 gtk_widget_set_sensitive(win
->vv
.voice
.level
, TRUE
);
3058 enable_voice_test(win
);
3060 g_signal_connect(win
->vv
.voice
.volume
, "value-changed",
3061 G_CALLBACK(on_volume_change_cb
), win
);
3062 g_signal_connect(test
, "destroy",
3063 G_CALLBACK(voice_test_destroy_cb
), win
);
3065 gtk_progress_bar_set_fraction(
3066 GTK_PROGRESS_BAR(win
->vv
.voice
.level
), 0.0);
3067 gtk_widget_set_sensitive(win
->vv
.voice
.level
, FALSE
);
3068 g_object_disconnect(win
->vv
.voice
.volume
,
3069 "any-signal::value-changed",
3070 G_CALLBACK(on_volume_change_cb
), win
, NULL
);
3071 g_object_disconnect(test
, "any-signal::destroy",
3072 G_CALLBACK(voice_test_destroy_cb
), win
,
3074 voice_test_destroy_cb(NULL
, win
);
3079 volume_changed_cb(GtkScaleButton
*button
, gdouble value
, gpointer data
)
3081 purple_prefs_set_int("/purple/media/audio/volume/input", value
* 100);
3085 threshold_value_changed_cb(GtkScale
*scale
, GtkWidget
*label
)
3090 value
= (int)gtk_range_get_value(GTK_RANGE(scale
));
3091 tmp
= g_strdup_printf(_("Silence threshold: %d%%"), value
);
3092 gtk_label_set_label(GTK_LABEL(label
), tmp
);
3095 purple_prefs_set_int("/purple/media/audio/silence_threshold", value
);
3099 bind_voice_test(PidginPrefsWindow
*win
, GtkBuilder
*builder
)
3107 volume
= gtk_builder_get_object(builder
, "vv.voice.volume");
3108 win
->vv
.voice
.volume
= GTK_WIDGET(volume
);
3109 gtk_scale_button_set_value(GTK_SCALE_BUTTON(volume
),
3110 purple_prefs_get_int("/purple/media/audio/volume/input") / 100.0);
3111 g_signal_connect(volume
, "value-changed",
3112 G_CALLBACK(volume_changed_cb
), NULL
);
3114 label
= gtk_builder_get_object(builder
, "vv.voice.threshold_label");
3115 tmp
= g_strdup_printf(_("Silence threshold: %d%%"),
3116 purple_prefs_get_int("/purple/media/audio/silence_threshold"));
3117 gtk_label_set_text(GTK_LABEL(label
), tmp
);
3120 threshold
= gtk_builder_get_object(builder
, "vv.voice.threshold");
3121 win
->vv
.voice
.threshold
= GTK_WIDGET(threshold
);
3122 gtk_range_set_value(GTK_RANGE(threshold
),
3123 purple_prefs_get_int("/purple/media/audio/silence_threshold"));
3124 g_signal_connect(threshold
, "value-changed",
3125 G_CALLBACK(threshold_value_changed_cb
), label
);
3127 win
->vv
.voice
.level
=
3128 GTK_WIDGET(gtk_builder_get_object(builder
, "vv.voice.level"));
3130 test
= gtk_builder_get_object(builder
, "vv.voice.test");
3131 g_signal_connect(test
, "toggled",
3132 G_CALLBACK(toggle_voice_test_cb
), win
);
3133 win
->vv
.voice
.test
= GTK_WIDGET(test
);
3137 create_video_pipeline(void)
3139 GstElement
*pipeline
;
3140 GstElement
*src
, *sink
;
3141 GstElement
*videoconvert
;
3142 GstElement
*videoscale
;
3144 pipeline
= gst_pipeline_new("videotest");
3145 src
= create_test_element(PURPLE_MEDIA_ELEMENT_VIDEO
| PURPLE_MEDIA_ELEMENT_SRC
);
3146 sink
= create_test_element(PURPLE_MEDIA_ELEMENT_VIDEO
| PURPLE_MEDIA_ELEMENT_SINK
);
3147 videoconvert
= gst_element_factory_make("videoconvert", NULL
);
3148 videoscale
= gst_element_factory_make("videoscale", NULL
);
3150 g_object_set_data(G_OBJECT(pipeline
), "sink", sink
);
3152 gst_bin_add_many(GST_BIN(pipeline
), src
, videoconvert
, videoscale
, sink
,
3154 gst_element_link_many(src
, videoconvert
, videoscale
, sink
, NULL
);
3160 video_test_destroy_cb(GtkWidget
*w
, gpointer data
)
3162 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
3164 if (!win
->vv
.video
.pipeline
) {
3168 gst_element_set_state(win
->vv
.video
.pipeline
, GST_STATE_NULL
);
3169 g_clear_pointer(&win
->vv
.video
.pipeline
, gst_object_unref
);
3173 window_id_cb(GstBus
*bus
, GstMessage
*msg
, gulong window_id
)
3175 if (GST_MESSAGE_TYPE(msg
) != GST_MESSAGE_ELEMENT
||
3176 !gst_is_video_overlay_prepare_window_handle_message(msg
)) {
3180 g_signal_handlers_disconnect_matched(bus
,
3181 G_SIGNAL_MATCH_FUNC
| G_SIGNAL_MATCH_DATA
,
3182 0, 0, NULL
, window_id_cb
,
3183 (gpointer
)window_id
);
3185 gst_video_overlay_set_window_handle(GST_VIDEO_OVERLAY(GST_MESSAGE_SRC(msg
)),
3190 enable_video_test(PidginPrefsWindow
*win
)
3193 GdkWindow
*window
= gtk_widget_get_window(win
->vv
.video
.drawing_area
);
3194 gulong window_id
= 0;
3196 #ifdef GDK_WINDOWING_WIN32
3197 if (GDK_IS_WIN32_WINDOW(window
))
3198 window_id
= GPOINTER_TO_UINT(GDK_WINDOW_HWND(window
));
3201 #ifdef GDK_WINDOWING_X11
3202 if (GDK_IS_X11_WINDOW(window
))
3203 window_id
= gdk_x11_window_get_xid(window
);
3206 #ifdef GDK_WINDOWING_QUARTZ
3207 if (GDK_IS_QUARTZ_WINDOW(window
))
3208 window_id
= (gulong
)gdk_quartz_window_get_nsview(window
);
3211 g_warning("Unsupported GDK backend");
3212 #if !(defined(GDK_WINDOWING_WIN32) \
3213 || defined(GDK_WINDOWING_X11) \
3214 || defined(GDK_WINDOWING_QUARTZ))
3215 # error "Unsupported GDK windowing system"
3218 win
->vv
.video
.pipeline
= create_video_pipeline();
3219 bus
= gst_pipeline_get_bus(GST_PIPELINE(win
->vv
.video
.pipeline
));
3220 gst_bus_set_sync_handler(bus
, gst_bus_sync_signal_handler
, NULL
, NULL
);
3221 g_signal_connect(bus
, "sync-message::element",
3222 G_CALLBACK(window_id_cb
), (gpointer
)window_id
);
3223 gst_object_unref(bus
);
3225 gst_element_set_state(GST_ELEMENT(win
->vv
.video
.pipeline
),
3230 toggle_video_test_cb(GtkToggleButton
*test
, gpointer data
)
3232 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
3234 if (gtk_toggle_button_get_active(test
)) {
3235 enable_video_test(win
);
3236 g_signal_connect(test
, "destroy",
3237 G_CALLBACK(video_test_destroy_cb
), win
);
3239 g_object_disconnect(test
, "any-signal::destroy",
3240 G_CALLBACK(video_test_destroy_cb
), win
,
3242 video_test_destroy_cb(NULL
, win
);
3247 bind_video_test(PidginPrefsWindow
*win
, GtkBuilder
*builder
)
3251 GdkRGBA color
= {0.0, 0.0, 0.0, 1.0};
3253 win
->vv
.video
.drawing_area
= video
= GTK_WIDGET(
3254 gtk_builder_get_object(builder
, "vv.video.test_area"));
3255 gtk_widget_override_background_color(video
, GTK_STATE_FLAG_NORMAL
,
3258 /* In order to enable client shadow decorations, GtkDialog from GTK+ 3.0
3259 * uses ARGB visual which by default gets inherited by its child
3260 * widgets. XVideo adaptors on the other hand often support just depth
3261 * 24 and rendering video through xvimagesink onto a widget inside a
3262 * GtkDialog then results in no visible output.
3264 * This ensures the default system visual of the drawing area doesn't
3265 * get overridden by the widget's parent.
3267 gtk_widget_set_visual(video
, gdk_screen_get_system_visual(
3268 gtk_widget_get_screen(video
)));
3270 test
= gtk_builder_get_object(builder
, "vv.video.test");
3271 g_signal_connect(test
, "toggled",
3272 G_CALLBACK(toggle_video_test_cb
), win
);
3273 win
->vv
.video
.test
= GTK_WIDGET(test
);
3277 vv_device_changed_cb(const gchar
*name
, PurplePrefType type
,
3278 gconstpointer value
, gpointer data
)
3280 PidginPrefsWindow
*win
= PIDGIN_PREFS_WINDOW(data
);
3282 PurpleMediaManager
*manager
;
3283 PurpleMediaElementInfo
*info
;
3285 manager
= purple_media_manager_get();
3286 info
= purple_media_manager_get_element_info(manager
, value
);
3287 purple_media_manager_set_active_element(manager
, info
);
3289 /* Refresh test viewers */
3290 if (strstr(name
, "audio") && win
->vv
.voice
.pipeline
) {
3291 voice_test_destroy_cb(NULL
, win
);
3292 enable_voice_test(win
);
3293 } else if (strstr(name
, "video") && win
->vv
.video
.pipeline
) {
3294 video_test_destroy_cb(NULL
, win
);
3295 enable_video_test(win
);
3300 purple_media_type_to_preference_key(PurpleMediaElementType type
)
3302 if (type
& PURPLE_MEDIA_ELEMENT_AUDIO
) {
3303 if (type
& PURPLE_MEDIA_ELEMENT_SRC
) {
3304 return PIDGIN_PREFS_ROOT
"/vvconfig/audio/src/device";
3305 } else if (type
& PURPLE_MEDIA_ELEMENT_SINK
) {
3306 return PIDGIN_PREFS_ROOT
"/vvconfig/audio/sink/device";
3308 } else if (type
& PURPLE_MEDIA_ELEMENT_VIDEO
) {
3309 if (type
& PURPLE_MEDIA_ELEMENT_SRC
) {
3310 return PIDGIN_PREFS_ROOT
"/vvconfig/video/src/device";
3311 } else if (type
& PURPLE_MEDIA_ELEMENT_SINK
) {
3312 return PIDGIN_PREFS_ROOT
"/vvconfig/video/sink/device";
3320 bind_vv_dropdown(PidginPrefCombo
*combo
, PurpleMediaElementType element_type
)
3322 const gchar
*preference_key
;
3325 preference_key
= purple_media_type_to_preference_key(element_type
);
3326 devices
= get_vv_device_menuitems(element_type
);
3328 if (g_list_find_custom(devices
, purple_prefs_get_string(preference_key
),
3329 (GCompareFunc
)strcmp
) == NULL
)
3331 GList
*next
= g_list_next(devices
);
3333 purple_prefs_set_string(preference_key
, next
->data
);
3336 combo
->type
= PURPLE_PREF_STRING
;
3337 combo
->key
= preference_key
;
3338 pidgin_prefs_bind_dropdown_from_list(combo
, devices
);
3339 g_list_free_full(devices
, g_free
);
3343 bind_vv_frame(PidginPrefsWindow
*win
, PidginPrefCombo
*combo
,
3344 PurpleMediaElementType type
)
3346 bind_vv_dropdown(combo
, type
);
3348 purple_prefs_connect_callback(combo
->combo
,
3349 purple_media_type_to_preference_key(type
),
3350 vv_device_changed_cb
, win
);
3351 g_signal_connect_swapped(combo
->combo
, "destroy",
3352 G_CALLBACK(purple_prefs_disconnect_by_handle
),
3355 g_object_set_data(G_OBJECT(combo
->combo
), "vv_media_type",
3357 g_object_set_data(G_OBJECT(combo
->combo
), "vv_combo", combo
);
3361 device_list_changed_cb(PurpleMediaManager
*manager
, GtkWidget
*widget
)
3363 PidginPrefCombo
*combo
;
3364 PurpleMediaElementType media_type
;
3365 GtkTreeModel
*model
;
3367 combo
= g_object_get_data(G_OBJECT(widget
), "vv_combo");
3368 media_type
= (PurpleMediaElementType
)g_object_get_data(G_OBJECT(widget
),
3371 /* Unbind original connections so we can repopulate the combo box. */
3372 g_object_disconnect(combo
->combo
, "any-signal::changed",
3373 G_CALLBACK(bind_dropdown_set
), combo
, NULL
);
3374 model
= gtk_combo_box_get_model(GTK_COMBO_BOX(combo
->combo
));
3375 gtk_list_store_clear(GTK_LIST_STORE(model
));
3377 bind_vv_dropdown(combo
, media_type
);
3381 vv_page(PidginPrefsWindow
*win
)
3383 GtkBuilder
*builder
;
3385 PurpleMediaManager
*manager
;
3387 builder
= gtk_builder_new_from_resource("/im/pidgin/Pidgin/Prefs/vv.ui");
3388 gtk_builder_set_translation_domain(builder
, PACKAGE
);
3390 ret
= GTK_WIDGET(gtk_builder_get_object(builder
, "vv.page"));
3392 manager
= purple_media_manager_get();
3394 win
->vv
.voice
.input
.combo
= GTK_WIDGET(
3395 gtk_builder_get_object(builder
, "vv.voice.input.combo"));
3396 bind_vv_frame(win
, &win
->vv
.voice
.input
,
3397 PURPLE_MEDIA_ELEMENT_AUDIO
| PURPLE_MEDIA_ELEMENT_SRC
);
3398 g_signal_connect_object(manager
, "elements-changed::audiosrc",
3399 G_CALLBACK(device_list_changed_cb
),
3400 win
->vv
.voice
.input
.combo
, 0);
3402 win
->vv
.voice
.output
.combo
= GTK_WIDGET(
3403 gtk_builder_get_object(builder
, "vv.voice.output.combo"));
3404 bind_vv_frame(win
, &win
->vv
.voice
.output
,
3405 PURPLE_MEDIA_ELEMENT_AUDIO
| PURPLE_MEDIA_ELEMENT_SINK
);
3406 g_signal_connect_object(manager
, "elements-changed::audiosink",
3407 G_CALLBACK(device_list_changed_cb
),
3408 win
->vv
.voice
.output
.combo
, 0);
3410 bind_voice_test(win
, builder
);
3412 win
->vv
.video
.input
.combo
= GTK_WIDGET(
3413 gtk_builder_get_object(builder
, "vv.video.input.combo"));
3414 bind_vv_frame(win
, &win
->vv
.video
.input
,
3415 PURPLE_MEDIA_ELEMENT_VIDEO
| PURPLE_MEDIA_ELEMENT_SRC
);
3416 g_signal_connect_object(manager
, "elements-changed::videosrc",
3417 G_CALLBACK(device_list_changed_cb
),
3418 win
->vv
.video
.input
.combo
, 0);
3420 win
->vv
.video
.output
.combo
= GTK_WIDGET(
3421 gtk_builder_get_object(builder
, "vv.video.output.combo"));
3422 bind_vv_frame(win
, &win
->vv
.video
.output
,
3423 PURPLE_MEDIA_ELEMENT_VIDEO
| PURPLE_MEDIA_ELEMENT_SINK
);
3424 g_signal_connect_object(manager
, "elements-changed::videosink",
3425 G_CALLBACK(device_list_changed_cb
),
3426 win
->vv
.video
.output
.combo
, 0);
3428 bind_video_test(win
, builder
);
3430 g_signal_connect(win
->stack
, "notify::visible-child",
3431 G_CALLBACK(vv_test_switch_page_cb
), win
);
3434 g_object_unref(builder
);
3441 prefs_stack_init(PidginPrefsWindow
*win
)
3444 GtkStack
*stack
= GTK_STACK(win
->stack
);
3448 bind_interface_page(win
);
3449 bind_browser_page(win
);
3450 bind_conv_page(win
);
3451 bind_logging_page(win
);
3452 bind_network_page(win
);
3453 bind_proxy_page(win
);
3454 bind_keyring_page(win
);
3455 bind_sound_page(win
);
3456 bind_away_page(win
);
3457 bind_theme_page(win
);
3460 gtk_container_add_with_properties(GTK_CONTAINER(stack
), vv
, "name",
3461 "vv", "title", _("Voice/Video"),
3468 pidgin_prefs_window_class_init(PidginPrefsWindowClass
*klass
)
3470 GtkWidgetClass
*widget_class
= GTK_WIDGET_CLASS(klass
);
3472 gtk_widget_class_set_template_from_resource(
3474 "/im/pidgin/Pidgin/Prefs/prefs.ui"
3478 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3480 gtk_widget_class_bind_template_callback(widget_class
, delete_prefs
);
3482 /* Interface page */
3483 gtk_widget_class_bind_template_child(
3484 widget_class
, PidginPrefsWindow
,
3485 iface
.docklet
.combo
);
3486 gtk_widget_class_bind_template_child(
3487 widget_class
, PidginPrefsWindow
,
3488 iface
.im
.hide_new
.combo
);
3489 gtk_widget_class_bind_template_child(
3490 widget_class
, PidginPrefsWindow
,
3491 iface
.win32
.minimize_new_convs
);
3492 gtk_widget_class_bind_template_child(
3493 widget_class
, PidginPrefsWindow
,
3494 iface
.conversations
.tabs
);
3495 gtk_widget_class_bind_template_child(
3496 widget_class
, PidginPrefsWindow
,
3497 iface
.conversations
.tabs_vbox
);
3498 gtk_widget_class_bind_template_child(
3499 widget_class
, PidginPrefsWindow
,
3500 iface
.conversations
.close_on_tabs
);
3501 gtk_widget_class_bind_template_child(
3502 widget_class
, PidginPrefsWindow
,
3503 iface
.conversations
.tab_side
.combo
);
3504 gtk_widget_class_bind_template_child(
3505 widget_class
, PidginPrefsWindow
,
3506 iface
.conversations
.placement
.combo
);
3509 gtk_widget_class_bind_template_child(
3510 widget_class
, PidginPrefsWindow
, browser
.page
);
3511 gtk_widget_class_bind_template_child(
3512 widget_class
, PidginPrefsWindow
, browser
.stack
);
3513 gtk_widget_class_bind_template_child(
3514 widget_class
, PidginPrefsWindow
, browser
.gnome_not_found
);
3515 gtk_widget_class_bind_template_child(
3516 widget_class
, PidginPrefsWindow
, browser
.gnome_program
);
3517 gtk_widget_class_bind_template_child(
3518 widget_class
, PidginPrefsWindow
, browser
.browser
.combo
);
3519 gtk_widget_class_bind_template_child(
3520 widget_class
, PidginPrefsWindow
, browser
.place_hbox
);
3521 gtk_widget_class_bind_template_child(
3522 widget_class
, PidginPrefsWindow
, browser
.place
.combo
);
3523 gtk_widget_class_bind_template_child(
3524 widget_class
, PidginPrefsWindow
, browser
.manual_command_hbox
);
3525 gtk_widget_class_bind_template_child(
3526 widget_class
, PidginPrefsWindow
, browser
.manual_command
);
3527 gtk_widget_class_bind_template_callback(widget_class
,
3528 browser_button_clicked_cb
);
3529 gtk_widget_class_bind_template_callback(widget_class
,
3530 manual_browser_set
);
3532 /* Conversations page */
3533 gtk_widget_class_bind_template_child(
3534 widget_class
, PidginPrefsWindow
,
3535 conversations
.notification_chat
.combo
);
3536 gtk_widget_class_bind_template_child(
3537 widget_class
, PidginPrefsWindow
,
3538 conversations
.show_incoming_formatting
);
3539 gtk_widget_class_bind_template_child(
3540 widget_class
, PidginPrefsWindow
,
3541 conversations
.im
.close_immediately
);
3542 gtk_widget_class_bind_template_child(
3543 widget_class
, PidginPrefsWindow
,
3544 conversations
.im
.show_buddy_icons
);
3545 gtk_widget_class_bind_template_child(
3546 widget_class
, PidginPrefsWindow
,
3547 conversations
.im
.animate_buddy_icons
);
3548 gtk_widget_class_bind_template_child(
3549 widget_class
, PidginPrefsWindow
,
3550 conversations
.im
.send_typing
);
3551 gtk_widget_class_bind_template_child(
3552 widget_class
, PidginPrefsWindow
,
3553 conversations
.spellcheck
);
3554 gtk_widget_class_bind_template_child(
3555 widget_class
, PidginPrefsWindow
,
3556 conversations
.use_smooth_scrolling
);
3557 gtk_widget_class_bind_template_child(
3558 widget_class
, PidginPrefsWindow
,
3559 conversations
.win32
.blink_im
);
3560 gtk_widget_class_bind_template_child(
3561 widget_class
, PidginPrefsWindow
,
3562 conversations
.resize_custom_smileys
);
3563 gtk_widget_class_bind_template_child(
3564 widget_class
, PidginPrefsWindow
,
3565 conversations
.custom_smileys_size
);
3566 gtk_widget_class_bind_template_child(
3567 widget_class
, PidginPrefsWindow
,
3568 conversations
.minimum_entry_lines
);
3569 gtk_widget_class_bind_template_child(
3570 widget_class
, PidginPrefsWindow
,
3571 conversations
.format_buffer
);
3572 gtk_widget_class_bind_template_child(
3573 widget_class
, PidginPrefsWindow
,
3574 conversations
.format_view
);
3576 gtk_widget_class_bind_template_child(
3577 widget_class
, PidginPrefsWindow
,
3578 conversations
.font_frame
);
3579 gtk_widget_class_bind_template_child(
3580 widget_class
, PidginPrefsWindow
,
3581 conversations
.use_theme_font
);
3582 gtk_widget_class_bind_template_child(
3583 widget_class
, PidginPrefsWindow
,
3584 conversations
.custom_font_hbox
);
3585 gtk_widget_class_bind_template_child(
3586 widget_class
, PidginPrefsWindow
,
3587 conversations
.custom_font
);
3589 /* Even though Win32-specific, must be bound to avoid Glade warnings. */
3590 gtk_widget_class_bind_template_callback(widget_class
,
3592 gtk_widget_class_bind_template_callback(widget_class
,
3593 pidgin_custom_font_set
);
3596 gtk_widget_class_bind_template_child(
3597 widget_class
, PidginPrefsWindow
, logging
.format
.combo
);
3598 gtk_widget_class_bind_template_child(
3599 widget_class
, PidginPrefsWindow
, logging
.log_ims
);
3600 gtk_widget_class_bind_template_child(
3601 widget_class
, PidginPrefsWindow
, logging
.log_chats
);
3602 gtk_widget_class_bind_template_child(
3603 widget_class
, PidginPrefsWindow
, logging
.log_system
);
3606 gtk_widget_class_bind_template_child(
3607 widget_class
, PidginPrefsWindow
, network
.stun_server
);
3608 gtk_widget_class_bind_template_child(
3609 widget_class
, PidginPrefsWindow
, network
.auto_ip
);
3610 gtk_widget_class_bind_template_child(
3611 widget_class
, PidginPrefsWindow
, network
.public_ip
);
3612 gtk_widget_class_bind_template_child(
3613 widget_class
, PidginPrefsWindow
,
3614 network
.public_ip_hbox
);
3615 gtk_widget_class_bind_template_child(
3616 widget_class
, PidginPrefsWindow
, network
.map_ports
);
3617 gtk_widget_class_bind_template_child(
3618 widget_class
, PidginPrefsWindow
,
3619 network
.ports_range_use
);
3620 gtk_widget_class_bind_template_child(
3621 widget_class
, PidginPrefsWindow
,
3622 network
.ports_range_hbox
);
3623 gtk_widget_class_bind_template_child(
3624 widget_class
, PidginPrefsWindow
,
3625 network
.ports_range_start
);
3626 gtk_widget_class_bind_template_child(
3627 widget_class
, PidginPrefsWindow
,
3628 network
.ports_range_end
);
3629 gtk_widget_class_bind_template_child(
3630 widget_class
, PidginPrefsWindow
, network
.turn_server
);
3631 gtk_widget_class_bind_template_child(
3632 widget_class
, PidginPrefsWindow
,
3633 network
.turn_port_udp
);
3634 gtk_widget_class_bind_template_child(
3635 widget_class
, PidginPrefsWindow
,
3636 network
.turn_port_tcp
);
3637 gtk_widget_class_bind_template_child(
3638 widget_class
, PidginPrefsWindow
,
3639 network
.turn_username
);
3640 gtk_widget_class_bind_template_child(
3641 widget_class
, PidginPrefsWindow
,
3642 network
.turn_password
);
3643 gtk_widget_class_bind_template_callback(widget_class
,
3644 network_stun_server_changed_cb
);
3645 gtk_widget_class_bind_template_callback(widget_class
,
3646 auto_ip_button_clicked_cb
);
3647 gtk_widget_class_bind_template_callback(widget_class
,
3648 network_ip_changed
);
3649 gtk_widget_class_bind_template_callback(widget_class
,
3650 network_turn_server_changed_cb
);
3653 gtk_widget_class_bind_template_child(
3654 widget_class
, PidginPrefsWindow
, proxy
.stack
);
3655 gtk_widget_class_bind_template_child(
3656 widget_class
, PidginPrefsWindow
, proxy
.gnome_not_found
);
3657 gtk_widget_class_bind_template_child(
3658 widget_class
, PidginPrefsWindow
, proxy
.gnome_program
);
3659 gtk_widget_class_bind_template_child(
3660 widget_class
, PidginPrefsWindow
,
3661 proxy
.socks4_remotedns
);
3662 gtk_widget_class_bind_template_child(
3663 widget_class
, PidginPrefsWindow
, proxy
.type
.combo
);
3664 gtk_widget_class_bind_template_child(
3665 widget_class
, PidginPrefsWindow
, proxy
.options
);
3666 gtk_widget_class_bind_template_child(
3667 widget_class
, PidginPrefsWindow
, proxy
.host
);
3668 gtk_widget_class_bind_template_child(
3669 widget_class
, PidginPrefsWindow
, proxy
.port
);
3670 gtk_widget_class_bind_template_child(
3671 widget_class
, PidginPrefsWindow
, proxy
.username
);
3672 gtk_widget_class_bind_template_child(
3673 widget_class
, PidginPrefsWindow
, proxy
.password
);
3674 gtk_widget_class_bind_template_callback(widget_class
,
3675 proxy_button_clicked_cb
);
3676 gtk_widget_class_bind_template_callback(widget_class
,
3677 proxy_print_option
);
3680 gtk_widget_class_bind_template_child(
3681 widget_class
, PidginPrefsWindow
, keyring
.active
.combo
);
3682 gtk_widget_class_bind_template_child(
3683 widget_class
, PidginPrefsWindow
, keyring
.vbox
);
3686 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3687 sound
.method
.combo
);
3688 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3690 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3692 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3693 sound
.command_hbox
);
3694 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3696 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3698 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3699 sound
.while_status
.combo
);
3700 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3702 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3704 gtk_widget_class_bind_template_child(widget_class
, PidginPrefsWindow
,
3706 gtk_widget_class_bind_template_callback(widget_class
, sound_cmd_yeah
);
3707 gtk_widget_class_bind_template_callback(widget_class
, prefs_sound_sel
);
3708 gtk_widget_class_bind_template_callback(widget_class
, event_toggled
);
3709 gtk_widget_class_bind_template_callback(widget_class
, select_sound
);
3710 gtk_widget_class_bind_template_callback(widget_class
, test_sound
);
3711 gtk_widget_class_bind_template_callback(widget_class
, reset_sound
);
3714 gtk_widget_class_bind_template_child(
3715 widget_class
, PidginPrefsWindow
,
3716 away
.idle_reporting
.combo
);
3717 gtk_widget_class_bind_template_child(
3718 widget_class
, PidginPrefsWindow
,
3719 away
.mins_before_away
);
3720 gtk_widget_class_bind_template_child(
3721 widget_class
, PidginPrefsWindow
, away
.away_when_idle
);
3722 gtk_widget_class_bind_template_child(
3723 widget_class
, PidginPrefsWindow
, away
.idle_hbox
);
3724 gtk_widget_class_bind_template_child(
3725 widget_class
, PidginPrefsWindow
,
3726 away
.auto_reply
.combo
);
3727 gtk_widget_class_bind_template_child(
3728 widget_class
, PidginPrefsWindow
,
3729 away
.startup_current_status
);
3730 gtk_widget_class_bind_template_child(
3731 widget_class
, PidginPrefsWindow
, away
.startup_hbox
);
3732 gtk_widget_class_bind_template_child(
3733 widget_class
, PidginPrefsWindow
, away
.startup_label
);
3736 gtk_widget_class_bind_template_child(
3737 widget_class
, PidginPrefsWindow
, theme
.blist
);
3738 gtk_widget_class_bind_template_child(
3739 widget_class
, PidginPrefsWindow
, theme
.status
);
3740 gtk_widget_class_bind_template_child(
3741 widget_class
, PidginPrefsWindow
, theme
.sound
);
3742 gtk_widget_class_bind_template_child(
3743 widget_class
, PidginPrefsWindow
, theme
.smiley
);
3744 gtk_widget_class_bind_template_callback(widget_class
,
3745 prefs_set_blist_theme_cb
);
3746 gtk_widget_class_bind_template_callback(widget_class
,
3747 prefs_set_status_icon_theme_cb
);
3748 gtk_widget_class_bind_template_callback(widget_class
,
3749 prefs_set_sound_theme_cb
);
3750 gtk_widget_class_bind_template_callback(widget_class
,
3751 prefs_set_smiley_theme_cb
);
3755 pidgin_prefs_window_init(PidginPrefsWindow
*win
)
3757 /* copy the preferences to tmp values...
3758 * I liked "take affect immediately" Oh well :-( */
3759 /* (that should have been "effect," right?) */
3761 /* Back to instant-apply! I win! BU-HAHAHA! */
3763 /* Create the window */
3764 gtk_widget_init_template(GTK_WIDGET(win
));
3766 prefs_stack_init(win
);
3768 /* Refresh the list of themes before showing the preferences window */
3769 prefs_themes_refresh();
3773 pidgin_prefs_show(void)
3775 if (prefs
== NULL
) {
3776 prefs
= PIDGIN_PREFS_WINDOW(g_object_new(
3777 pidgin_prefs_window_get_type(), NULL
));
3780 gtk_window_present(GTK_WINDOW(prefs
));
3784 smiley_theme_pref_cb(const char *name
, PurplePrefType type
,
3785 gconstpointer value
, gpointer data
)
3787 const gchar
*theme_name
= value
;
3790 if (purple_strequal(theme_name
, "none")) {
3791 purple_smiley_theme_set_current(NULL
);
3795 /* XXX: could be cached when initializing prefs view */
3796 themes
= pidgin_smiley_theme_get_all();
3798 for (it
= themes
; it
; it
= g_list_next(it
)) {
3799 PidginSmileyTheme
*theme
= it
->data
;
3801 if (!purple_strequal(pidgin_smiley_theme_get_name(theme
), theme_name
))
3804 purple_smiley_theme_set_current(PURPLE_SMILEY_THEME(theme
));
3809 pidgin_prefs_init(void)
3811 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"");
3812 purple_prefs_add_none("/plugins/gtk");
3816 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/browsers");
3817 purple_prefs_add_int(PIDGIN_PREFS_ROOT
"/browsers/place", PIDGIN_BROWSER_DEFAULT
);
3818 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/browsers/manual_command", "");
3819 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/browsers/browser", "xdg-open");
3823 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/plugins");
3824 purple_prefs_add_path_list(PIDGIN_PREFS_ROOT
"/plugins/loaded", NULL
);
3826 /* File locations */
3827 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/filelocations");
3828 purple_prefs_add_path(PIDGIN_PREFS_ROOT
"/filelocations/last_save_folder", "");
3829 purple_prefs_add_path(PIDGIN_PREFS_ROOT
"/filelocations/last_open_folder", "");
3830 purple_prefs_add_path(PIDGIN_PREFS_ROOT
"/filelocations/last_icon_folder", "");
3833 prefs_themes_init();
3836 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/smileys");
3837 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/smileys/theme", "Default");
3839 /* Smiley Callbacks */
3840 purple_prefs_connect_callback(&prefs
, PIDGIN_PREFS_ROOT
"/smileys/theme",
3841 smiley_theme_pref_cb
, NULL
);
3845 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig");
3846 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/audio");
3847 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/audio/src");
3848 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/vvconfig/audio/src/device", "");
3849 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/audio/sink");
3850 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/vvconfig/audio/sink/device", "");
3851 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/video");
3852 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/video/src");
3853 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/vvconfig/video/src/device", "");
3854 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/video");
3855 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/vvconfig/video/sink");
3856 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/vvconfig/video/sink/device", "");
3859 pidgin_prefs_update_old();
3863 pidgin_prefs_update_old(void)
3865 /* Rename some old prefs */
3866 purple_prefs_rename(PIDGIN_PREFS_ROOT
"/logging/log_ims", "/purple/logging/log_ims");
3867 purple_prefs_rename(PIDGIN_PREFS_ROOT
"/logging/log_chats", "/purple/logging/log_chats");
3868 purple_prefs_rename("/purple/conversations/placement",
3869 PIDGIN_PREFS_ROOT
"/conversations/placement");
3871 purple_prefs_rename(PIDGIN_PREFS_ROOT
"/conversations/im/raise_on_events", "/plugins/gtk/X11/notify/method_raise");
3873 purple_prefs_rename_boolean_toggle(PIDGIN_PREFS_ROOT
"/conversations/ignore_colors",
3874 PIDGIN_PREFS_ROOT
"/conversations/show_incoming_formatting");
3877 * This path pref changed to a string, so migrate. I know this will
3878 * break things for and confuse users that use multiple versions with
3879 * the same config directory, but I'm not inclined to want to deal with
3880 * that at the moment. -- rekkanoryo
3882 if (purple_prefs_exists(PIDGIN_PREFS_ROOT
"/browsers/command") &&
3883 purple_prefs_get_pref_type(PIDGIN_PREFS_ROOT
"/browsers/command") ==
3886 const char *str
= purple_prefs_get_path(
3887 PIDGIN_PREFS_ROOT
"/browsers/command");
3888 purple_prefs_set_string(
3889 PIDGIN_PREFS_ROOT
"/browsers/manual_command", str
);
3890 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/browsers/command");
3893 /* Remove some no-longer-used prefs */
3894 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/auto_expand_contacts");
3895 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/button_style");
3896 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/grey_idle_buddies");
3897 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/raise_on_events");
3898 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/show_group_count");
3899 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/show_warning_level");
3900 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/tooltip_delay");
3901 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/x");
3902 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/blist/y");
3903 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/button_type");
3904 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/ctrl_enter_sends");
3905 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/enter_sends");
3906 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/escape_closes");
3907 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/html_shortcuts");
3908 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/icons_on_tabs");
3909 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/send_formatting");
3910 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/show_smileys");
3911 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/show_urls_as_links");
3912 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/smiley_shortcuts");
3913 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/use_custom_bgcolor");
3914 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/use_custom_fgcolor");
3915 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/use_custom_font");
3916 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/use_custom_size");
3917 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/old_tab_complete");
3918 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/tab_completion");
3919 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/im/hide_on_send");
3920 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/color_nicks");
3921 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/raise_on_events");
3922 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/ignore_fonts");
3923 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/ignore_font_sizes");
3924 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/passthrough_unknown_commands");
3925 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/debug/timestamps");
3926 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/idle");
3927 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/logging/individual_logs");
3928 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/sound/signon");
3929 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/sound/silent_signon");
3931 /* Convert old queuing prefs to hide_new 3-way pref. */
3932 if (purple_prefs_exists("/plugins/gtk/docklet/queue_messages") &&
3933 purple_prefs_get_bool("/plugins/gtk/docklet/queue_messages"))
3935 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/conversations/im/hide_new", "always");
3937 else if (purple_prefs_exists(PIDGIN_PREFS_ROOT
"/away/queue_messages") &&
3938 purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/away/queue_messages"))
3940 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/conversations/im/hide_new", "away");
3942 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/away/queue_messages");
3943 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/away");
3944 purple_prefs_remove("/plugins/gtk/docklet/queue_messages");
3946 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/default_width");
3947 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/chat/default_height");
3948 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/im/default_width");
3949 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/conversations/im/default_height");
3950 purple_prefs_rename(PIDGIN_PREFS_ROOT
"/conversations/x",
3951 PIDGIN_PREFS_ROOT
"/conversations/im/x");
3952 purple_prefs_rename(PIDGIN_PREFS_ROOT
"/conversations/y",
3953 PIDGIN_PREFS_ROOT
"/conversations/im/y");
3955 /* Fixup vvconfig plugin prefs */
3956 if (purple_prefs_exists("/plugins/core/vvconfig/audio/src/device")) {
3957 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/vvconfig/audio/src/device",
3958 purple_prefs_get_string("/plugins/core/vvconfig/audio/src/device"));
3960 if (purple_prefs_exists("/plugins/core/vvconfig/audio/sink/device")) {
3961 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/vvconfig/audio/sink/device",
3962 purple_prefs_get_string("/plugins/core/vvconfig/audio/sink/device"));
3964 if (purple_prefs_exists("/plugins/core/vvconfig/video/src/device")) {
3965 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/vvconfig/video/src/device",
3966 purple_prefs_get_string("/plugins/core/vvconfig/video/src/device"));
3968 if (purple_prefs_exists("/plugins/gtk/vvconfig/video/sink/device")) {
3969 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/vvconfig/video/sink/device",
3970 purple_prefs_get_string("/plugins/gtk/vvconfig/video/sink/device"));
3973 purple_prefs_remove("/plugins/core/vvconfig");
3974 purple_prefs_remove("/plugins/gtk/vvconfig");
3976 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/vvconfig/audio/src/plugin");
3977 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/vvconfig/audio/sink/plugin");
3978 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/vvconfig/video/src/plugin");
3979 purple_prefs_remove(PIDGIN_PREFS_ROOT
"/vvconfig/video/sink/plugin");
3982 /* Added in 3.0.0. */
3983 if (purple_prefs_get_int(PIDGIN_PREFS_ROOT
"/browsers/place") == 1) {
3984 /* If the "open link in" pref is set to the old value for "existing
3985 window" then change it to "default." */
3986 purple_prefs_set_int(PIDGIN_PREFS_ROOT
"/browsers/place",
3987 PIDGIN_BROWSER_DEFAULT
);
3990 /* Added in 3.0.0. */
3992 purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/browsers/browser"),
3994 purple_prefs_set_string(PIDGIN_PREFS_ROOT
"/browsers/browser", "xdg-open");
3996 #endif /* !_WIN32 */