Display a context menu when clicking on audio/video icons (#590051)
[empathy-mirror.git] / libempathy-gtk / empathy-protocol-chooser.c
blob9c489176582c7b38c2220ebc8fb19d31e62ad690
1 /* -*- Mode: C; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2; -*- */
2 /*
3 * Copyright (C) 2007-2009 Collabora Ltd.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 * Authors: Xavier Claessens <xclaesse@gmail.com>
20 * Jonny Lamb <jonny.lamb@collabora.co.uk>
23 #include <config.h>
25 #include <string.h>
27 #include <telepathy-glib/util.h>
29 #include <gtk/gtk.h>
31 #include <glib/gi18n-lib.h>
33 #include <libempathy/empathy-utils.h>
34 #include <libempathy/empathy-connection-managers.h>
36 #include "empathy-protocol-chooser.h"
37 #include "empathy-ui-utils.h"
39 #define DEBUG_FLAG EMPATHY_DEBUG_ACCOUNT
40 #include <libempathy/empathy-debug.h>
42 /**
43 * SECTION:empathy-protocol-chooser
44 * @title: EmpathyProtocolChooser
45 * @short_description: A widget used to choose from a list of protocols
46 * @include: libempathy-gtk/empathy-protocol-chooser.h
48 * #EmpathyProtocolChooser is a widget which extends #GtkComboBox to provides a
49 * chooser of available protocols.
52 /**
53 * EmpathyProtocolChooser:
54 * @parent: parent object
56 * Widget which extends #GtkComboBox to provide a chooser of available
57 * protocols.
60 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyProtocolChooser)
61 typedef struct
63 GtkListStore *store;
65 gboolean dispose_run;
66 EmpathyConnectionManagers *cms;
68 EmpathyProtocolChooserFilterFunc filter_func;
69 gpointer filter_user_data;
71 GHashTable *protocols;
72 } EmpathyProtocolChooserPriv;
74 enum
76 COL_ICON,
77 COL_LABEL,
78 COL_CM,
79 COL_PROTOCOL_NAME,
80 COL_IS_GTALK,
81 COL_COUNT
84 G_DEFINE_TYPE (EmpathyProtocolChooser, empathy_protocol_chooser,
85 GTK_TYPE_COMBO_BOX);
87 static gint
88 protocol_chooser_sort_protocol_value (const gchar *protocol_name)
90 guint i;
91 const gchar *names[] = {
92 "jabber",
93 "local-xmpp",
94 "gtalk",
95 NULL
98 for (i = 0 ; names[i]; i++)
100 if (strcmp (protocol_name, names[i]) == 0)
101 return i;
104 return i;
107 static gint
108 protocol_chooser_sort_func (GtkTreeModel *model,
109 GtkTreeIter *iter_a,
110 GtkTreeIter *iter_b,
111 gpointer user_data)
113 gchar *protocol_a;
114 gchar *protocol_b;
115 gint cmp = 0;
117 gtk_tree_model_get (model, iter_a,
118 COL_PROTOCOL_NAME, &protocol_a,
119 -1);
120 gtk_tree_model_get (model, iter_b,
121 COL_PROTOCOL_NAME, &protocol_b,
122 -1);
124 cmp = protocol_chooser_sort_protocol_value (protocol_a);
125 cmp -= protocol_chooser_sort_protocol_value (protocol_b);
126 if (cmp == 0)
128 cmp = strcmp (protocol_a, protocol_b);
129 /* only happens for jabber where there is one entry for gtalk and one for
130 * non-gtalk */
131 if (cmp == 0)
133 gboolean is_gtalk;
134 gtk_tree_model_get (model, iter_a,
135 COL_IS_GTALK, &is_gtalk,
136 -1);
138 cmp = is_gtalk ? 1 : -1;
142 g_free (protocol_a);
143 g_free (protocol_b);
144 return cmp;
147 static void
148 protocol_choosers_add_cm (EmpathyProtocolChooser *chooser,
149 TpConnectionManager *cm)
151 EmpathyProtocolChooserPriv *priv = GET_PRIV (chooser);
152 const TpConnectionManagerProtocol * const *iter;
154 for (iter = cm->protocols; iter != NULL && *iter != NULL; iter++)
156 const TpConnectionManagerProtocol *proto = *iter;
157 gchar *icon_name;
158 const gchar *display_name;
159 const gchar *saved_cm_name;
161 saved_cm_name = g_hash_table_lookup (priv->protocols, proto->name);
163 if (!tp_strdiff (cm->name, "haze") && saved_cm_name != NULL &&
164 tp_strdiff (saved_cm_name, "haze"))
165 /* the CM we're adding is a haze implementation of something we already
166 * have; drop it.
168 continue;
170 if (tp_strdiff (cm->name, "haze") && !tp_strdiff (saved_cm_name, "haze"))
172 GtkTreeIter titer;
173 gboolean valid;
174 TpConnectionManager *haze_cm;
176 /* let's this CM replace the haze implementation */
177 valid = gtk_tree_model_get_iter_first (GTK_TREE_MODEL (priv->store),
178 &titer);
180 while (valid)
182 gchar *haze_proto_name = NULL;
184 gtk_tree_model_get (GTK_TREE_MODEL (priv->store), &titer,
185 COL_PROTOCOL_NAME, &haze_proto_name,
186 COL_CM, &haze_cm, -1);
188 if (haze_cm == NULL)
189 continue;
191 if (!tp_strdiff (haze_cm->name, "haze") &&
192 !tp_strdiff (haze_proto_name, proto->name))
194 gtk_list_store_remove (priv->store, &titer);
195 g_object_unref (haze_cm);
196 g_free (haze_proto_name);
197 break;
200 g_object_unref (haze_cm);
201 g_free (haze_proto_name);
202 valid = gtk_tree_model_iter_next (GTK_TREE_MODEL (priv->store),
203 &titer);
207 g_hash_table_insert (priv->protocols,
208 g_strdup (proto->name), g_strdup (cm->name));
210 icon_name = empathy_protocol_icon_name (proto->name);
211 display_name = empathy_protocol_name_to_display_name (proto->name);
213 if (display_name == NULL)
214 display_name = proto->name;
216 gtk_list_store_insert_with_values (priv->store,
217 NULL, 0,
218 COL_ICON, icon_name,
219 COL_LABEL, display_name,
220 COL_CM, cm,
221 COL_PROTOCOL_NAME, proto->name,
222 COL_IS_GTALK, FALSE,
223 -1);
225 if (!tp_strdiff (proto->name, "jabber") &&
226 !tp_strdiff (cm->name, "gabble"))
228 display_name = empathy_protocol_name_to_display_name ("gtalk");
229 gtk_list_store_insert_with_values (priv->store,
230 NULL, 0,
231 COL_ICON, "im-google-talk",
232 COL_LABEL, display_name,
233 COL_CM, cm,
234 COL_PROTOCOL_NAME, proto->name,
235 COL_IS_GTALK, TRUE,
236 -1);
239 g_free (icon_name);
243 static void
244 protocol_chooser_add_cms_list (EmpathyProtocolChooser *protocol_chooser,
245 GList *cms)
247 GList *l;
249 for (l = cms; l != NULL; l = l->next)
250 protocol_choosers_add_cm (protocol_chooser, l->data);
252 gtk_combo_box_set_active (GTK_COMBO_BOX (protocol_chooser), 0);
255 static void
256 protocol_chooser_cms_ready_cb (EmpathyConnectionManagers *cms,
257 GParamSpec *pspec,
258 EmpathyProtocolChooser *protocol_chooser)
260 if (empathy_connection_managers_is_ready (cms))
261 protocol_chooser_add_cms_list
262 (protocol_chooser, empathy_connection_managers_get_cms (cms));
265 static void
266 protocol_chooser_constructed (GObject *object)
268 EmpathyProtocolChooser *protocol_chooser;
269 EmpathyProtocolChooserPriv *priv;
270 GtkCellRenderer *renderer;
272 priv = GET_PRIV (object);
273 protocol_chooser = EMPATHY_PROTOCOL_CHOOSER (object);
275 /* set up combo box with new store */
276 priv->store = gtk_list_store_new (COL_COUNT,
277 G_TYPE_STRING, /* Icon name */
278 G_TYPE_STRING, /* Label */
279 G_TYPE_OBJECT, /* CM */
280 G_TYPE_STRING, /* protocol name */
281 G_TYPE_BOOLEAN); /* is gtalk */
283 /* Set the protocol sort function */
284 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (priv->store),
285 COL_PROTOCOL_NAME,
286 protocol_chooser_sort_func,
287 NULL, NULL);
288 gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (priv->store),
289 COL_PROTOCOL_NAME,
290 GTK_SORT_ASCENDING);
292 gtk_combo_box_set_model (GTK_COMBO_BOX (object),
293 GTK_TREE_MODEL (priv->store));
295 renderer = gtk_cell_renderer_pixbuf_new ();
296 gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (object), renderer, FALSE);
297 gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (object), renderer,
298 "icon-name", COL_ICON,
299 NULL);
300 g_object_set (renderer, "stock-size", GTK_ICON_SIZE_BUTTON, NULL);
302 renderer = gtk_cell_renderer_text_new ();
303 gtk_cell_layout_pack_start (GTK_CELL_LAYOUT (object), renderer, TRUE);
304 gtk_cell_layout_set_attributes (GTK_CELL_LAYOUT (object), renderer,
305 "text", COL_LABEL,
306 NULL);
308 if (empathy_connection_managers_is_ready (priv->cms))
309 protocol_chooser_add_cms_list (protocol_chooser,
310 empathy_connection_managers_get_cms (priv->cms));
311 else
312 g_signal_connect (priv->cms, "notify::ready",
313 G_CALLBACK (protocol_chooser_cms_ready_cb), protocol_chooser);
315 if (G_OBJECT_CLASS (empathy_protocol_chooser_parent_class)->constructed)
316 G_OBJECT_CLASS
317 (empathy_protocol_chooser_parent_class)->constructed (object);
320 static void
321 empathy_protocol_chooser_init (EmpathyProtocolChooser *protocol_chooser)
323 EmpathyProtocolChooserPriv *priv =
324 G_TYPE_INSTANCE_GET_PRIVATE (protocol_chooser,
325 EMPATHY_TYPE_PROTOCOL_CHOOSER, EmpathyProtocolChooserPriv);
327 priv->dispose_run = FALSE;
328 priv->cms = empathy_connection_managers_dup_singleton ();
329 priv->protocols = g_hash_table_new_full (g_str_hash, g_str_equal,
330 g_free, g_free);
332 protocol_chooser->priv = priv;
335 static void
336 protocol_chooser_finalize (GObject *object)
338 EmpathyProtocolChooser *protocol_chooser = EMPATHY_PROTOCOL_CHOOSER (object);
339 EmpathyProtocolChooserPriv *priv = GET_PRIV (protocol_chooser);
341 if (priv->protocols)
343 g_hash_table_destroy (priv->protocols);
344 priv->protocols = NULL;
347 (G_OBJECT_CLASS (empathy_protocol_chooser_parent_class)->finalize) (object);
350 static void
351 protocol_chooser_dispose (GObject *object)
353 EmpathyProtocolChooser *protocol_chooser = EMPATHY_PROTOCOL_CHOOSER (object);
354 EmpathyProtocolChooserPriv *priv = GET_PRIV (protocol_chooser);
356 if (priv->dispose_run)
357 return;
359 priv->dispose_run = TRUE;
361 if (priv->store)
363 g_object_unref (priv->store);
364 priv->store = NULL;
367 if (priv->cms)
369 g_object_unref (priv->cms);
370 priv->cms = NULL;
373 (G_OBJECT_CLASS (empathy_protocol_chooser_parent_class)->dispose) (object);
376 static void
377 empathy_protocol_chooser_class_init (EmpathyProtocolChooserClass *klass)
379 GObjectClass *object_class = G_OBJECT_CLASS (klass);
381 object_class->constructed = protocol_chooser_constructed;
382 object_class->dispose = protocol_chooser_dispose;
383 object_class->finalize = protocol_chooser_finalize;
385 g_type_class_add_private (object_class, sizeof (EmpathyProtocolChooserPriv));
388 static gboolean
389 protocol_chooser_filter_visible_func (GtkTreeModel *model,
390 GtkTreeIter *iter,
391 gpointer user_data)
393 EmpathyProtocolChooser *protocol_chooser = user_data;
394 EmpathyProtocolChooserPriv *priv = GET_PRIV (protocol_chooser);
395 TpConnectionManager *cm = NULL;
396 gchar *protocol_name = NULL;
397 gboolean visible = FALSE;
399 gtk_tree_model_get (model, iter,
400 COL_CM, &cm,
401 COL_PROTOCOL_NAME, &protocol_name,
402 -1);
404 if (cm != NULL && protocol_name != NULL)
406 TpConnectionManagerProtocol *protocol;
408 protocol = (TpConnectionManagerProtocol *)
409 tp_connection_manager_get_protocol (cm, protocol_name);
411 if (protocol != NULL)
413 visible = priv->filter_func (cm, protocol, priv->filter_user_data);
417 if (cm != NULL)
418 g_object_unref (cm);
420 return visible;
423 /* public methods */
426 * empathy_protocol_chooser_get_selected_protocol:
427 * @protocol_chooser: an #EmpathyProtocolChooser
429 * Returns a pointer to the selected #TpConnectionManagerProtocol in
430 * @protocol_chooser.
432 * Return value: a pointer to the selected #TpConnectionManagerProtocol
434 TpConnectionManager *
435 empathy_protocol_chooser_dup_selected (
436 EmpathyProtocolChooser *protocol_chooser,
437 TpConnectionManagerProtocol **protocol,
438 gboolean *is_gtalk)
440 GtkTreeIter iter;
441 TpConnectionManager *cm = NULL;
442 GtkTreeModel *cur_model;
444 g_return_val_if_fail (EMPATHY_IS_PROTOCOL_CHOOSER (protocol_chooser), NULL);
446 /* get the current model from the chooser, as we could either be filtering
447 * or not.
449 cur_model = gtk_combo_box_get_model (GTK_COMBO_BOX (protocol_chooser));
451 if (gtk_combo_box_get_active_iter (GTK_COMBO_BOX (protocol_chooser), &iter))
453 gtk_tree_model_get (GTK_TREE_MODEL (cur_model), &iter,
454 COL_CM, &cm,
455 -1);
457 if (protocol != NULL)
459 gchar *protocol_name = NULL;
461 gtk_tree_model_get (GTK_TREE_MODEL (cur_model), &iter,
462 COL_PROTOCOL_NAME, &protocol_name,
463 -1);
465 *protocol = (TpConnectionManagerProtocol *)
466 tp_connection_manager_get_protocol (cm, protocol_name);
468 g_free (protocol_name);
471 if (is_gtalk != NULL)
473 gtk_tree_model_get (GTK_TREE_MODEL (cur_model), &iter,
474 COL_IS_GTALK, is_gtalk,
475 -1);
479 return cm;
483 * empathy_protocol_chooser_new:
485 * Triggers the creation of a new #EmpathyProtocolChooser.
487 * Return value: a new #EmpathyProtocolChooser widget
490 GtkWidget *
491 empathy_protocol_chooser_new (void)
493 return GTK_WIDGET (g_object_new (EMPATHY_TYPE_PROTOCOL_CHOOSER, NULL));
496 void
497 empathy_protocol_chooser_set_visible (EmpathyProtocolChooser *protocol_chooser,
498 EmpathyProtocolChooserFilterFunc func,
499 gpointer user_data)
501 EmpathyProtocolChooserPriv *priv;
502 GtkTreeModel *filter_model;
504 g_return_if_fail (EMPATHY_IS_PROTOCOL_CHOOSER (protocol_chooser));
506 priv = GET_PRIV (protocol_chooser);
507 priv->filter_func = func;
508 priv->filter_user_data = user_data;
510 filter_model = gtk_tree_model_filter_new (GTK_TREE_MODEL (priv->store),
511 NULL);
512 gtk_combo_box_set_model (GTK_COMBO_BOX (protocol_chooser), filter_model);
513 g_object_unref (filter_model);
515 gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER
516 (filter_model), protocol_chooser_filter_visible_func,
517 protocol_chooser, NULL);
519 gtk_tree_model_filter_refilter (GTK_TREE_MODEL_FILTER (filter_model));
521 gtk_combo_box_set_active (GTK_COMBO_BOX (protocol_chooser), 0);