2 * Copyright (C) 2007-2008 Collabora Ltd.
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18 * Authors: Xavier Claessens <xclaesse@gmail.com>
19 * Sjoerd Simons <sjoerd.simons@collabora.co.uk>
25 #include <glib/gi18n.h>
27 #include <telepathy-glib/util.h>
29 #include <libempathy/empathy-dispatcher.h>
30 #include <libempathy/empathy-contact-factory.h>
31 #include <libempathy/empathy-contact-manager.h>
32 #include <libempathy/empathy-tp-chat.h>
33 #include <libempathy/empathy-tp-call.h>
34 #include <libempathy/empathy-utils.h>
35 #include <libempathy/empathy-call-factory.h>
37 #include <extensions/extensions.h>
39 #include <libempathy-gtk/empathy-images.h>
40 #include <libempathy-gtk/empathy-contact-dialogs.h>
41 #include <libempathy-gtk/empathy-ui-utils.h>
43 #include "empathy-event-manager.h"
44 #include "empathy-main-window.h"
45 #include "empathy-tube-dispatch.h"
47 #define DEBUG_FLAG EMPATHY_DEBUG_DISPATCHER
48 #include <libempathy/empathy-debug.h>
50 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyEventManager)
53 EmpathyEventManager *manager;
54 EmpathyDispatchOperation *operation;
55 gulong approved_handler;
56 gulong claimed_handler;
57 gulong invalidated_handler;
58 /* Remove contact if applicable */
59 EmpathyContact *contact;
60 /* Tube dispatcher if applicable */
61 EmpathyTubeDispatch *tube_dispatch;
62 /* option signal handler */
64 /* optional accept widget */
66 } EventManagerApproval;
69 EmpathyDispatcher *dispatcher;
70 EmpathyContactManager *contact_manager;
72 /* Ongoing approvals */
75 /* voip ringing sound */
78 } EmpathyEventManagerPriv;
80 typedef struct _EventPriv EventPriv;
81 typedef void (*EventFunc) (EventPriv *event);
85 EmpathyEventManager *manager;
86 EventManagerApproval *approval;
99 static guint signals[LAST_SIGNAL];
101 G_DEFINE_TYPE (EmpathyEventManager, empathy_event_manager, G_TYPE_OBJECT);
103 static EmpathyEventManager * manager_singleton = NULL;
105 static EventManagerApproval *
106 event_manager_approval_new (EmpathyEventManager *manager,
107 EmpathyDispatchOperation *operation)
109 EventManagerApproval *result = g_slice_new0 (EventManagerApproval);
110 result->operation = g_object_ref (operation);
111 result->manager = manager;
117 event_manager_approval_free (EventManagerApproval *approval)
119 g_signal_handler_disconnect (approval->operation,
120 approval->approved_handler);
121 g_signal_handler_disconnect (approval->operation,
122 approval->claimed_handler);
123 g_signal_handler_disconnect (approval->operation,
124 approval->invalidated_handler);
125 g_object_unref (approval->operation);
127 if (approval->contact != NULL)
128 g_object_unref (approval->contact);
130 if (approval->tube_dispatch != NULL)
131 g_object_unref (approval->tube_dispatch);
133 if (approval->dialog != NULL)
135 gtk_widget_destroy (approval->dialog);
138 g_slice_free (EventManagerApproval, approval);
141 static void event_remove (EventPriv *event);
144 event_free (EventPriv *event)
146 g_free (event->public.icon_name);
147 g_free (event->public.header);
148 g_free (event->public.message);
150 if (event->public.contact)
152 g_object_unref (event->public.contact);
155 g_slice_free (EventPriv, event);
158 static void event_manager_ringing_finished_cb (ca_context *c, guint id,
159 int error_code, gpointer user_data);
162 event_manager_ringing_timeout_cb (gpointer data)
164 EmpathyEventManager *manager = EMPATHY_EVENT_MANAGER (data);
165 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
167 priv->voip_timeout = 0;
169 empathy_sound_play_full (empathy_main_window_get (),
170 EMPATHY_SOUND_PHONE_INCOMING, event_manager_ringing_finished_cb,
177 event_manager_ringing_idle_cb (gpointer data)
179 EmpathyEventManager *manager = EMPATHY_EVENT_MANAGER (data);
180 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
182 if (priv->ringing > 0)
183 priv->voip_timeout = g_timeout_add (500, event_manager_ringing_timeout_cb,
190 event_manager_ringing_finished_cb (ca_context *c, guint id, int error_code,
193 if (error_code == CA_ERROR_CANCELED)
196 g_idle_add (event_manager_ringing_idle_cb, user_data);
200 event_manager_start_ringing (EmpathyEventManager *manager)
202 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
206 if (priv->ringing == 1)
208 empathy_sound_play_full (empathy_main_window_get (),
209 EMPATHY_SOUND_PHONE_INCOMING, event_manager_ringing_finished_cb,
215 event_manager_stop_ringing (EmpathyEventManager *manager)
217 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
221 if (priv->ringing > 0)
224 empathy_sound_stop (EMPATHY_SOUND_PHONE_INCOMING);
226 if (priv->voip_timeout != 0)
228 g_source_remove (priv->voip_timeout);
229 priv->voip_timeout = 0;
234 event_remove (EventPriv *event)
236 EmpathyEventManagerPriv *priv = GET_PRIV (event->manager);
238 DEBUG ("Removing event %p", event);
239 priv->events = g_slist_remove (priv->events, event);
240 g_signal_emit (event->manager, signals[EVENT_REMOVED], 0, event);
245 event_manager_add (EmpathyEventManager *manager, EmpathyContact *contact,
246 const gchar *icon_name, const gchar *header, const gchar *message,
247 EventManagerApproval *approval, EventFunc func, gpointer user_data)
249 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
252 event = g_slice_new0 (EventPriv);
253 event->public.contact = contact ? g_object_ref (contact) : NULL;
254 event->public.icon_name = g_strdup (icon_name);
255 event->public.header = g_strdup (header);
256 event->public.message = g_strdup (message);
257 event->inhibit = FALSE;
259 event->user_data = user_data;
260 event->manager = manager;
261 event->approval = approval;
263 DEBUG ("Adding event %p", event);
264 priv->events = g_slist_prepend (priv->events, event);
265 g_signal_emit (event->manager, signals[EVENT_ADDED], 0, event);
269 event_channel_process_func (EventPriv *event)
271 empathy_dispatch_operation_approve (event->approval->operation);
275 event_text_channel_process_func (EventPriv *event)
277 EmpathyTpChat *tp_chat;
279 if (event->approval->handler != 0)
281 tp_chat = EMPATHY_TP_CHAT
282 (empathy_dispatch_operation_get_channel_wrapper (event->approval->operation));
284 g_signal_handler_disconnect (tp_chat, event->approval->handler);
285 event->approval->handler = 0;
288 empathy_dispatch_operation_approve (event->approval->operation);
292 event_lookup_by_approval (EmpathyEventManager *manager,
293 EventManagerApproval *approval)
295 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
297 EventPriv *retval = NULL;
299 for (l = priv->events; l; l = l->next)
301 EventPriv *event = l->data;
303 if (event->approval == approval)
314 event_update (EmpathyEventManager *manager, EventPriv *event,
315 const char *icon_name, const char *header, const char *msg)
317 g_free (event->public.icon_name);
318 g_free (event->public.header);
319 g_free (event->public.message);
321 event->public.icon_name = g_strdup (icon_name);
322 event->public.header = g_strdup (header);
323 event->public.message = g_strdup (msg);
325 g_signal_emit (manager, signals[EVENT_UPDATED], 0, event);
329 event_manager_call_window_confirmation_dialog_response_cb (GtkDialog *dialog,
330 gint response, gpointer user_data)
332 EventManagerApproval *approval = user_data;
334 gtk_widget_destroy (approval->dialog);
335 approval->dialog = NULL;
337 if (response != GTK_RESPONSE_ACCEPT)
339 EmpathyTpCall *call =
341 empathy_dispatch_operation_get_channel_wrapper (
342 approval->operation));
345 if (empathy_dispatch_operation_claim (approval->operation))
346 empathy_tp_call_close (call);
347 g_object_unref (call);
352 EmpathyCallFactory *factory = empathy_call_factory_get ();
353 empathy_call_factory_claim_channel (factory, approval->operation);
358 event_channel_process_voip_func (EventPriv *event)
364 if (event->approval->dialog != NULL)
366 gtk_window_present (GTK_WINDOW (event->approval->dialog));
370 dialog = gtk_message_dialog_new (GTK_WINDOW (empathy_main_window_get()),
371 GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
372 GTK_MESSAGE_QUESTION, GTK_BUTTONS_NONE, _("Incoming call"));
373 gtk_message_dialog_format_secondary_text (
374 GTK_MESSAGE_DIALOG (dialog),
375 _("%s is calling you, do you want to answer?"),
376 empathy_contact_get_name (event->approval->contact));
378 gtk_dialog_set_default_response (GTK_DIALOG (dialog),
381 button = gtk_dialog_add_button (GTK_DIALOG (dialog),
382 _("_Reject"), GTK_RESPONSE_REJECT);
383 image = gtk_image_new_from_icon_name (GTK_STOCK_CANCEL,
384 GTK_ICON_SIZE_BUTTON);
385 gtk_button_set_image (GTK_BUTTON (button), image);
387 button = gtk_dialog_add_button (GTK_DIALOG (dialog),
388 _("_Answer"), GTK_RESPONSE_ACCEPT);
390 image = gtk_image_new_from_icon_name (GTK_STOCK_APPLY, GTK_ICON_SIZE_BUTTON);
391 gtk_button_set_image (GTK_BUTTON (button), image);
393 g_signal_connect (dialog, "response",
394 G_CALLBACK (event_manager_call_window_confirmation_dialog_response_cb),
397 gtk_widget_show (dialog);
399 event->approval->dialog = dialog;
403 event_manager_chat_message_received_cb (EmpathyTpChat *tp_chat,
404 EmpathyMessage *message, EventManagerApproval *approval)
406 EmpathyContact *sender;
412 /* try to update the event if it's referring to a chat which is already in the
414 event = event_lookup_by_approval (approval->manager, approval);
416 if (event != NULL && event->inhibit && approval->handler != 0)
418 g_signal_handler_disconnect (tp_chat, approval->handler);
419 approval->handler = 0;
423 sender = empathy_message_get_sender (message);
424 header = g_strdup_printf (_("New message from %s"),
425 empathy_contact_get_name (sender));
426 msg = empathy_message_get_body (message);
428 channel = empathy_tp_chat_get_channel (tp_chat);
431 event_update (approval->manager, event, EMPATHY_IMAGE_NEW_MESSAGE, header, msg);
433 event_manager_add (approval->manager, sender, EMPATHY_IMAGE_NEW_MESSAGE, header,
434 msg, approval, event_text_channel_process_func, NULL);
437 empathy_sound_play (empathy_main_window_get (),
438 EMPATHY_SOUND_CONVERSATION_NEW);
442 event_manager_approval_done (EventManagerApproval *approval)
444 EmpathyEventManagerPriv *priv = GET_PRIV (approval->manager);
447 if (approval->operation != NULL)
451 channel_type = empathy_dispatch_operation_get_channel_type_id (
452 approval->operation);
453 if (channel_type == TP_IFACE_QUARK_CHANNEL_TYPE_STREAMED_MEDIA)
455 event_manager_stop_ringing (approval->manager);
459 priv->approvals = g_slist_remove (priv->approvals, approval);
461 for (l = priv->events; l; l = l->next)
463 EventPriv *event = l->data;
465 if (event->approval == approval)
467 event_remove (event);
472 event_manager_approval_free (approval);
476 event_manager_operation_approved_cb (EmpathyDispatchOperation *operation,
477 EventManagerApproval *approval)
479 event_manager_approval_done (approval);
483 event_manager_operation_claimed_cb (EmpathyDispatchOperation *operation,
484 EventManagerApproval *approval)
486 event_manager_approval_done (approval);
490 event_manager_operation_invalidated_cb (EmpathyDispatchOperation *operation,
491 guint domain, gint code, gchar *message,
492 EventManagerApproval *approval)
494 event_manager_approval_done (approval);
498 event_manager_media_channel_got_name_cb (EmpathyContact *contact,
499 const GError *error, gpointer user_data, GObject *object)
501 EventManagerApproval *approval = user_data;
506 /* FIXME just returning assuming the operation will be invalidated as
511 header = g_strdup_printf (_("Incoming call from %s"),
512 empathy_contact_get_name (contact));
514 event_manager_add (approval->manager,
515 approval->contact, EMPATHY_IMAGE_VOIP, header, NULL,
516 approval, event_channel_process_voip_func, NULL);
519 event_manager_start_ringing (approval->manager);
523 event_manager_media_channel_got_contact (EventManagerApproval *approval)
525 empathy_contact_call_when_ready (approval->contact,
526 EMPATHY_CONTACT_READY_NAME, event_manager_media_channel_got_name_cb,
527 approval, NULL, G_OBJECT (approval->manager));
531 event_manager_media_channel_contact_changed_cb (EmpathyTpCall *call,
532 GParamSpec *param, EventManagerApproval *approval)
534 EmpathyContact *contact;
536 g_object_get (G_OBJECT (call), "contact", &contact, NULL);
541 approval->contact = contact;
542 event_manager_media_channel_got_contact (approval);
546 event_manager_tube_approved_cb (EventPriv *event)
548 empathy_tube_dispatch_handle (event->approval->tube_dispatch);
552 event_manager_add_tube_approval (EventManagerApproval *approval,
553 EmpathyTubeDispatchAbility ability)
555 const gchar *icon_name;
559 header = g_strdup_printf (_("%s is offering you an invitation"),
560 empathy_contact_get_name (approval->contact));
562 if (ability == EMPATHY_TUBE_DISPATCHABILITY_POSSIBLE)
564 icon_name = GTK_STOCK_EXECUTE;
565 msg = _("An external application will be started to handle it.");
569 icon_name = GTK_STOCK_DIALOG_ERROR;
570 msg = _("You don't have the needed external "
571 "application to handle it.");
574 event_manager_add (approval->manager, approval->contact, icon_name, header,
575 msg, approval, event_manager_tube_approved_cb, approval);
578 /* FIXME better sound for incoming tubes ? */
579 empathy_sound_play (empathy_main_window_get (),
580 EMPATHY_SOUND_CONVERSATION_NEW);
584 event_manager_tube_dispatch_ability_cb (GObject *object,
585 GParamSpec *spec, gpointer user_data)
587 EventManagerApproval *approval = (EventManagerApproval *)user_data;
588 EmpathyTubeDispatchAbility dispatchability;
591 empathy_tube_dispatch_is_dispatchable (approval->tube_dispatch);
593 if (dispatchability != EMPATHY_TUBE_DISPATCHABILITY_UNKNOWN)
595 event_manager_add_tube_approval (approval, dispatchability);
596 g_signal_handler_disconnect (object, approval->handler);
597 approval->handler = 0;
602 event_manager_tube_got_contact_name_cb (EmpathyContact *contact,
603 const GError *error, gpointer user_data, GObject *object)
605 EventManagerApproval *approval = (EventManagerApproval *)user_data;
606 EmpathyTubeDispatchAbility dispatchability;
610 /* FIXME?, we assume that the operation gets invalidated as well (if it
615 dispatchability = empathy_tube_dispatch_is_dispatchable
616 (approval->tube_dispatch);
619 switch (dispatchability)
621 case EMPATHY_TUBE_DISPATCHABILITY_UNKNOWN:
622 approval->handler = g_signal_connect (approval->tube_dispatch,
623 "notify::dispatchability",
624 G_CALLBACK (event_manager_tube_dispatch_ability_cb), approval);
626 case EMPATHY_TUBE_DISPATCHABILITY_POSSIBLE:
628 case EMPATHY_TUBE_DISPATCHABILITY_IMPOSSIBLE:
629 event_manager_add_tube_approval (approval, dispatchability);
635 invite_dialog_response_cb (GtkDialog *dialog,
637 EventManagerApproval *approval)
639 EmpathyTpChat *tp_chat;
641 TpHandle self_handle;
644 gtk_widget_destroy (GTK_WIDGET (approval->dialog));
645 approval->dialog = NULL;
647 tp_chat = EMPATHY_TP_CHAT (empathy_dispatch_operation_get_channel_wrapper (
648 approval->operation));
650 if (response != GTK_RESPONSE_OK)
653 DEBUG ("Muc invitation rejected");
655 if (empathy_dispatch_operation_claim (approval->operation))
656 empathy_tp_chat_close (tp_chat);
660 DEBUG ("Muc invitation accepted");
663 channel = empathy_tp_chat_get_channel (tp_chat);
665 self_handle = tp_channel_group_get_self_handle (channel);
666 members = g_array_sized_new (FALSE, FALSE, sizeof (TpHandle), 1);
667 g_array_append_val (members, self_handle);
669 tp_cli_channel_interface_group_call_add_members (channel, -1, members,
670 "", NULL, NULL, NULL, NULL);
672 empathy_dispatch_operation_approve (approval->operation);
674 g_array_free (members, TRUE);
678 event_room_channel_process_func (EventPriv *event)
680 GtkWidget *dialog, *button, *image;
681 TpChannel *channel = empathy_dispatch_operation_get_channel (
682 event->approval->operation);
684 if (event->approval->dialog != NULL)
686 gtk_window_present (GTK_WINDOW (event->approval->dialog));
691 dialog = gtk_message_dialog_new (NULL, 0,
692 GTK_MESSAGE_QUESTION, GTK_BUTTONS_NONE, _("Room invitation"));
694 gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog),
695 _("%s is inviting you to join %s"),
696 empathy_contact_get_name (event->approval->contact),
697 tp_channel_get_identifier (channel));
699 gtk_dialog_set_default_response (GTK_DIALOG (dialog),
702 button = gtk_dialog_add_button (GTK_DIALOG (dialog),
703 _("_Decline"), GTK_RESPONSE_CANCEL);
704 image = gtk_image_new_from_icon_name (GTK_STOCK_CANCEL, GTK_ICON_SIZE_BUTTON);
705 gtk_button_set_image (GTK_BUTTON (button), image);
707 button = gtk_dialog_add_button (GTK_DIALOG (dialog),
708 _("_Join"), GTK_RESPONSE_OK);
709 image = gtk_image_new_from_icon_name (GTK_STOCK_APPLY, GTK_ICON_SIZE_BUTTON);
710 gtk_button_set_image (GTK_BUTTON (button), image);
712 g_signal_connect (dialog, "response",
713 G_CALLBACK (invite_dialog_response_cb), event->approval);
715 gtk_widget_show (dialog);
717 event->approval->dialog = dialog;
721 event_manager_muc_invite_got_contact_name_cb (EmpathyContact *contact,
726 EventManagerApproval *approval = (EventManagerApproval *) user_data;
728 const gchar *invite_msg;
730 TpHandle self_handle;
732 channel = empathy_dispatch_operation_get_channel (approval->operation);
734 self_handle = tp_channel_group_get_self_handle (channel);
735 tp_channel_group_get_local_pending_info (channel, self_handle, NULL, NULL,
738 msg = g_strdup_printf (_("%s invited you to join %s"),
739 empathy_contact_get_name (approval->contact),
740 tp_channel_get_identifier (channel));
742 event_manager_add (approval->manager,
743 approval->contact, EMPATHY_IMAGE_GROUP_MESSAGE, msg, invite_msg,
744 approval, event_room_channel_process_func, NULL);
746 empathy_sound_play (empathy_main_window_get (),
747 EMPATHY_SOUND_CONVERSATION_NEW);
753 event_manager_approve_channel_cb (EmpathyDispatcher *dispatcher,
754 EmpathyDispatchOperation *operation, EmpathyEventManager *manager)
756 const gchar *channel_type;
757 EventManagerApproval *approval;
758 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
760 channel_type = empathy_dispatch_operation_get_channel_type (operation);
762 approval = event_manager_approval_new (manager, operation);
763 priv->approvals = g_slist_prepend (priv->approvals, approval);
765 approval->approved_handler = g_signal_connect (operation, "approved",
766 G_CALLBACK (event_manager_operation_approved_cb), approval);
768 approval->claimed_handler = g_signal_connect (operation, "claimed",
769 G_CALLBACK (event_manager_operation_claimed_cb), approval);
771 approval->invalidated_handler = g_signal_connect (operation, "invalidated",
772 G_CALLBACK (event_manager_operation_invalidated_cb), approval);
774 if (!tp_strdiff (channel_type, TP_IFACE_CHANNEL_TYPE_TEXT))
776 EmpathyTpChat *tp_chat =
778 empathy_dispatch_operation_get_channel_wrapper (operation));
779 TpChannel *channel = empathy_tp_chat_get_channel (tp_chat);
781 if (tp_proxy_has_interface (channel, TP_IFACE_CHANNEL_INTERFACE_GROUP))
783 /* Are we in local-pending ? */
784 TpHandle self_handle, inviter;
786 self_handle = tp_channel_group_get_self_handle (channel);
788 if (self_handle != 0 && tp_channel_group_get_local_pending_info (
789 channel, self_handle, &inviter, NULL, NULL))
791 /* We are invited to a room */
792 EmpathyContactFactory *contact_factory;
795 DEBUG ("Have been invited to %s. Ask user if he wants to accept",
796 tp_channel_get_identifier (channel));
798 account = empathy_tp_chat_get_account (tp_chat);
799 contact_factory = empathy_contact_factory_dup_singleton ();
801 approval->contact = empathy_contact_factory_get_from_handle (
802 contact_factory, account, inviter);
804 empathy_contact_call_when_ready (approval->contact,
805 EMPATHY_CONTACT_READY_NAME,
806 event_manager_muc_invite_got_contact_name_cb, approval, NULL,
809 g_object_unref (contact_factory);
813 /* if we are not invited, let's wait for the first message */
816 /* 1-1 text channel, wait for the first message */
817 approval->handler = g_signal_connect (tp_chat, "message-received",
818 G_CALLBACK (event_manager_chat_message_received_cb), approval);
820 else if (!tp_strdiff (channel_type, TP_IFACE_CHANNEL_TYPE_STREAMED_MEDIA))
822 EmpathyContact *contact;
823 EmpathyTpCall *call = EMPATHY_TP_CALL (
824 empathy_dispatch_operation_get_channel_wrapper (operation));
826 g_object_get (G_OBJECT (call), "contact", &contact, NULL);
830 g_signal_connect (call, "notify::contact",
831 G_CALLBACK (event_manager_media_channel_contact_changed_cb),
836 approval->contact = contact;
837 event_manager_media_channel_got_contact (approval);
841 else if (!tp_strdiff (channel_type, TP_IFACE_CHANNEL_TYPE_FILE_TRANSFER))
843 EmpathyContact *contact;
847 EmpathyContactFactory *factory;
848 TpChannel *channel = empathy_dispatch_operation_get_channel (operation);
850 factory = empathy_contact_factory_dup_singleton ();
851 handle = tp_channel_get_handle (channel, NULL);
852 account = empathy_channel_get_account (channel);
854 contact = empathy_contact_factory_get_from_handle (factory, account,
857 empathy_contact_run_until_ready (contact,
858 EMPATHY_CONTACT_READY_NAME, NULL);
860 header = g_strdup_printf (_("Incoming file transfer from %s"),
861 empathy_contact_get_name (contact));
863 event_manager_add (manager, contact, EMPATHY_IMAGE_DOCUMENT_SEND,
864 header, NULL, approval, event_channel_process_func, NULL);
866 /* FIXME better sound for incoming file transfers ?*/
867 empathy_sound_play (empathy_main_window_get (),
868 EMPATHY_SOUND_CONVERSATION_NEW);
870 g_object_unref (factory);
871 g_object_unref (account);
874 else if (!tp_strdiff (channel_type, EMP_IFACE_CHANNEL_TYPE_STREAM_TUBE) ||
875 !tp_strdiff (channel_type, EMP_IFACE_CHANNEL_TYPE_DBUS_TUBE))
877 EmpathyContact *contact;
879 TpHandleType handle_type;
881 EmpathyContactFactory *factory;
882 EmpathyTubeDispatch *tube_dispatch;
885 channel = empathy_dispatch_operation_get_channel (operation);
887 handle = tp_channel_get_handle (channel, &handle_type);
889 /* Only understand p2p tubes */
890 if (handle_type != TP_HANDLE_TYPE_CONTACT)
893 factory = empathy_contact_factory_dup_singleton ();
894 account = empathy_channel_get_account (channel);
896 contact = empathy_contact_factory_get_from_handle (factory, account,
899 tube_dispatch = empathy_tube_dispatch_new (operation);
901 approval->contact = contact;
902 approval->tube_dispatch = tube_dispatch;
904 empathy_contact_call_when_ready (contact,
905 EMPATHY_CONTACT_READY_NAME, event_manager_tube_got_contact_name_cb,
906 approval, NULL, G_OBJECT (manager));
908 g_object_unref (factory);
909 g_object_unref (account);
913 DEBUG ("Unknown channel type, ignoring..");
918 event_pending_subscribe_func (EventPriv *event)
920 empathy_subscription_dialog_show (event->public.contact, NULL);
921 event_remove (event);
925 event_manager_pendings_changed_cb (EmpathyContactList *list,
926 EmpathyContact *contact, EmpathyContact *actor,
927 guint reason, gchar *message, gboolean is_pending,
928 EmpathyEventManager *manager)
930 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
931 gchar *header, *event_msg;
937 for (l = priv->events; l; l = l->next)
939 EventPriv *event = l->data;
941 if (event->public.contact == contact &&
942 event->func == event_pending_subscribe_func)
944 event_remove (event);
952 empathy_contact_run_until_ready (contact, EMPATHY_CONTACT_READY_NAME, NULL);
954 header = g_strdup_printf (_("Subscription requested by %s"),
955 empathy_contact_get_name (contact));
957 if (!EMP_STR_EMPTY (message))
958 event_msg = g_strdup_printf (_("\nMessage: %s"), message);
962 event_manager_add (manager, contact, GTK_STOCK_DIALOG_QUESTION, header,
963 event_msg, NULL, event_pending_subscribe_func, NULL);
970 event_manager_constructor (GType type,
972 GObjectConstructParam *props)
976 if (manager_singleton) {
977 retval = g_object_ref (manager_singleton);
979 retval = G_OBJECT_CLASS (empathy_event_manager_parent_class)->constructor
980 (type, n_props, props);
982 manager_singleton = EMPATHY_EVENT_MANAGER (retval);
983 g_object_add_weak_pointer (retval, (gpointer) &manager_singleton);
990 event_manager_finalize (GObject *object)
992 EmpathyEventManagerPriv *priv = GET_PRIV (object);
994 g_slist_foreach (priv->events, (GFunc) event_free, NULL);
995 g_slist_free (priv->events);
996 g_slist_foreach (priv->approvals, (GFunc) event_manager_approval_free, NULL);
997 g_slist_free (priv->approvals);
998 g_object_unref (priv->contact_manager);
999 g_object_unref (priv->dispatcher);
1003 empathy_event_manager_class_init (EmpathyEventManagerClass *klass)
1005 GObjectClass *object_class = G_OBJECT_CLASS (klass);
1007 object_class->finalize = event_manager_finalize;
1008 object_class->constructor = event_manager_constructor;
1010 signals[EVENT_ADDED] =
1011 g_signal_new ("event-added",
1012 G_TYPE_FROM_CLASS (klass),
1016 g_cclosure_marshal_VOID__POINTER,
1020 signals[EVENT_REMOVED] =
1021 g_signal_new ("event-removed",
1022 G_TYPE_FROM_CLASS (klass),
1026 g_cclosure_marshal_VOID__POINTER,
1027 G_TYPE_NONE, 1, G_TYPE_POINTER);
1029 signals[EVENT_UPDATED] =
1030 g_signal_new ("event-updated",
1031 G_TYPE_FROM_CLASS (klass),
1035 g_cclosure_marshal_VOID__POINTER,
1036 G_TYPE_NONE, 1, G_TYPE_POINTER);
1039 g_type_class_add_private (object_class, sizeof (EmpathyEventManagerPriv));
1043 empathy_event_manager_init (EmpathyEventManager *manager)
1045 EmpathyEventManagerPriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (manager,
1046 EMPATHY_TYPE_EVENT_MANAGER, EmpathyEventManagerPriv);
1048 manager->priv = priv;
1050 priv->dispatcher = empathy_dispatcher_dup_singleton ();
1051 priv->contact_manager = empathy_contact_manager_dup_singleton ();
1052 g_signal_connect (priv->dispatcher, "approve",
1053 G_CALLBACK (event_manager_approve_channel_cb), manager);
1054 g_signal_connect (priv->contact_manager, "pendings-changed",
1055 G_CALLBACK (event_manager_pendings_changed_cb), manager);
1058 EmpathyEventManager *
1059 empathy_event_manager_dup_singleton (void)
1061 return g_object_new (EMPATHY_TYPE_EVENT_MANAGER, NULL);
1065 empathy_event_manager_get_events (EmpathyEventManager *manager)
1067 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
1069 g_return_val_if_fail (EMPATHY_IS_EVENT_MANAGER (manager), NULL);
1071 return priv->events;
1075 empathy_event_manager_get_top_event (EmpathyEventManager *manager)
1077 EmpathyEventManagerPriv *priv = GET_PRIV (manager);
1079 g_return_val_if_fail (EMPATHY_IS_EVENT_MANAGER (manager), NULL);
1081 return priv->events ? priv->events->data : NULL;
1085 empathy_event_activate (EmpathyEvent *event_public)
1087 EventPriv *event = (EventPriv*) event_public;
1089 g_return_if_fail (event_public != NULL);
1092 event->func (event);
1094 event_remove (event);
1098 empathy_event_inhibit_updates (EmpathyEvent *event_public)
1100 EventPriv *event = (EventPriv *) event_public;
1102 g_return_if_fail (event_public != NULL);
1104 event->inhibit = TRUE;