2 * System tray icon (aka docklet) plugin for Purple
4 * Copyright (C) 2002-3 Robert McQueen <robot101@debian.org>
5 * Copyright (C) 2003 Herman Bloggs <hermanator12002@yahoo.com>
6 * Inspired by a similar plugin by:
7 * John (J5) Palmieri <johnp@martianrock.com>
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License as
11 * published by the Free Software Foundation; either version 2 of the
12 * License, or (at your option) any later version.
14 * This program is distributed in the hope that it will be useful, but
15 * WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * General Public License for more details.
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
28 #include "conversation.h"
34 #include "gtkaccount.h"
37 #include "gtkplugin.h"
39 #include "gtksavedstatuses.h"
41 #include "gtkstatusbox.h"
43 #include "pidginstock.h"
44 #include "gtkdocklet.h"
45 #include "gtkdialogs.h"
47 #ifndef DOCKLET_TOOLTIP_LINE_LIMIT
48 #define DOCKLET_TOOLTIP_LINE_LIMIT 5
52 static struct docklet_ui_ops
*ui_ops
= NULL
;
53 static PurpleStatusPrimitive status
= PURPLE_STATUS_OFFLINE
;
54 static gboolean pending
= FALSE
;
55 static gboolean connecting
= FALSE
;
56 static gboolean enable_join_chat
= FALSE
;
57 static guint docklet_blinking_timer
= 0;
58 static gboolean visible
= FALSE
;
59 static gboolean visibility_manager
= FALSE
;
61 /**************************************************************************
62 * docklet status and utility functions
63 **************************************************************************/
65 docklet_blink_icon(gpointer data
)
67 static gboolean blinked
= FALSE
;
68 gboolean ret
= FALSE
; /* by default, don't keep blinking */
72 if(pending
&& !connecting
) {
74 if (ui_ops
&& ui_ops
->blank_icon
)
77 pidgin_docklet_update_icon();
79 ret
= TRUE
; /* keep blinking */
81 docklet_blinking_timer
= 0;
89 get_pending_list(guint max
)
93 l_im
= pidgin_conversations_find_unseen_list(PURPLE_CONV_TYPE_IM
,
97 /* Short circuit if we have our information already */
98 if (max
== 1 && l_im
!= NULL
)
101 l_chat
= pidgin_conversations_find_unseen_list(PURPLE_CONV_TYPE_CHAT
,
105 if (l_im
!= NULL
&& l_chat
!= NULL
)
106 return g_list_concat(l_im
, l_chat
);
107 else if (l_im
!= NULL
)
114 docklet_update_status(void)
118 PurpleSavedStatus
*saved_status
;
119 PurpleStatusPrimitive newstatus
= PURPLE_STATUS_OFFLINE
;
120 gboolean newpending
= FALSE
, newconnecting
= FALSE
;
122 /* get the current savedstatus */
123 saved_status
= purple_savedstatus_get_current();
125 /* determine if any ims have unseen messages */
126 convs
= get_pending_list(DOCKLET_TOOLTIP_LINE_LIMIT
);
128 if (purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/docklet/show"), "pending")) {
129 if (convs
&& ui_ops
->create
&& !visible
) {
133 } else if (!convs
&& ui_ops
->destroy
&& visible
) {
147 /* set tooltip if messages are pending */
148 if (ui_ops
->set_tooltip
) {
149 GString
*tooltip_text
= g_string_new("");
150 for (l
= convs
, count
= 0 ; l
!= NULL
; l
= l
->next
, count
++) {
151 PurpleConversation
*conv
= (PurpleConversation
*)l
->data
;
152 PidginConversation
*gtkconv
= PIDGIN_CONVERSATION(conv
);
154 if (count
== DOCKLET_TOOLTIP_LINE_LIMIT
- 1) {
155 g_string_append(tooltip_text
, _("Right-click for more unread messages...\n"));
157 g_string_append_printf(tooltip_text
,
158 ngettext("%d unread message from %s\n", "%d unread messages from %s\n", gtkconv
->unseen_count
),
159 gtkconv
->unseen_count
,
160 purple_conversation_get_title(conv
));
162 g_string_append_printf(tooltip_text
,
163 ngettext("%d unread message from %s\n", "%d unread messages from %s\n",
164 GPOINTER_TO_INT(purple_conversation_get_data(conv
, "unseen-count"))),
165 GPOINTER_TO_INT(purple_conversation_get_data(conv
, "unseen-count")),
166 purple_conversation_get_title(conv
));
170 /* get rid of the last newline */
171 if (tooltip_text
->len
> 0)
172 tooltip_text
= g_string_truncate(tooltip_text
, tooltip_text
->len
- 1);
174 ui_ops
->set_tooltip(tooltip_text
->str
);
176 g_string_free(tooltip_text
, TRUE
);
181 } else if (ui_ops
->set_tooltip
) {
182 char *tooltip_text
= g_strconcat(PIDGIN_NAME
, " - ",
183 purple_savedstatus_get_title(saved_status
), NULL
);
184 ui_ops
->set_tooltip(tooltip_text
);
185 g_free(tooltip_text
);
188 for(l
= purple_accounts_get_all(); l
!= NULL
; l
= l
->next
) {
190 PurpleAccount
*account
= (PurpleAccount
*)l
->data
;
192 if (!purple_account_get_enabled(account
, PIDGIN_UI
))
195 if (purple_account_is_disconnected(account
))
198 if (purple_account_is_connecting(account
))
199 newconnecting
= TRUE
;
202 newstatus
= purple_savedstatus_get_type(saved_status
);
204 /* update the icon if we changed status */
205 if (status
!= newstatus
|| pending
!=newpending
|| connecting
!=newconnecting
) {
207 pending
= newpending
;
208 connecting
= newconnecting
;
210 pidgin_docklet_update_icon();
212 /* and schedule the blinker function if messages are pending */
213 if (purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/docklet/blink")
214 && pending
&& !connecting
&& docklet_blinking_timer
== 0) {
215 docklet_blinking_timer
= g_timeout_add(500, docklet_blink_icon
, NULL
);
219 return FALSE
; /* for when we're called by the glib idle handler */
223 online_account_supports_chat(void)
226 c
= purple_connections_get_all();
229 PurpleConnection
*gc
= c
->data
;
230 PurplePluginProtocolInfo
*prpl_info
= PURPLE_PLUGIN_PROTOCOL_INFO(gc
->prpl
);
231 if (prpl_info
!= NULL
&& prpl_info
->chat_info
!= NULL
)
239 /**************************************************************************
240 * callbacks and signal handlers
241 **************************************************************************/
246 /* TODO: confirm quit while pending */
251 docklet_update_status_cb(void *data
)
253 docklet_update_status();
257 docklet_conv_updated_cb(PurpleConversation
*conv
, PurpleConvUpdateType type
)
259 if (type
== PURPLE_CONV_UPDATE_UNSEEN
)
260 docklet_update_status();
264 docklet_signed_on_cb(PurpleConnection
*gc
)
266 if (!enable_join_chat
) {
267 if (PURPLE_PLUGIN_PROTOCOL_INFO(gc
->prpl
)->chat_info
!= NULL
)
268 enable_join_chat
= TRUE
;
270 docklet_update_status();
274 docklet_signed_off_cb(PurpleConnection
*gc
)
276 if (enable_join_chat
) {
277 if (PURPLE_PLUGIN_PROTOCOL_INFO(gc
->prpl
)->chat_info
!= NULL
)
278 enable_join_chat
= online_account_supports_chat();
280 docklet_update_status();
284 docklet_show_pref_changed_cb(const char *name
, PurplePrefType type
,
285 gconstpointer value
, gpointer data
)
287 const char *val
= value
;
288 if (purple_strequal(val
, "always")) {
289 if (ui_ops
->create
) {
292 else if (!visibility_manager
) {
293 pidgin_blist_visibility_manager_add();
294 visibility_manager
= TRUE
;
297 } else if (purple_strequal(val
, "never")) {
298 if (visible
&& ui_ops
->destroy
)
301 if (visibility_manager
) {
302 pidgin_blist_visibility_manager_remove();
303 visibility_manager
= FALSE
;
305 docklet_update_status();
310 /**************************************************************************
311 * docklet pop-up menu
312 **************************************************************************/
314 docklet_toggle_mute(GtkWidget
*toggle
, void *data
)
316 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/sound/mute", GTK_CHECK_MENU_ITEM(toggle
)->active
);
320 docklet_toggle_blink(GtkWidget
*toggle
, void *data
)
322 purple_prefs_set_bool(PIDGIN_PREFS_ROOT
"/docklet/blink", GTK_CHECK_MENU_ITEM(toggle
)->active
);
326 docklet_toggle_blist(GtkWidget
*toggle
, void *data
)
328 purple_blist_set_visible(GTK_CHECK_MENU_ITEM(toggle
)->active
);
332 /* This is a workaround for a bug in windows GTK+. Clicking outside of the
333 menu does not get rid of it, so instead we get rid of it as soon as the
334 pointer leaves the menu. */
336 hide_docklet_menu(gpointer data
)
339 gtk_menu_popdown(GTK_MENU(data
));
345 docklet_menu_leave_enter(GtkWidget
*menu
, GdkEventCrossing
*event
, void *data
)
347 static guint hide_docklet_timer
= 0;
349 if (event
->type
== GDK_LEAVE_NOTIFY
&& (event
->detail
== GDK_NOTIFY_ANCESTOR
||
350 event
->detail
== GDK_NOTIFY_UNKNOWN
)) {
351 purple_debug(PURPLE_DEBUG_INFO
, "docklet", "menu leave-notify-event\n");
352 /* Add some slop so that the menu doesn't annoyingly disappear when mousing around */
353 if (hide_docklet_timer
== 0) {
354 hide_docklet_timer
= purple_timeout_add(500,
355 hide_docklet_menu
, menu
);
357 } else if (event
->type
== GDK_ENTER_NOTIFY
&& event
->detail
== GDK_NOTIFY_ANCESTOR
) {
358 purple_debug(PURPLE_DEBUG_INFO
, "docklet", "menu enter-notify-event\n");
359 if (hide_docklet_timer
!= 0) {
360 /* Cancel the hiding if we reenter */
362 purple_timeout_remove(hide_docklet_timer
);
363 hide_docklet_timer
= 0;
370 /* There is a lot of code here for handling the status submenu, much of
371 * which is duplicated from the gtkstatusbox. It'd be nice to add API
372 * somewhere to simplify this (either in the statusbox, or in libpurple).
375 show_custom_status_editor_cb(GtkMenuItem
*menuitem
, gpointer user_data
)
377 PurpleSavedStatus
*saved_status
;
378 saved_status
= purple_savedstatus_get_current();
380 if (purple_savedstatus_get_type(saved_status
) == PURPLE_STATUS_AVAILABLE
)
381 saved_status
= purple_savedstatus_new(NULL
, PURPLE_STATUS_AWAY
);
383 pidgin_status_editor_show(FALSE
,
384 purple_savedstatus_is_transient(saved_status
) ? saved_status
: NULL
);
387 static PurpleSavedStatus
*
388 create_transient_status(PurpleStatusPrimitive primitive
, PurpleStatusType
*status_type
)
390 PurpleSavedStatus
*saved_status
= purple_savedstatus_new(NULL
, primitive
);
392 if(status_type
!= NULL
) {
393 GList
*tmp
, *active_accts
= purple_accounts_get_all_active();
394 for (tmp
= active_accts
; tmp
!= NULL
; tmp
= tmp
->next
) {
395 purple_savedstatus_set_substatus(saved_status
,
396 (PurpleAccount
*) tmp
->data
, status_type
, NULL
);
398 g_list_free(active_accts
);
405 activate_status_account_cb(GtkMenuItem
*menuitem
, gpointer user_data
)
407 PurpleStatusType
*status_type
;
408 PurpleStatusPrimitive primitive
;
409 PurpleSavedStatus
*saved_status
= NULL
;
410 GList
*iter
= purple_savedstatuses_get_all();
411 GList
*tmp
, *active_accts
= purple_accounts_get_all_active();
413 status_type
= (PurpleStatusType
*)user_data
;
414 primitive
= purple_status_type_get_primitive(status_type
);
416 for (; iter
!= NULL
; iter
= iter
->next
) {
417 PurpleSavedStatus
*ss
= iter
->data
;
418 if ((purple_savedstatus_get_type(ss
) == primitive
) && purple_savedstatus_is_transient(ss
) &&
419 purple_savedstatus_has_substatuses(ss
))
421 gboolean found
= FALSE
;
422 /* The currently enabled accounts must have substatuses for all the active accts */
423 for(tmp
= active_accts
; tmp
!= NULL
; tmp
= tmp
->next
) {
424 PurpleAccount
*acct
= tmp
->data
;
425 PurpleSavedStatusSub
*sub
= purple_savedstatus_get_substatus(ss
, acct
);
427 const PurpleStatusType
*sub_type
= purple_savedstatus_substatus_get_type(sub
);
428 const char *subtype_status_id
= purple_status_type_get_id(sub_type
);
429 if (subtype_status_id
&& purple_strequal(subtype_status_id
,
430 purple_status_type_get_id(status_type
)))
441 g_list_free(active_accts
);
443 /* Create a new transient saved status if we weren't able to find one */
444 if (saved_status
== NULL
)
445 saved_status
= create_transient_status(primitive
, status_type
);
447 /* Set the status for each account */
448 purple_savedstatus_activate(saved_status
);
452 activate_status_primitive_cb(GtkMenuItem
*menuitem
, gpointer user_data
)
454 PurpleStatusPrimitive primitive
;
455 PurpleSavedStatus
*saved_status
;
457 primitive
= GPOINTER_TO_INT(user_data
);
459 /* Try to lookup an already existing transient saved status */
460 saved_status
= purple_savedstatus_find_transient_by_type_and_message(primitive
, NULL
);
462 /* Create a new transient saved status if we weren't able to find one */
463 if (saved_status
== NULL
)
464 saved_status
= create_transient_status(primitive
, NULL
);
466 /* Set the status for each account */
467 purple_savedstatus_activate(saved_status
);
471 activate_saved_status_cb(GtkMenuItem
*menuitem
, gpointer user_data
)
473 time_t creation_time
;
474 PurpleSavedStatus
*saved_status
;
476 creation_time
= GPOINTER_TO_INT(user_data
);
477 saved_status
= purple_savedstatus_find_by_creation_time(creation_time
);
478 if (saved_status
!= NULL
)
479 purple_savedstatus_activate(saved_status
);
483 new_menu_item_with_status_icon(GtkWidget
*menu
, const char *str
, PurpleStatusPrimitive primitive
, GCallback cb
, gpointer data
, guint accel_key
, guint accel_mods
, char *mod
)
489 menuitem
= gtk_image_menu_item_new_with_label(str
);
492 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
495 g_signal_connect(G_OBJECT(menuitem
), "activate", cb
, data
);
497 pixbuf
= pidgin_create_status_icon(primitive
, menu
, PIDGIN_ICON_SIZE_TANGO_EXTRA_SMALL
);
498 image
= gtk_image_new_from_pixbuf(pixbuf
);
499 g_object_unref(pixbuf
);
500 gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(menuitem
), image
);
502 gtk_widget_show_all(menuitem
);
508 add_account_statuses(GtkWidget
*menu
, PurpleAccount
*account
)
512 for (l
= purple_account_get_status_types(account
); l
!= NULL
; l
= l
->next
) {
513 PurpleStatusType
*status_type
= (PurpleStatusType
*)l
->data
;
514 PurpleStatusPrimitive prim
;
516 if (!purple_status_type_is_user_settable(status_type
))
519 prim
= purple_status_type_get_primitive(status_type
);
521 new_menu_item_with_status_icon(menu
,
522 purple_status_type_get_name(status_type
),
523 prim
, G_CALLBACK(activate_status_account_cb
),
524 status_type
, 0, 0, NULL
);
529 docklet_status_submenu(void)
531 GtkWidget
*submenu
, *menuitem
;
532 GList
*popular_statuses
, *cur
;
533 PidginStatusBox
*statusbox
= NULL
;
535 submenu
= gtk_menu_new();
536 menuitem
= gtk_menu_item_new_with_mnemonic(_("_Change Status"));
537 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem
), submenu
);
539 if(pidgin_blist_get_default_gtk_blist() != NULL
) {
540 statusbox
= PIDGIN_STATUS_BOX(pidgin_blist_get_default_gtk_blist()->statusbox
);
543 if(statusbox
&& statusbox
->account
!= NULL
) {
544 add_account_statuses(submenu
, statusbox
->account
);
545 } else if(statusbox
&& statusbox
->token_status_account
!= NULL
) {
546 add_account_statuses(submenu
, statusbox
->token_status_account
);
548 new_menu_item_with_status_icon(submenu
, _("Available"),
549 PURPLE_STATUS_AVAILABLE
, G_CALLBACK(activate_status_primitive_cb
),
550 GINT_TO_POINTER(PURPLE_STATUS_AVAILABLE
), 0, 0, NULL
);
552 new_menu_item_with_status_icon(submenu
, _("Away"),
553 PURPLE_STATUS_AWAY
, G_CALLBACK(activate_status_primitive_cb
),
554 GINT_TO_POINTER(PURPLE_STATUS_AWAY
), 0, 0, NULL
);
556 new_menu_item_with_status_icon(submenu
, _("Do not disturb"),
557 PURPLE_STATUS_UNAVAILABLE
, G_CALLBACK(activate_status_primitive_cb
),
558 GINT_TO_POINTER(PURPLE_STATUS_UNAVAILABLE
), 0, 0, NULL
);
560 new_menu_item_with_status_icon(submenu
, _("Invisible"),
561 PURPLE_STATUS_INVISIBLE
, G_CALLBACK(activate_status_primitive_cb
),
562 GINT_TO_POINTER(PURPLE_STATUS_INVISIBLE
), 0, 0, NULL
);
564 new_menu_item_with_status_icon(submenu
, _("Offline"),
565 PURPLE_STATUS_OFFLINE
, G_CALLBACK(activate_status_primitive_cb
),
566 GINT_TO_POINTER(PURPLE_STATUS_OFFLINE
), 0, 0, NULL
);
569 popular_statuses
= purple_savedstatuses_get_popular(6);
570 if (popular_statuses
!= NULL
)
571 pidgin_separator(submenu
);
572 for (cur
= popular_statuses
; cur
!= NULL
; cur
= cur
->next
)
574 PurpleSavedStatus
*saved_status
= cur
->data
;
575 time_t creation_time
= purple_savedstatus_get_creation_time(saved_status
);
576 new_menu_item_with_status_icon(submenu
,
577 purple_savedstatus_get_title(saved_status
),
578 purple_savedstatus_get_type(saved_status
), G_CALLBACK(activate_saved_status_cb
),
579 GINT_TO_POINTER(creation_time
), 0, 0, NULL
);
581 g_list_free(popular_statuses
);
583 pidgin_separator(submenu
);
585 pidgin_new_item_from_stock(submenu
, _("New..."), NULL
, G_CALLBACK(show_custom_status_editor_cb
), NULL
, 0, 0, NULL
);
586 pidgin_new_item_from_stock(submenu
, _("Saved..."), NULL
, G_CALLBACK(pidgin_status_window_show
), NULL
, 0, 0, NULL
);
594 plugin_act(GtkObject
*obj
, PurplePluginAction
*pam
)
596 if (pam
&& pam
->callback
)
601 build_plugin_actions(GtkWidget
*menu
, PurplePlugin
*plugin
,
605 PurplePluginAction
*action
= NULL
;
608 actions
= PURPLE_PLUGIN_ACTIONS(plugin
, context
);
610 for (l
= actions
; l
!= NULL
; l
= l
->next
)
614 action
= (PurplePluginAction
*) l
->data
;
615 action
->plugin
= plugin
;
616 action
->context
= context
;
618 menuitem
= gtk_menu_item_new_with_label(action
->label
);
619 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
621 g_signal_connect(G_OBJECT(menuitem
), "activate",
622 G_CALLBACK(plugin_act
), action
);
623 g_object_set_data_full(G_OBJECT(menuitem
), "plugin_action",
625 (GDestroyNotify
)purple_plugin_action_free
);
626 gtk_widget_show(menuitem
);
629 pidgin_separator(menu
);
632 g_list_free(actions
);
637 docklet_plugin_actions(GtkWidget
*menu
)
639 GtkWidget
*menuitem
, *submenu
;
640 PurplePlugin
*plugin
= NULL
;
644 g_return_if_fail(menu
!= NULL
);
646 /* Add a submenu for each plugin with custom actions */
647 for (l
= purple_plugins_get_loaded(); l
; l
= l
->next
) {
648 plugin
= (PurplePlugin
*) l
->data
;
650 if (PURPLE_IS_PROTOCOL_PLUGIN(plugin
))
653 if (!PURPLE_PLUGIN_HAS_ACTIONS(plugin
))
656 menuitem
= gtk_image_menu_item_new_with_label(_(plugin
->info
->name
));
657 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
659 submenu
= gtk_menu_new();
660 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem
), submenu
);
662 build_plugin_actions(submenu
, plugin
, NULL
);
667 pidgin_separator(menu
);
673 static GtkWidget
*menu
= NULL
;
677 gtk_widget_destroy(menu
);
680 menu
= gtk_menu_new();
682 menuitem
= gtk_check_menu_item_new_with_mnemonic(_("Show Buddy _List"));
683 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menuitem
), purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/blist/list_visible"));
684 g_signal_connect(G_OBJECT(menuitem
), "toggled", G_CALLBACK(docklet_toggle_blist
), NULL
);
685 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
687 menuitem
= gtk_menu_item_new_with_mnemonic(_("_Unread Messages"));
690 GtkWidget
*submenu
= gtk_menu_new();
691 GList
*l
= get_pending_list(0);
693 gtk_widget_set_sensitive(menuitem
, FALSE
);
694 purple_debug_warning("docklet",
695 "status indicates messages pending, but no conversations with unseen messages were found.");
697 pidgin_conversations_fill_menu(submenu
, l
);
699 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem
), submenu
);
702 gtk_widget_set_sensitive(menuitem
, FALSE
);
704 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
706 pidgin_separator(menu
);
708 menuitem
= pidgin_new_item_from_stock(menu
, _("New _Message..."), PIDGIN_STOCK_TOOLBAR_MESSAGE_NEW
, G_CALLBACK(pidgin_dialogs_im
), NULL
, 0, 0, NULL
);
709 if (status
== PURPLE_STATUS_OFFLINE
)
710 gtk_widget_set_sensitive(menuitem
, FALSE
);
712 menuitem
= pidgin_new_item_from_stock(menu
, _("Join Chat..."), PIDGIN_STOCK_CHAT
,
713 G_CALLBACK(pidgin_blist_joinchat_show
), NULL
, 0, 0, NULL
);
714 if (status
== PURPLE_STATUS_OFFLINE
)
715 gtk_widget_set_sensitive(menuitem
, FALSE
);
717 menuitem
= docklet_status_submenu();
718 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
720 pidgin_separator(menu
);
722 pidgin_new_item_from_stock(menu
, _("_Accounts"), NULL
, G_CALLBACK(pidgin_accounts_window_show
), NULL
, 0, 0, NULL
);
723 pidgin_new_item_from_stock(menu
, _("Plu_gins"), PIDGIN_STOCK_TOOLBAR_PLUGINS
, G_CALLBACK(pidgin_plugin_dialog_show
), NULL
, 0, 0, NULL
);
724 pidgin_new_item_from_stock(menu
, _("Pr_eferences"), GTK_STOCK_PREFERENCES
, G_CALLBACK(pidgin_prefs_show
), NULL
, 0, 0, NULL
);
726 pidgin_separator(menu
);
728 menuitem
= gtk_check_menu_item_new_with_mnemonic(_("Mute _Sounds"));
729 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menuitem
), purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/sound/mute"));
730 if (purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/sound/method"), "none"))
731 gtk_widget_set_sensitive(GTK_WIDGET(menuitem
), FALSE
);
732 g_signal_connect(G_OBJECT(menuitem
), "toggled", G_CALLBACK(docklet_toggle_mute
), NULL
);
733 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
735 menuitem
= gtk_check_menu_item_new_with_mnemonic(_("_Blink on New Message"));
736 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(menuitem
), purple_prefs_get_bool(PIDGIN_PREFS_ROOT
"/docklet/blink"));
737 g_signal_connect(G_OBJECT(menuitem
), "toggled", G_CALLBACK(docklet_toggle_blink
), NULL
);
738 gtk_menu_shell_append(GTK_MENU_SHELL(menu
), menuitem
);
740 pidgin_separator(menu
);
742 /* add plugin actions */
743 docklet_plugin_actions(menu
);
745 pidgin_new_item_from_stock(menu
, _("_Quit"), GTK_STOCK_QUIT
, G_CALLBACK(purple_core_quit
), NULL
, 0, 0, NULL
);
748 g_signal_connect(menu
, "leave-notify-event", G_CALLBACK(docklet_menu_leave_enter
), NULL
);
749 g_signal_connect(menu
, "enter-notify-event", G_CALLBACK(docklet_menu_leave_enter
), NULL
);
751 gtk_widget_show_all(menu
);
752 gtk_menu_popup(GTK_MENU(menu
), NULL
, NULL
,
753 ui_ops
->position_menu
,
754 NULL
, 0, gtk_get_current_event_time());
757 /**************************************************************************
758 * public api for ui_ops
759 **************************************************************************/
761 pidgin_docklet_update_icon()
763 if (ui_ops
&& ui_ops
->update_icon
)
764 ui_ops
->update_icon(status
, connecting
, pending
);
768 pidgin_docklet_clicked(int button_type
)
770 switch (button_type
) {
773 GList
*l
= get_pending_list(1);
775 pidgin_conv_present_conversation((PurpleConversation
*)l
->data
);
779 pidgin_blist_toggle_visibility();
789 pidgin_docklet_embedded()
791 if (!visibility_manager
792 && !purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/docklet/show"), "pending")) {
793 pidgin_blist_visibility_manager_add();
794 visibility_manager
= TRUE
;
797 docklet_update_status();
798 pidgin_docklet_update_icon();
802 pidgin_docklet_remove()
805 if (visibility_manager
) {
806 pidgin_blist_visibility_manager_remove();
807 visibility_manager
= FALSE
;
809 if (docklet_blinking_timer
) {
810 g_source_remove(docklet_blinking_timer
);
811 docklet_blinking_timer
= 0;
814 status
= PURPLE_STATUS_OFFLINE
;
819 pidgin_docklet_set_ui_ops(struct docklet_ui_ops
*ops
)
825 pidgin_docklet_get_handle()
832 pidgin_docklet_init()
834 void *conn_handle
= purple_connections_get_handle();
835 void *conv_handle
= purple_conversations_get_handle();
836 void *accounts_handle
= purple_accounts_get_handle();
837 void *status_handle
= purple_savedstatuses_get_handle();
838 void *docklet_handle
= pidgin_docklet_get_handle();
840 purple_prefs_add_none(PIDGIN_PREFS_ROOT
"/docklet");
841 purple_prefs_add_bool(PIDGIN_PREFS_ROOT
"/docklet/blink", FALSE
);
842 purple_prefs_add_string(PIDGIN_PREFS_ROOT
"/docklet/show", "always");
843 purple_prefs_connect_callback(docklet_handle
, PIDGIN_PREFS_ROOT
"/docklet/show",
844 docklet_show_pref_changed_cb
, NULL
);
847 if (purple_strequal(purple_prefs_get_string(PIDGIN_PREFS_ROOT
"/docklet/show"), "always") && ui_ops
&& ui_ops
->create
)
850 purple_signal_connect(conn_handle
, "signed-on",
851 docklet_handle
, PURPLE_CALLBACK(docklet_signed_on_cb
), NULL
);
852 purple_signal_connect(conn_handle
, "signed-off",
853 docklet_handle
, PURPLE_CALLBACK(docklet_signed_off_cb
), NULL
);
854 purple_signal_connect(accounts_handle
, "account-connecting",
855 docklet_handle
, PURPLE_CALLBACK(docklet_update_status_cb
), NULL
);
856 purple_signal_connect(conv_handle
, "received-im-msg",
857 docklet_handle
, PURPLE_CALLBACK(docklet_update_status_cb
), NULL
);
858 purple_signal_connect(conv_handle
, "conversation-created",
859 docklet_handle
, PURPLE_CALLBACK(docklet_update_status_cb
), NULL
);
860 purple_signal_connect(conv_handle
, "deleting-conversation",
861 docklet_handle
, PURPLE_CALLBACK(docklet_update_status_cb
), NULL
);
862 purple_signal_connect(conv_handle
, "conversation-updated",
863 docklet_handle
, PURPLE_CALLBACK(docklet_conv_updated_cb
), NULL
);
864 purple_signal_connect(status_handle
, "savedstatus-changed",
865 docklet_handle
, PURPLE_CALLBACK(docklet_update_status_cb
), NULL
);
867 purple_signal_connect(purple_get_core(), "quitting",
868 docklet_handle
, PURPLE_CALLBACK(purple_quit_cb
), NULL
);
871 enable_join_chat
= online_account_supports_chat();
875 pidgin_docklet_uninit()
877 if (visible
&& ui_ops
&& ui_ops
->destroy
)