]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-call-utils.c
Merge branch 'empathy-av-ocrete'
[empathy.git] / libempathy-gtk / empathy-call-utils.c
1 /*
2  * Copyright (C) 2011 Collabora Ltd.
3  *
4  * The code contained in this file is free software; you can redistribute
5  * it and/or modify it under the terms of the GNU Lesser General Public
6  * License as published by the Free Software Foundation; either version
7  * 2.1 of the License, or (at your option) any later version.
8  *
9  * This file 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.
13  *
14  * You should have received a copy of the GNU Lesser General Public
15  * License along with this code; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
17  *
18  * Authors: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
19  */
20
21 #include "config.h"
22
23 #include <glib/gi18n.h>
24
25 #include <gtk/gtk.h>
26 #include <pulse/pulseaudio.h>
27
28 #include <telepathy-glib/telepathy-glib.h>
29
30 #include "empathy-call-utils.h"
31
32 #include <libempathy/empathy-gsettings.h>
33 #include <libempathy/empathy-request-util.h>
34
35 #define DEBUG_FLAG EMPATHY_DEBUG_OTHER
36 #include <libempathy/empathy-debug.h>
37
38 static const gchar *
39 get_error_display_message (GError *error)
40 {
41   if (error->domain != TP_ERROR)
42     return _("There was an error starting the call");
43
44   switch (error->code)
45     {
46       case TP_ERROR_NETWORK_ERROR:
47         return _("Network error");
48       case TP_ERROR_NOT_CAPABLE:
49         return _("The specified contact doesn't support calls");
50       case TP_ERROR_OFFLINE:
51         return _("The specified contact is offline");
52       case TP_ERROR_INVALID_HANDLE:
53         return _("The specified contact is not valid");
54       case TP_ERROR_EMERGENCY_CALLS_NOT_SUPPORTED:
55         return _("Emergency calls are not supported on this protocol");
56       case TP_ERROR_INSUFFICIENT_BALANCE:
57         return _("You don't have enough credit in order to place this call");
58     }
59
60   return _("There was an error starting the call");
61 }
62
63 static void
64 show_call_error (GError *error)
65 {
66   GtkWidget *dialog;
67
68   dialog = gtk_message_dialog_new (NULL, 0,
69       GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
70       "%s", get_error_display_message (error));
71
72   g_signal_connect_swapped (dialog, "response",
73       G_CALLBACK (gtk_widget_destroy),
74       dialog);
75
76   gtk_widget_show (dialog);
77 }
78
79 GHashTable *
80 empathy_call_create_call_request (const gchar *contact,
81     gboolean initial_audio,
82     gboolean initial_video)
83 {
84   return tp_asv_new (
85     TP_PROP_CHANNEL_CHANNEL_TYPE, G_TYPE_STRING,
86       TP_IFACE_CHANNEL_TYPE_CALL,
87     TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, G_TYPE_UINT,
88       TP_HANDLE_TYPE_CONTACT,
89     TP_PROP_CHANNEL_TARGET_ID, G_TYPE_STRING,
90       contact,
91     TP_PROP_CHANNEL_TYPE_CALL_INITIAL_AUDIO, G_TYPE_BOOLEAN,
92       initial_audio,
93     TP_PROP_CHANNEL_TYPE_CALL_INITIAL_VIDEO, G_TYPE_BOOLEAN,
94       initial_video,
95     NULL);
96 }
97
98 GHashTable *
99 empathy_call_create_streamed_media_request (const gchar *contact,
100     gboolean initial_audio,
101     gboolean initial_video)
102 {
103   return tp_asv_new (
104     TP_PROP_CHANNEL_CHANNEL_TYPE, G_TYPE_STRING,
105       TP_IFACE_CHANNEL_TYPE_STREAMED_MEDIA,
106     TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, G_TYPE_UINT,
107       TP_HANDLE_TYPE_CONTACT,
108     TP_PROP_CHANNEL_TARGET_ID, G_TYPE_STRING,
109       contact,
110     TP_PROP_CHANNEL_TYPE_STREAMED_MEDIA_INITIAL_AUDIO, G_TYPE_BOOLEAN,
111       initial_audio,
112     TP_PROP_CHANNEL_TYPE_STREAMED_MEDIA_INITIAL_VIDEO, G_TYPE_BOOLEAN,
113       initial_video,
114     NULL);
115 }
116
117 static void
118 create_streamed_media_channel_cb (GObject *source,
119     GAsyncResult *result,
120     gpointer user_data)
121 {
122   GError *error = NULL;
123
124   if (!tp_account_channel_request_create_channel_finish (
125            TP_ACCOUNT_CHANNEL_REQUEST (source),
126            result,
127            &error))
128     {
129       DEBUG ("Failed to create StreamedMedia channel: %s", error->message);
130       show_call_error (error);
131       g_error_free (error);
132     }
133 }
134
135 static void
136 create_call_channel_cb (GObject *source,
137     GAsyncResult *result,
138     gpointer user_data)
139 {
140   TpAccountChannelRequest *streamed_media_req = user_data;
141   GError *error = NULL;
142
143   if (tp_account_channel_request_create_channel_finish (
144       TP_ACCOUNT_CHANNEL_REQUEST (source), result, &error))
145     {
146       g_clear_object (&streamed_media_req);
147       return;
148     }
149
150   DEBUG ("Failed to create Call channel: %s", error->message);
151
152   if (streamed_media_req != NULL)
153     {
154       DEBUG ("Let's try with an StreamedMedia channel");
155       g_error_free (error);
156       tp_account_channel_request_create_channel_async (streamed_media_req,
157           EMPATHY_AV_BUS_NAME, NULL,
158           create_streamed_media_channel_cb,
159           NULL);
160       return;
161     }
162
163   show_call_error (error);
164 }
165
166 /* Try to request a Call channel and fallback to StreamedMedia if that fails */
167 static void
168 call_new_with_streams (const gchar *contact,
169     TpAccount *account,
170     gboolean initial_audio,
171     gboolean initial_video,
172     gint64 timestamp)
173 {
174   GHashTable *call_request;
175   TpAccountChannelRequest *call_req, *streamed_media_req = NULL;
176 #ifdef HAVE_EMPATHY_AV
177   GHashTable *streamed_media_request;
178 #endif
179
180   /* Call */
181   call_request = empathy_call_create_call_request (contact,
182       initial_audio,
183       initial_video);
184
185   call_req = tp_account_channel_request_new (account, call_request, timestamp);
186
187   g_hash_table_unref (call_request);
188
189 #ifdef HAVE_EMPATHY_AV
190   /* StreamedMedia */
191   streamed_media_request = empathy_call_create_streamed_media_request (
192       contact, initial_audio, initial_video);
193
194   streamed_media_req = tp_account_channel_request_new (account,
195       streamed_media_request,
196       timestamp);
197
198   g_hash_table_unref (streamed_media_request);
199 #endif
200
201   tp_account_channel_request_create_channel_async (call_req,
202       EMPATHY_CALL_BUS_NAME, NULL, create_call_channel_cb, streamed_media_req);
203
204   g_object_unref (call_req);
205 }
206
207 void
208 empathy_call_new_with_streams (const gchar *contact,
209     TpAccount *account,
210     gboolean initial_audio,
211     gboolean initial_video,
212     gint64 timestamp)
213 {
214   call_new_with_streams (contact, account, initial_audio, initial_video,
215       timestamp);
216 }
217
218 void
219 empathy_call_set_stream_properties (GstElement *element,
220   gboolean echo_cancellation)
221 {
222   GstStructure *props;
223   GSettings *gsettings_call;
224   gboolean echo_cancellation_setting;
225
226   gsettings_call = g_settings_new (EMPATHY_PREFS_CALL_SCHEMA);
227
228   echo_cancellation_setting = g_settings_get_boolean (gsettings_call,
229       EMPATHY_PREFS_CALL_ECHO_CANCELLATION);
230
231   DEBUG ("Echo cancellation: element allowed: %s, user enabled: %s",
232     echo_cancellation ? " yes" : "no",
233     echo_cancellation_setting ? " yes" : "no");
234
235
236   props = gst_structure_new ("props",
237       PA_PROP_MEDIA_ROLE, G_TYPE_STRING, "phone",
238       NULL);
239
240   if (echo_cancellation && echo_cancellation_setting)
241     {
242       gst_structure_set (props,
243           "filter.want", G_TYPE_STRING, "echo-cancel",
244           NULL);
245     }
246
247   g_object_set (element, "stream-properties", props, NULL);
248   gst_structure_free (props);
249
250   g_object_unref (gsettings_call);
251 }
252
253 /* Copied from telepathy-yell call-channel.c */
254 void
255 empathy_call_channel_send_video (TpCallChannel *self,
256     gboolean send)
257 {
258   GPtrArray *contents;
259   gboolean found = FALSE;
260   guint i;
261
262   g_return_if_fail (TP_IS_CALL_CHANNEL (self));
263
264   /* Loop over all the contents, if some of them a video set all their
265    * streams to sending, otherwise request a video channel in case we want to
266    * sent */
267   contents = tp_call_channel_get_contents (self);
268   for (i = 0 ; i < contents->len ; i++)
269     {
270       TpCallContent *content = g_ptr_array_index (contents, i);
271
272       if (tp_call_content_get_media_type (content) ==
273               TP_MEDIA_STREAM_TYPE_VIDEO)
274         {
275           GPtrArray *streams;
276           guint j;
277
278           found = TRUE;
279           streams = tp_call_content_get_streams (content);
280           for (j = 0; j < streams->len; j++)
281             {
282               TpCallStream *stream = g_ptr_array_index (streams, j);
283
284               tp_call_stream_set_sending_async (stream, send, NULL, NULL);
285             }
286         }
287     }
288
289   if (send && !found)
290     {
291       tp_call_channel_add_content_async (self, "video",
292           TP_MEDIA_STREAM_TYPE_VIDEO, TP_MEDIA_STREAM_DIRECTION_BIDIRECTIONAL,
293           NULL, NULL);
294     }
295 }
296
297 /* Copied from telepathy-yell call-channel.c */
298 TpSendingState
299 empathy_call_channel_get_video_state (TpCallChannel *self)
300 {
301   TpSendingState result = TP_SENDING_STATE_NONE;
302   GPtrArray *contents;
303   guint i;
304
305   g_return_val_if_fail (TP_IS_CALL_CHANNEL (self), TP_SENDING_STATE_NONE);
306
307   contents = tp_call_channel_get_contents (self);
308   for (i = 0 ; i < contents->len ; i++)
309     {
310       TpCallContent *content = g_ptr_array_index (contents, i);
311
312       if (tp_call_content_get_media_type (content) ==
313               TP_MEDIA_STREAM_TYPE_VIDEO)
314         {
315           GPtrArray *streams;
316           guint j;
317
318           streams = tp_call_content_get_streams (content);
319           for (j = 0; j < streams->len; j++)
320             {
321               TpCallStream *stream = g_ptr_array_index (streams, j);
322               TpSendingState state;
323
324               state = tp_call_stream_get_local_sending_state (stream);
325               if (state != TP_SENDING_STATE_PENDING_STOP_SENDING &&
326                   state > result)
327                 result = state;
328             }
329         }
330     }
331
332   return result;
333 }