]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-new-call-dialog.c
Merge branch 'call'
[empathy.git] / libempathy-gtk / empathy-new-call-dialog.c
1 /*
2  * Copyright (C) 2009 Collabora Ltd.
3  *
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.
8  *
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.
13  *
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
17  *
18  * Authors: Guillaume Desmottes <guillaume.desmottes@collabora.co.uk>
19  */
20
21 #include <config.h>
22
23 #include <string.h>
24 #include <stdlib.h>
25
26 #include <gtk/gtk.h>
27 #include <glib/gi18n-lib.h>
28
29 #include <telepathy-glib/interfaces.h>
30
31 #include <telepathy-yell/telepathy-yell.h>
32
33 #include <libempathy/empathy-tp-contact-factory.h>
34 #include <libempathy/empathy-contact-manager.h>
35 #include <libempathy/empathy-utils.h>
36 #include <libempathy/empathy-request-util.h>
37
38 #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT
39 #include <libempathy/empathy-debug.h>
40
41 #include <libempathy-gtk/empathy-ui-utils.h>
42 #include <libempathy-gtk/empathy-images.h>
43
44 #include "empathy-new-call-dialog.h"
45 #include "empathy-account-chooser.h"
46 #include "empathy-call-utils.h"
47
48 static EmpathyNewCallDialog *dialog_singleton = NULL;
49
50 G_DEFINE_TYPE(EmpathyNewCallDialog, empathy_new_call_dialog,
51                EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG)
52
53 typedef struct _EmpathyNewCallDialogPriv EmpathyNewCallDialogPriv;
54
55 typedef struct {
56   EmpathyAccountChooserFilterResultCallback callback;
57   gpointer                                  user_data;
58 } FilterCallbackData;
59
60 struct _EmpathyNewCallDialogPriv {
61   GtkWidget *check_video;
62 };
63
64 #define GET_PRIV(o) \
65   (G_TYPE_INSTANCE_GET_PRIVATE ((o), EMPATHY_TYPE_NEW_CALL_DIALOG, \
66     EmpathyNewCallDialogPriv))
67
68 /**
69  * SECTION:empathy-new-call-dialog
70  * @title: EmpathyNewCallDialog
71  * @short_description: A dialog to show a new call
72  * @include: libempathy-gtk/empathy-new-call-dialog.h
73  *
74  * #EmpathyNewCallDialog is a dialog which allows a call
75  * to be started with any contact on any enabled account.
76  */
77
78 static void
79 empathy_new_call_dialog_response (GtkDialog *dialog, int response_id)
80 {
81   EmpathyNewCallDialogPriv *priv = GET_PRIV (dialog);
82   gboolean video;
83   TpAccount *account;
84   const gchar *contact_id;
85
86   if (response_id != GTK_RESPONSE_ACCEPT) goto out;
87
88   contact_id = empathy_contact_selector_dialog_get_selected (
89       EMPATHY_CONTACT_SELECTOR_DIALOG (dialog), NULL, &account);
90
91   if (EMP_STR_EMPTY (contact_id) || account == NULL) goto out;
92
93   /* check if video is enabled now because the dialog will be destroyed once
94    * we return from this function. */
95   video = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (priv->check_video));
96
97   empathy_call_new_with_streams (contact_id,
98       account, TRUE, video,
99       empathy_get_current_action_time ());
100
101 out:
102   gtk_widget_destroy (GTK_WIDGET (dialog));
103 }
104
105 static void
106 conn_prepared_cb (GObject *conn,
107     GAsyncResult *result,
108     gpointer user_data)
109 {
110   FilterCallbackData *data = user_data;
111   GError *myerr = NULL;
112   TpCapabilities *caps;
113   GPtrArray *classes;
114   guint i;
115
116   if (!tp_proxy_prepare_finish (conn, result, &myerr))
117       goto out;
118
119   caps = tp_connection_get_capabilities (TP_CONNECTION (conn));
120   classes = tp_capabilities_get_channel_classes (caps);
121
122   for (i = 0; i < classes->len; i++)
123     {
124       GHashTable *fixed;
125       GStrv allowed;
126       const gchar *chan_type;
127
128       tp_value_array_unpack (g_ptr_array_index (classes, i), 2,
129           &fixed, &allowed);
130
131       chan_type = tp_asv_get_string (fixed, TP_PROP_CHANNEL_CHANNEL_TYPE);
132
133       if (tp_strdiff (chan_type, TP_IFACE_CHANNEL_TYPE_STREAMED_MEDIA)
134           && tp_strdiff (chan_type, TPY_IFACE_CHANNEL_TYPE_CALL))
135         continue;
136
137       if (tp_asv_get_uint32 (fixed, TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, NULL) !=
138           TP_HANDLE_TYPE_CONTACT)
139         continue;
140
141       data->callback (TRUE, data->user_data);
142       g_slice_free (FilterCallbackData, data);
143       return;
144     }
145
146 out:
147   data->callback (FALSE, data->user_data);
148   g_slice_free (FilterCallbackData, data);
149 }
150
151 static void
152 empathy_new_call_dialog_account_filter (EmpathyContactSelectorDialog *dialog,
153     EmpathyAccountChooserFilterResultCallback callback,
154     gpointer callback_data,
155     TpAccount *account)
156 {
157   TpConnection *connection;
158   FilterCallbackData *cb_data;
159   GQuark features[] = { TP_CONNECTION_FEATURE_CAPABILITIES, 0 };
160
161   if (tp_account_get_connection_status (account, NULL) !=
162       TP_CONNECTION_STATUS_CONNECTED)
163     {
164       callback (FALSE, callback_data);
165       return;
166     }
167
168   /* check if CM supports calls */
169   connection = tp_account_get_connection (account);
170   if (connection == NULL)
171     {
172       callback (FALSE, callback_data);
173       return;
174     }
175
176   cb_data = g_slice_new0 (FilterCallbackData);
177   cb_data->callback = callback;
178   cb_data->user_data = callback_data;
179   tp_proxy_prepare_async (connection, features, conn_prepared_cb, cb_data);
180 }
181
182 static GObject *
183 empathy_new_call_dialog_constructor (GType type,
184     guint n_props,
185     GObjectConstructParam *props)
186 {
187   GObject *retval;
188
189   if (dialog_singleton)
190     {
191       retval = G_OBJECT (dialog_singleton);
192       g_object_ref (retval);
193     }
194   else
195     {
196       retval = G_OBJECT_CLASS (
197       empathy_new_call_dialog_parent_class)->constructor (type,
198         n_props, props);
199
200       dialog_singleton = EMPATHY_NEW_CALL_DIALOG (retval);
201       g_object_add_weak_pointer (retval, (gpointer) &dialog_singleton);
202     }
203
204   return retval;
205 }
206
207 static void
208 empathy_new_call_dialog_init (EmpathyNewCallDialog *dialog)
209 {
210   EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG (
211         dialog);
212   EmpathyNewCallDialogPriv *priv = GET_PRIV (dialog);
213   GtkWidget *image;
214
215   /* add video toggle */
216   priv->check_video = gtk_check_button_new_with_mnemonic (_("Send _Video"));
217
218   gtk_box_pack_end (GTK_BOX (parent->vbox), priv->check_video,
219       FALSE, TRUE, 0);
220
221   gtk_widget_show (priv->check_video);
222
223   /* add chat button */
224   parent->button_action = gtk_button_new_with_mnemonic (_("C_all"));
225   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_VOIP,
226       GTK_ICON_SIZE_BUTTON);
227   gtk_button_set_image (GTK_BUTTON (parent->button_action), image);
228
229   gtk_dialog_add_action_widget (GTK_DIALOG (dialog), parent->button_action,
230       GTK_RESPONSE_ACCEPT);
231   gtk_widget_show (parent->button_action);
232
233   /* Tweak the dialog */
234   gtk_window_set_title (GTK_WINDOW (dialog), _("New Call"));
235   gtk_window_set_role (GTK_WINDOW (dialog), "new_call");
236
237   gtk_widget_set_sensitive (parent->button_action, FALSE);
238 }
239
240 static void
241 empathy_new_call_dialog_class_init (
242   EmpathyNewCallDialogClass *class)
243 {
244   GObjectClass *object_class = G_OBJECT_CLASS (class);
245   GtkDialogClass *dialog_class = GTK_DIALOG_CLASS (class);
246   EmpathyContactSelectorDialogClass *selector_dialog_class = \
247     EMPATHY_CONTACT_SELECTOR_DIALOG_CLASS (class);
248
249   g_type_class_add_private (class, sizeof (EmpathyNewCallDialogPriv));
250
251   object_class->constructor = empathy_new_call_dialog_constructor;
252
253   dialog_class->response = empathy_new_call_dialog_response;
254
255   selector_dialog_class->account_filter = empathy_new_call_dialog_account_filter;
256 }
257
258 /**
259  * empathy_new_call_dialog_new:
260  * @parent: parent #GtkWindow of the dialog
261  *
262  * Create a new #EmpathyNewCallDialog it.
263  *
264  * Return value: the new #EmpathyNewCallDialog
265  */
266 GtkWidget *
267 empathy_new_call_dialog_show (GtkWindow *parent)
268 {
269   GtkWidget *dialog;
270
271   dialog = g_object_new (EMPATHY_TYPE_NEW_CALL_DIALOG, NULL);
272
273   if (parent)
274     {
275       gtk_window_set_transient_for (GTK_WINDOW (dialog),
276                   GTK_WINDOW (parent));
277     }
278
279   gtk_widget_show (dialog);
280   return dialog;
281 }