2 * Copyright (C) 2007-2011 Collabora Ltd.
4 * This program is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU General Public License as
6 * published by the Free Software Foundation; either version 2 of the
7 * License, or (at your option) any later version.
9 * This program 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 * General Public License for more details.
14 * You should have received a copy of the GNU General Public
15 * License along with this program; if not, write to the
16 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
17 * Boston, MA 02110-1301 USA
19 * Authors: Xavier Claessens <xclaesse@gmail.com>
20 * Guillaume Desmottes <guillaume.desmottes@collabora.co.uk>
26 #include <glib/gi18n.h>
29 #include <clutter/clutter.h>
30 #include <clutter-gtk/clutter-gtk.h>
31 #include <clutter-gst/clutter-gst.h>
33 #ifdef CLUTTER_WINDOWING_X11
37 #include <telepathy-glib/debug-sender.h>
39 #include <telepathy-yell/telepathy-yell.h>
41 #include <libempathy/empathy-client-factory.h>
43 #include <libempathy-gtk/empathy-ui-utils.h>
45 #include "empathy-call-window.h"
46 #include "empathy-call-factory.h"
48 #define DEBUG_FLAG EMPATHY_DEBUG_VOIP
49 #include <libempathy/empathy-debug.h>
53 /* Exit after $TIMEOUT seconds if not displaying any call window */
56 #define EMPATHY_CALL_DBUS_NAME "org.gnome.Empathy.Call"
58 static GtkApplication *app = NULL;
59 static gboolean activated = FALSE;
60 static gboolean use_timer = TRUE;
62 static EmpathyCallFactory *call_factory = NULL;
64 /* An EmpathyContact -> EmpathyCallWindow hash table for all existing
65 * Call windows. We own a ref on the EmpathyContacts. */
66 static GHashTable *call_windows;
69 call_window_destroyed_cb (GtkWidget *window,
70 EmpathyContact *contact)
72 g_hash_table_remove (call_windows, contact);
74 g_application_release (G_APPLICATION (app));
78 find_window_for_handle (gpointer key,
82 EmpathyContact *contact = key;
83 guint handle = GPOINTER_TO_UINT (user_data);
85 if (handle == empathy_contact_get_handle (contact))
92 incoming_call_cb (EmpathyCallFactory *factory,
94 TpyCallChannel *channel,
95 TpChannelDispatchOperation *dispatch_operation,
96 TpAddDispatchOperationContext *context,
99 EmpathyCallWindow *window = g_hash_table_find (call_windows,
100 find_window_for_handle, GUINT_TO_POINTER (handle));
104 /* The window takes care of accepting or rejecting the context. */
105 empathy_call_window_start_ringing (window,
106 channel, dispatch_operation, context);
114 new_call_handler_cb (EmpathyCallFactory *factory,
115 EmpathyCallHandler *handler,
119 EmpathyCallWindow *window;
120 EmpathyContact *contact;
122 DEBUG ("Show the call window");
124 g_object_get (handler, "target-contact", &contact, NULL);
126 window = g_hash_table_lookup (call_windows, contact);
130 empathy_call_window_present (window, handler);
134 window = empathy_call_window_new (handler);
136 g_hash_table_insert (call_windows, g_object_ref (contact), window);
137 g_application_hold (G_APPLICATION (app));
138 g_signal_connect (window, "destroy",
139 G_CALLBACK (call_window_destroyed_cb), contact);
141 gtk_widget_show (GTK_WIDGET (window));
146 activate_cb (GApplication *application)
148 GError *error = NULL;
157 /* keep a 'ref' to the application */
158 g_application_hold (G_APPLICATION (app));
161 g_assert (call_factory == NULL);
162 call_factory = empathy_call_factory_initialise ();
164 g_signal_connect (G_OBJECT (call_factory), "new-call-handler",
165 G_CALLBACK (new_call_handler_cb), NULL);
166 g_signal_connect (G_OBJECT (call_factory), "incoming-call",
167 G_CALLBACK (incoming_call_cb), NULL);
169 if (!empathy_call_factory_register (call_factory, &error))
171 g_critical ("Failed to register Handler: %s", error->message);
172 g_error_free (error);
180 GOptionContext *optcontext;
181 GOptionEntry options[] = {
185 TpDebugSender *debug_sender;
187 GError *error = NULL;
189 GtkSettings *gtk_settings;
192 g_thread_init (NULL);
194 #ifdef GDK_WINDOWING_X11
195 /* We can't call clutter_gst_init() before gtk_clutter_init(), so no choice
196 * but to intiialise X11 threading ourself */
200 /* Clutter needs this */
201 gdk_disable_multidevice ();
203 optcontext = g_option_context_new (N_("- Empathy Audio/Video Client"));
204 g_option_context_add_group (optcontext, gst_init_get_option_group ());
205 g_option_context_add_group (optcontext, gtk_get_option_group (TRUE));
206 g_option_context_add_group (optcontext, cogl_get_option_group ());
207 g_option_context_add_group (optcontext,
208 clutter_get_option_group_without_init ());
209 g_option_context_add_group (optcontext, gtk_clutter_get_option_group ());
210 g_option_context_add_main_entries (optcontext, options, GETTEXT_PACKAGE);
212 if (!g_option_context_parse (optcontext, &argc, &argv, &error)) {
213 g_print ("%s\nRun '%s --help' to see a full list of available command "
215 error->message, argv[0]);
216 g_warning ("Error in empathy-call init: %s", error->message);
220 g_option_context_free (optcontext);
224 gtk_clutter_init (&argc, &argv);
225 clutter_gst_init (&argc, &argv);
228 g_set_application_name (_("Empathy Audio/Video Client"));
230 /* Make empathy and empathy-call appear as the same app in gnome-shell */
231 gdk_set_program_class ("Empathy");
232 gtk_window_set_default_icon_name ("empathy");
233 textdomain (GETTEXT_PACKAGE);
235 gtk_settings = gtk_settings_get_default ();
236 g_object_set (G_OBJECT (gtk_settings), "gtk-application-prefer-dark-theme",
239 app = gtk_application_new (EMPATHY_CALL_DBUS_NAME, G_APPLICATION_FLAGS_NONE);
240 g_signal_connect (app, "activate", G_CALLBACK (activate_cb), NULL);
243 /* Set up debug sender */
244 debug_sender = tp_debug_sender_dup ();
245 g_log_set_default_handler (tp_debug_sender_log_handler, G_LOG_DOMAIN);
248 if (g_getenv ("EMPATHY_PERSIST") != NULL)
250 DEBUG ("Disable timer");
255 call_windows = g_hash_table_new_full (g_direct_hash, g_direct_equal,
256 g_object_unref, NULL);
258 /* the inactivity timeout can only be set while the application is held */
259 g_application_hold (G_APPLICATION (app));
260 g_application_set_inactivity_timeout (G_APPLICATION (app), TIMEOUT * 1000);
261 g_application_release (G_APPLICATION (app));
263 retval = g_application_run (G_APPLICATION (app), argc, argv);
265 g_hash_table_unref (call_windows);
266 g_object_unref (app);
267 tp_clear_object (&call_factory);
270 g_object_unref (debug_sender);