]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-contact-list-view.c
Merge branch 'label-align'
[empathy.git] / libempathy-gtk / empathy-contact-list-view.c
1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
2 /*
3  * Copyright (C) 2005-2007 Imendio AB
4  * Copyright (C) 2007-2008 Collabora Ltd.
5  *
6  * This program is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU General Public License as
8  * published by the Free Software Foundation; either version 2 of the
9  * License, or (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public
17  * License along with this program; if not, write to the
18  * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19  * Boston, MA  02110-1301  USA
20  *
21  * Authors: Mikael Hallendal <micke@imendio.com>
22  *          Martyn Russell <martyn@imendio.com>
23  *          Xavier Claessens <xclaesse@gmail.com>
24  */
25
26 #include "config.h"
27
28 #include <string.h>
29
30 #include <glib/gi18n-lib.h>
31 #include <gdk/gdkkeysyms.h>
32 #include <gtk/gtk.h>
33
34 #include <telepathy-glib/util.h>
35
36 #include <libempathy/empathy-account-manager.h>
37 #include <libempathy/empathy-call-factory.h>
38 #include <libempathy/empathy-tp-contact-factory.h>
39 #include <libempathy/empathy-contact-list.h>
40 #include <libempathy/empathy-contact-groups.h>
41 #include <libempathy/empathy-dispatcher.h>
42 #include <libempathy/empathy-utils.h>
43
44 #include "empathy-contact-list-view.h"
45 #include "empathy-contact-list-store.h"
46 #include "empathy-images.h"
47 #include "empathy-cell-renderer-expander.h"
48 #include "empathy-cell-renderer-text.h"
49 #include "empathy-cell-renderer-activatable.h"
50 #include "empathy-ui-utils.h"
51 #include "empathy-gtk-enum-types.h"
52 #include "empathy-gtk-marshal.h"
53
54 #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT
55 #include <libempathy/empathy-debug.h>
56
57 /* Active users are those which have recently changed state
58  * (e.g. online, offline or from normal to a busy state).
59  */
60
61 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyContactListView)
62 typedef struct {
63         EmpathyContactListStore        *store;
64         GtkTreeRowReference            *drag_row;
65         EmpathyContactListFeatureFlags  list_features;
66         EmpathyContactFeatureFlags      contact_features;
67         GtkWidget                      *tooltip_widget;
68 } EmpathyContactListViewPriv;
69
70 typedef struct {
71         EmpathyContactListView *view;
72         GtkTreePath           *path;
73         guint                  timeout_id;
74 } DragMotionData;
75
76 typedef struct {
77         EmpathyContactListView *view;
78         EmpathyContact         *contact;
79         gboolean               remove;
80 } ShowActiveData;
81
82 enum {
83         PROP_0,
84         PROP_STORE,
85         PROP_LIST_FEATURES,
86         PROP_CONTACT_FEATURES,
87 };
88
89 enum DndDragType {
90         DND_DRAG_TYPE_CONTACT_ID,
91         DND_DRAG_TYPE_URL,
92         DND_DRAG_TYPE_STRING,
93 };
94
95 static const GtkTargetEntry drag_types_dest[] = {
96         { "text/contact-id", 0, DND_DRAG_TYPE_CONTACT_ID },
97         { "text/uri-list",   0, DND_DRAG_TYPE_URL },
98         { "text/plain",      0, DND_DRAG_TYPE_STRING },
99         { "STRING",          0, DND_DRAG_TYPE_STRING },
100 };
101
102 static const GtkTargetEntry drag_types_source[] = {
103         { "text/contact-id", 0, DND_DRAG_TYPE_CONTACT_ID },
104 };
105
106 static GdkAtom drag_atoms_dest[G_N_ELEMENTS (drag_types_dest)];
107 static GdkAtom drag_atoms_source[G_N_ELEMENTS (drag_types_source)];
108
109 enum {
110         DRAG_CONTACT_RECEIVED,
111         LAST_SIGNAL
112 };
113
114 static guint signals[LAST_SIGNAL];
115
116 G_DEFINE_TYPE (EmpathyContactListView, empathy_contact_list_view, GTK_TYPE_TREE_VIEW);
117
118 static void
119 contact_list_view_tooltip_destroy_cb (GtkWidget              *widget,
120                                       EmpathyContactListView *view)
121 {
122         EmpathyContactListViewPriv *priv = GET_PRIV (view);
123
124         if (priv->tooltip_widget) {
125                 DEBUG ("Tooltip destroyed");
126                 g_object_unref (priv->tooltip_widget);
127                 priv->tooltip_widget = NULL;
128         }
129 }
130
131 static gboolean
132 contact_list_view_query_tooltip_cb (EmpathyContactListView *view,
133                                     gint                    x,
134                                     gint                    y,
135                                     gboolean                keyboard_mode,
136                                     GtkTooltip             *tooltip,
137                                     gpointer                user_data)
138 {
139         EmpathyContactListViewPriv *priv = GET_PRIV (view);
140         EmpathyContact             *contact;
141         GtkTreeModel               *model;
142         GtkTreeIter                 iter;
143         GtkTreePath                *path;
144         static gint                 running = 0;
145         gboolean                    ret = FALSE;
146
147         /* Avoid an infinite loop. See GNOME bug #574377 */
148         if (running > 0) {
149                 return FALSE;
150         }
151         running++;
152
153         if (!gtk_tree_view_get_tooltip_context (GTK_TREE_VIEW (view), &x, &y,
154                                                 keyboard_mode,
155                                                 &model, &path, &iter)) {
156                 goto OUT;
157         }
158
159         gtk_tree_view_set_tooltip_row (GTK_TREE_VIEW (view), tooltip, path);
160         gtk_tree_path_free (path);
161
162         gtk_tree_model_get (model, &iter,
163                             EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
164                             -1);
165         if (!contact) {
166                 goto OUT;
167         }
168
169         if (!priv->tooltip_widget) {
170                 priv->tooltip_widget = empathy_contact_widget_new (contact,
171                         EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP |
172                         EMPATHY_CONTACT_WIDGET_SHOW_LOCATION);
173                 g_object_ref (priv->tooltip_widget);
174                 g_signal_connect (priv->tooltip_widget, "destroy",
175                                   G_CALLBACK (contact_list_view_tooltip_destroy_cb),
176                                   view);
177                 gtk_widget_show (priv->tooltip_widget);
178         } else {
179                 empathy_contact_widget_set_contact (priv->tooltip_widget,
180                                                     contact);
181         }
182
183         gtk_tooltip_set_custom (tooltip, priv->tooltip_widget);
184         ret = TRUE;
185
186         g_object_unref (contact);
187 OUT:
188         running--;
189
190         return ret;
191 }
192
193 typedef struct {
194         gchar *new_group;
195         gchar *old_group;
196         GdkDragAction action;
197 } DndGetContactData;
198
199 static void
200 contact_list_view_dnd_get_contact_free (DndGetContactData *data)
201 {
202         g_free (data->new_group);
203         g_free (data->old_group);
204         g_slice_free (DndGetContactData, data);
205 }
206
207 static void
208 contact_list_view_drag_got_contact (EmpathyTpContactFactory *factory,
209                                     EmpathyContact          *contact,
210                                     const GError            *error,
211                                     gpointer                 user_data,
212                                     GObject                 *view)
213 {
214         EmpathyContactListViewPriv *priv = GET_PRIV (view);
215         DndGetContactData          *data = user_data;
216         EmpathyContactList         *list;
217
218         if (error != NULL) {
219                 DEBUG ("Error: %s", error->message);
220                 return;
221         }
222
223         DEBUG ("contact %s (%d) dragged from '%s' to '%s'",
224                 empathy_contact_get_id (contact),
225                 empathy_contact_get_handle (contact),
226                 data->old_group, data->new_group);
227
228         list = empathy_contact_list_store_get_list_iface (priv->store);
229         if (data->new_group) {
230                 empathy_contact_list_add_to_group (list, contact, data->new_group);
231         }
232         if (data->old_group && data->action == GDK_ACTION_MOVE) {
233                 empathy_contact_list_remove_from_group (list, contact, data->old_group);
234         }
235 }
236
237 static void
238 contact_list_view_drag_data_received (GtkWidget         *view,
239                                       GdkDragContext    *context,
240                                       gint               x,
241                                       gint               y,
242                                       GtkSelectionData  *selection,
243                                       guint              info,
244                                       guint              time)
245 {
246         EmpathyContactListViewPriv *priv;
247         EmpathyAccountManager      *account_manager;
248         EmpathyTpContactFactory    *factory = NULL;
249         EmpathyAccount             *account;
250         GtkTreeModel               *model;
251         GtkTreeViewDropPosition     position;
252         GtkTreePath                *path;
253         const gchar                *id;
254         gchar                     **strv = NULL;
255         const gchar                *account_id;
256         const gchar                *contact_id;
257         gchar                      *new_group = NULL;
258         gchar                      *old_group = NULL;
259         DndGetContactData          *data;
260         gboolean                    is_row;
261         gboolean                    success = TRUE;
262
263         priv = GET_PRIV (view);
264         model = gtk_tree_view_get_model (GTK_TREE_VIEW (view));
265
266         /* Get destination group information. */
267         is_row = gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (view),
268                                                     x,
269                                                     y,
270                                                     &path,
271                                                     &position);
272
273         if (is_row) {
274                 new_group = empathy_contact_list_store_get_parent_group (model,
275                         path, NULL);
276                 gtk_tree_path_free (path);
277         }
278
279         /* Get source group information. */
280         if (priv->drag_row) {
281                 path = gtk_tree_row_reference_get_path (priv->drag_row);
282                 if (path) {
283                         old_group = empathy_contact_list_store_get_parent_group (
284                                 model, path, NULL);
285                         gtk_tree_path_free (path);
286                 }
287         }
288
289         if (!tp_strdiff (old_group, new_group)) {
290                 g_free (new_group);
291                 g_free (old_group);
292                 goto OUT;
293         }
294
295         id = (const gchar*) gtk_selection_data_get_data (selection);
296         DEBUG ("Received %s%s drag & drop contact from roster with id:'%s'",
297                 context->action == GDK_ACTION_MOVE ? "move" : "",
298                 context->action == GDK_ACTION_COPY ? "copy" : "",
299                 id);
300
301         strv = g_strsplit (id, "/", 2);
302         account_id = strv[0];
303         contact_id = strv[1];
304   account_manager = empathy_account_manager_dup_singleton ();
305         account = empathy_account_manager_lookup (account_manager, account_id);
306         if (account) {
307                 TpConnection *connection;
308
309                 connection = empathy_account_get_connection (account);
310                 if (connection) {
311                         factory = empathy_tp_contact_factory_dup_singleton (connection);
312                 }
313         }
314         g_object_unref (account_manager);
315
316         if (!factory) {
317                 DEBUG ("Failed to get factory for account '%s'", account_id);
318                 success = FALSE;
319                 g_free (new_group);
320                 g_free (old_group);
321                 goto OUT;
322         }
323
324         data = g_slice_new0 (DndGetContactData);
325         data->new_group = new_group;
326         data->old_group = old_group;
327         data->action = context->action;
328
329         /* FIXME: We should probably wait for the cb before calling
330          * gtk_drag_finish */
331         empathy_tp_contact_factory_get_from_id (factory, contact_id,
332                 contact_list_view_drag_got_contact,
333                 data, (GDestroyNotify) contact_list_view_dnd_get_contact_free,
334                 G_OBJECT (view));
335
336         g_object_unref (factory);
337
338 OUT:
339         g_strfreev (strv);
340         gtk_drag_finish (context, success, FALSE, GDK_CURRENT_TIME);
341 }
342
343 static gboolean
344 contact_list_view_drag_motion_cb (DragMotionData *data)
345 {
346         gtk_tree_view_expand_row (GTK_TREE_VIEW (data->view),
347                                   data->path,
348                                   FALSE);
349
350         data->timeout_id = 0;
351
352         return FALSE;
353 }
354
355 static gboolean
356 contact_list_view_drag_motion (GtkWidget      *widget,
357                                GdkDragContext *context,
358                                gint            x,
359                                gint            y,
360                                guint           time)
361 {
362         static DragMotionData *dm = NULL;
363         GtkTreePath           *path;
364         gboolean               is_row;
365         gboolean               is_different = FALSE;
366         gboolean               cleanup = TRUE;
367
368         is_row = gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (widget),
369                                                 x,
370                                                 y,
371                                                 &path,
372                                                 NULL,
373                                                 NULL,
374                                                 NULL);
375
376         cleanup &= (!dm);
377
378         if (is_row) {
379                 cleanup &= (dm && gtk_tree_path_compare (dm->path, path) != 0);
380                 is_different = (!dm || (dm && gtk_tree_path_compare (dm->path, path) != 0));
381         } else {
382                 cleanup &= FALSE;
383         }
384
385         if (!is_different && !cleanup) {
386                 return TRUE;
387         }
388
389         if (dm) {
390                 gtk_tree_path_free (dm->path);
391                 if (dm->timeout_id) {
392                         g_source_remove (dm->timeout_id);
393                 }
394
395                 g_free (dm);
396
397                 dm = NULL;
398         }
399
400         if (!gtk_tree_view_row_expanded (GTK_TREE_VIEW (widget), path)) {
401                 dm = g_new0 (DragMotionData, 1);
402
403                 dm->view = EMPATHY_CONTACT_LIST_VIEW (widget);
404                 dm->path = gtk_tree_path_copy (path);
405
406                 dm->timeout_id = g_timeout_add_seconds (1,
407                         (GSourceFunc) contact_list_view_drag_motion_cb,
408                         dm);
409         }
410
411         return TRUE;
412 }
413
414 static void
415 contact_list_view_drag_begin (GtkWidget      *widget,
416                               GdkDragContext *context)
417 {
418         EmpathyContactListViewPriv *priv;
419         GtkTreeSelection          *selection;
420         GtkTreeModel              *model;
421         GtkTreePath               *path;
422         GtkTreeIter                iter;
423
424         priv = GET_PRIV (widget);
425
426         GTK_WIDGET_CLASS (empathy_contact_list_view_parent_class)->drag_begin (widget,
427                                                                               context);
428
429         selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (widget));
430         if (!gtk_tree_selection_get_selected (selection, &model, &iter)) {
431                 return;
432         }
433
434         path = gtk_tree_model_get_path (model, &iter);
435         priv->drag_row = gtk_tree_row_reference_new (model, path);
436         gtk_tree_path_free (path);
437 }
438
439 static void
440 contact_list_view_drag_data_get (GtkWidget        *widget,
441                                  GdkDragContext   *context,
442                                  GtkSelectionData *selection,
443                                  guint             info,
444                                  guint             time)
445 {
446         EmpathyContactListViewPriv *priv;
447         GtkTreePath                *src_path;
448         GtkTreeIter                 iter;
449         GtkTreeModel               *model;
450         EmpathyContact             *contact;
451         EmpathyAccount             *account;
452         const gchar                *contact_id;
453         const gchar                *account_id;
454         gchar                      *str;
455
456         priv = GET_PRIV (widget);
457
458         model = gtk_tree_view_get_model (GTK_TREE_VIEW (widget));
459         if (!priv->drag_row) {
460                 return;
461         }
462
463         src_path = gtk_tree_row_reference_get_path (priv->drag_row);
464         if (!src_path) {
465                 return;
466         }
467
468         if (!gtk_tree_model_get_iter (model, &iter, src_path)) {
469                 gtk_tree_path_free (src_path);
470                 return;
471         }
472
473         gtk_tree_path_free (src_path);
474
475         contact = empathy_contact_list_view_dup_selected (EMPATHY_CONTACT_LIST_VIEW (widget));
476         if (!contact) {
477                 return;
478         }
479
480         account = empathy_contact_get_account (contact);
481         account_id = empathy_account_get_unique_name (account);
482         contact_id = empathy_contact_get_id (contact);
483         g_object_unref (contact);
484         str = g_strconcat (account_id, "/", contact_id, NULL);
485
486         switch (info) {
487         case DND_DRAG_TYPE_CONTACT_ID:
488                 gtk_selection_data_set (selection, drag_atoms_source[info], 8,
489                                         (guchar *) str, strlen (str) + 1);
490                 break;
491         }
492
493         g_free (str);
494 }
495
496 static void
497 contact_list_view_drag_end (GtkWidget      *widget,
498                             GdkDragContext *context)
499 {
500         EmpathyContactListViewPriv *priv;
501
502         priv = GET_PRIV (widget);
503
504         GTK_WIDGET_CLASS (empathy_contact_list_view_parent_class)->drag_end (widget,
505                                                                             context);
506
507         if (priv->drag_row) {
508                 gtk_tree_row_reference_free (priv->drag_row);
509                 priv->drag_row = NULL;
510         }
511 }
512
513 static gboolean
514 contact_list_view_drag_drop (GtkWidget      *widget,
515                              GdkDragContext *drag_context,
516                              gint            x,
517                              gint            y,
518                              guint           time)
519 {
520         return FALSE;
521 }
522
523 typedef struct {
524         EmpathyContactListView *view;
525         guint                   button;
526         guint32                 time;
527 } MenuPopupData;
528
529 static gboolean
530 contact_list_view_popup_menu_idle_cb (gpointer user_data)
531 {
532         MenuPopupData *data = user_data;
533         GtkWidget     *menu;
534
535         menu = empathy_contact_list_view_get_contact_menu (data->view);
536         if (!menu) {
537                 menu = empathy_contact_list_view_get_group_menu (data->view);
538         }
539
540         if (menu) {
541                 gtk_widget_show (menu);
542                 gtk_menu_popup (GTK_MENU (menu),
543                                 NULL, NULL, NULL, NULL,
544                                 data->button, data->time);
545         }
546
547         g_slice_free (MenuPopupData, data);
548
549         return FALSE;
550 }
551
552 static gboolean
553 contact_list_view_button_press_event_cb (EmpathyContactListView *view,
554                                          GdkEventButton         *event,
555                                          gpointer                user_data)
556 {
557         if (event->button == 3) {
558                 MenuPopupData *data;
559
560                 data = g_slice_new (MenuPopupData);
561                 data->view = view;
562                 data->button = event->button;
563                 data->time = event->time;
564                 g_idle_add (contact_list_view_popup_menu_idle_cb, data);
565         }
566
567         return FALSE;
568 }
569
570 static gboolean
571 contact_list_view_key_press_event_cb (EmpathyContactListView *view,
572                                       GdkEventKey            *event,
573                                       gpointer                user_data)
574 {
575         if (event->keyval == GDK_Menu) {
576                 MenuPopupData *data;
577
578                 data = g_slice_new (MenuPopupData);
579                 data->view = view;
580                 data->button = 0;
581                 data->time = event->time;
582                 g_idle_add (contact_list_view_popup_menu_idle_cb, data);
583         }
584
585         return FALSE;
586 }
587
588 static void
589 contact_list_view_row_activated (GtkTreeView       *view,
590                                  GtkTreePath       *path,
591                                  GtkTreeViewColumn *column)
592 {
593         EmpathyContactListViewPriv *priv = GET_PRIV (view);
594         EmpathyContact             *contact;
595         GtkTreeModel               *model;
596         GtkTreeIter                 iter;
597
598         if (!(priv->contact_features & EMPATHY_CONTACT_FEATURE_CHAT)) {
599                 return;
600         }
601
602         model = GTK_TREE_MODEL (priv->store);
603         gtk_tree_model_get_iter (model, &iter, path);
604         gtk_tree_model_get (model, &iter,
605                             EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
606                             -1);
607
608         if (contact) {
609                 DEBUG ("Starting a chat");
610                 empathy_dispatcher_chat_with_contact (contact, NULL, NULL);
611                 g_object_unref (contact);
612         }
613 }
614
615 static void
616 contact_list_start_voip_call (EmpathyCellRendererActivatable *cell,
617     const gchar                    *path_string,
618     EmpathyContactListView         *view,
619     gboolean with_video)
620 {
621         EmpathyContactListViewPriv *priv = GET_PRIV (view);
622         GtkTreeModel               *model;
623         GtkTreeIter                 iter;
624         EmpathyContact             *contact;
625
626         if (!(priv->contact_features & EMPATHY_CONTACT_FEATURE_CALL)) {
627                 return;
628         }
629
630         model = gtk_tree_view_get_model (GTK_TREE_VIEW (view));
631         if (!gtk_tree_model_get_iter_from_string (model, &iter, path_string)) {
632                 return;
633         }
634
635         gtk_tree_model_get (model, &iter,
636                             EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
637                             -1);
638
639         if (contact) {
640                 EmpathyCallFactory *factory;
641                 factory = empathy_call_factory_get ();
642                 empathy_call_factory_new_call_with_streams (factory, contact,
643                         TRUE, with_video);
644                 g_object_unref (contact);
645         }
646 }
647
648 static void
649 contact_list_view_video_call_activated_cb (
650     EmpathyCellRendererActivatable *cell,
651     const gchar                    *path_string,
652     EmpathyContactListView         *view)
653 {
654   contact_list_start_voip_call (cell, path_string, view, TRUE);
655 }
656
657
658 static void
659 contact_list_view_audio_call_activated_cb (EmpathyCellRendererActivatable *cell,
660                                      const gchar                    *path_string,
661                                      EmpathyContactListView         *view)
662 {
663   contact_list_start_voip_call (cell, path_string, view, FALSE);
664 }
665
666 static void
667 contact_list_view_cell_set_background (EmpathyContactListView *view,
668                                        GtkCellRenderer       *cell,
669                                        gboolean               is_group,
670                                        gboolean               is_active)
671 {
672         GdkColor  color;
673         GtkStyle *style;
674
675         style = gtk_widget_get_style (GTK_WIDGET (view));
676
677         if (!is_group && is_active) {
678                 color = style->bg[GTK_STATE_SELECTED];
679
680                 /* Here we take the current theme colour and add it to
681                  * the colour for white and average the two. This
682                  * gives a colour which is inline with the theme but
683                  * slightly whiter.
684                  */
685                 color.red = (color.red + (style->white).red) / 2;
686                 color.green = (color.green + (style->white).green) / 2;
687                 color.blue = (color.blue + (style->white).blue) / 2;
688
689                 g_object_set (cell,
690                               "cell-background-gdk", &color,
691                               NULL);
692         } else {
693                 g_object_set (cell,
694                               "cell-background-gdk", NULL,
695                               NULL);
696         }
697 }
698
699 static void
700 contact_list_view_pixbuf_cell_data_func (GtkTreeViewColumn     *tree_column,
701                                          GtkCellRenderer       *cell,
702                                          GtkTreeModel          *model,
703                                          GtkTreeIter           *iter,
704                                          EmpathyContactListView *view)
705 {
706         gchar    *icon_name;
707         gboolean  is_group;
708         gboolean  is_active;
709
710         gtk_tree_model_get (model, iter,
711                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
712                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
713                             EMPATHY_CONTACT_LIST_STORE_COL_ICON_STATUS, &icon_name,
714                             -1);
715
716         g_object_set (cell,
717                       "visible", !is_group,
718                       "icon-name", icon_name,
719                       NULL);
720
721         g_free (icon_name);
722
723         contact_list_view_cell_set_background (view, cell, is_group, is_active);
724 }
725
726 static void
727 contact_list_view_audio_call_cell_data_func (
728                                        GtkTreeViewColumn      *tree_column,
729                                        GtkCellRenderer        *cell,
730                                        GtkTreeModel           *model,
731                                        GtkTreeIter            *iter,
732                                        EmpathyContactListView *view)
733 {
734         gboolean is_group;
735         gboolean is_active;
736         gboolean can_voip;
737
738         gtk_tree_model_get (model, iter,
739                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
740                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
741                             EMPATHY_CONTACT_LIST_STORE_COL_CAN_AUDIO_CALL, &can_voip,
742                             -1);
743
744         g_object_set (cell,
745                       "visible", !is_group && can_voip,
746                       "icon-name", EMPATHY_IMAGE_VOIP,
747                       NULL);
748
749         contact_list_view_cell_set_background (view, cell, is_group, is_active);
750 }
751
752 static void
753 contact_list_view_video_call_cell_data_func (
754                                        GtkTreeViewColumn      *tree_column,
755                                        GtkCellRenderer        *cell,
756                                        GtkTreeModel           *model,
757                                        GtkTreeIter            *iter,
758                                        EmpathyContactListView *view)
759 {
760         gboolean is_group;
761         gboolean is_active;
762         gboolean can_voip;
763
764         gtk_tree_model_get (model, iter,
765                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
766                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
767                             EMPATHY_CONTACT_LIST_STORE_COL_CAN_VIDEO_CALL, &can_voip,
768                             -1);
769
770         g_object_set (cell,
771                       "visible", !is_group && can_voip,
772                       "icon-name", EMPATHY_IMAGE_VIDEO_CALL,
773                       NULL);
774
775         contact_list_view_cell_set_background (view, cell, is_group, is_active);
776 }
777
778
779 static void
780 contact_list_view_avatar_cell_data_func (GtkTreeViewColumn     *tree_column,
781                                          GtkCellRenderer       *cell,
782                                          GtkTreeModel          *model,
783                                          GtkTreeIter           *iter,
784                                          EmpathyContactListView *view)
785 {
786         GdkPixbuf *pixbuf;
787         gboolean   show_avatar;
788         gboolean   is_group;
789         gboolean   is_active;
790
791         gtk_tree_model_get (model, iter,
792                             EMPATHY_CONTACT_LIST_STORE_COL_PIXBUF_AVATAR, &pixbuf,
793                             EMPATHY_CONTACT_LIST_STORE_COL_PIXBUF_AVATAR_VISIBLE, &show_avatar,
794                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
795                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
796                             -1);
797
798         g_object_set (cell,
799                       "visible", !is_group && show_avatar,
800                       "pixbuf", pixbuf,
801                       NULL);
802
803         if (pixbuf) {
804                 g_object_unref (pixbuf);
805         }
806
807         contact_list_view_cell_set_background (view, cell, is_group, is_active);
808 }
809
810 static void
811 contact_list_view_text_cell_data_func (GtkTreeViewColumn     *tree_column,
812                                        GtkCellRenderer       *cell,
813                                        GtkTreeModel          *model,
814                                        GtkTreeIter           *iter,
815                                        EmpathyContactListView *view)
816 {
817         gboolean is_group;
818         gboolean is_active;
819         gboolean show_status;
820
821         gtk_tree_model_get (model, iter,
822                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
823                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
824                             EMPATHY_CONTACT_LIST_STORE_COL_STATUS_VISIBLE, &show_status,
825                             -1);
826
827         g_object_set (cell,
828                       "show-status", show_status,
829                       NULL);
830
831         contact_list_view_cell_set_background (view, cell, is_group, is_active);
832 }
833
834 static void
835 contact_list_view_expander_cell_data_func (GtkTreeViewColumn     *column,
836                                            GtkCellRenderer       *cell,
837                                            GtkTreeModel          *model,
838                                            GtkTreeIter           *iter,
839                                            EmpathyContactListView *view)
840 {
841         gboolean is_group;
842         gboolean is_active;
843
844         gtk_tree_model_get (model, iter,
845                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
846                             EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, &is_active,
847                             -1);
848
849         if (gtk_tree_model_iter_has_child (model, iter)) {
850                 GtkTreePath *path;
851                 gboolean     row_expanded;
852
853                 path = gtk_tree_model_get_path (model, iter);
854                 row_expanded = gtk_tree_view_row_expanded (GTK_TREE_VIEW (gtk_tree_view_column_get_tree_view (column)), path);
855                 gtk_tree_path_free (path);
856
857                 g_object_set (cell,
858                               "visible", TRUE,
859                               "expander-style", row_expanded ? GTK_EXPANDER_EXPANDED : GTK_EXPANDER_COLLAPSED,
860                               NULL);
861         } else {
862                 g_object_set (cell, "visible", FALSE, NULL);
863         }
864
865         contact_list_view_cell_set_background (view, cell, is_group, is_active);
866 }
867
868 static void
869 contact_list_view_row_expand_or_collapse_cb (EmpathyContactListView *view,
870                                              GtkTreeIter           *iter,
871                                              GtkTreePath           *path,
872                                              gpointer               user_data)
873 {
874         EmpathyContactListViewPriv *priv = GET_PRIV (view);
875         GtkTreeModel               *model;
876         gchar                      *name;
877         gboolean                    expanded;
878
879         if (!(priv->list_features & EMPATHY_CONTACT_LIST_FEATURE_GROUPS_SAVE)) {
880                 return;
881         }
882
883         model = gtk_tree_view_get_model (GTK_TREE_VIEW (view));
884
885         gtk_tree_model_get (model, iter,
886                             EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name,
887                             -1);
888
889         expanded = GPOINTER_TO_INT (user_data);
890         empathy_contact_group_set_expanded (name, expanded);
891
892         g_free (name);
893 }
894
895 static void
896 contact_list_view_row_has_child_toggled_cb (GtkTreeModel          *model,
897                                             GtkTreePath           *path,
898                                             GtkTreeIter           *iter,
899                                             EmpathyContactListView *view)
900 {
901         EmpathyContactListViewPriv *priv = GET_PRIV (view);
902         gboolean  is_group = FALSE;
903         gchar    *name = NULL;
904
905         gtk_tree_model_get (model, iter,
906                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
907                             EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name,
908                             -1);
909
910         if (!is_group || EMP_STR_EMPTY (name)) {
911                 g_free (name);
912                 return;
913         }
914
915         if (!(priv->list_features & EMPATHY_CONTACT_LIST_FEATURE_GROUPS_SAVE) ||
916             empathy_contact_group_get_expanded (name)) {
917                 g_signal_handlers_block_by_func (view,
918                                                  contact_list_view_row_expand_or_collapse_cb,
919                                                  GINT_TO_POINTER (TRUE));
920                 gtk_tree_view_expand_row (GTK_TREE_VIEW (view), path, TRUE);
921                 g_signal_handlers_unblock_by_func (view,
922                                                    contact_list_view_row_expand_or_collapse_cb,
923                                                    GINT_TO_POINTER (TRUE));
924         } else {
925                 g_signal_handlers_block_by_func (view,
926                                                  contact_list_view_row_expand_or_collapse_cb,
927                                                  GINT_TO_POINTER (FALSE));
928                 gtk_tree_view_collapse_row (GTK_TREE_VIEW (view), path);
929                 g_signal_handlers_unblock_by_func (view,
930                                                    contact_list_view_row_expand_or_collapse_cb,
931                                                    GINT_TO_POINTER (FALSE));
932         }
933
934         g_free (name);
935 }
936
937 static void
938 contact_list_view_setup (EmpathyContactListView *view)
939 {
940         EmpathyContactListViewPriv *priv;
941         GtkCellRenderer           *cell;
942         GtkTreeViewColumn         *col;
943         gint                       i;
944
945         priv = GET_PRIV (view);
946
947         gtk_tree_view_set_search_equal_func (GTK_TREE_VIEW (view),
948                                              empathy_contact_list_store_search_equal_func,
949                                              NULL, NULL);
950
951         g_signal_connect (priv->store, "row-has-child-toggled",
952                           G_CALLBACK (contact_list_view_row_has_child_toggled_cb),
953                           view);
954         gtk_tree_view_set_model (GTK_TREE_VIEW (view),
955                                  GTK_TREE_MODEL (priv->store));
956
957         /* Setup view */
958         g_object_set (view,
959                       "headers-visible", FALSE,
960                       "reorderable", TRUE,
961                       "show-expanders", FALSE,
962                       NULL);
963
964         col = gtk_tree_view_column_new ();
965
966         /* State */
967         cell = gtk_cell_renderer_pixbuf_new ();
968         gtk_tree_view_column_pack_start (col, cell, FALSE);
969         gtk_tree_view_column_set_cell_data_func (
970                 col, cell,
971                 (GtkTreeCellDataFunc) contact_list_view_pixbuf_cell_data_func,
972                 view, NULL);
973
974         g_object_set (cell,
975                       "xpad", 5,
976                       "ypad", 1,
977                       "visible", FALSE,
978                       NULL);
979
980         /* Name */
981         cell = empathy_cell_renderer_text_new ();
982         gtk_tree_view_column_pack_start (col, cell, TRUE);
983         gtk_tree_view_column_set_cell_data_func (
984                 col, cell,
985                 (GtkTreeCellDataFunc) contact_list_view_text_cell_data_func,
986                 view, NULL);
987
988         gtk_tree_view_column_add_attribute (col, cell,
989                                             "name", EMPATHY_CONTACT_LIST_STORE_COL_NAME);
990         gtk_tree_view_column_add_attribute (col, cell,
991                                             "status", EMPATHY_CONTACT_LIST_STORE_COL_STATUS);
992         gtk_tree_view_column_add_attribute (col, cell,
993                                             "is_group", EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP);
994
995         /* Audio Call Icon */
996         cell = empathy_cell_renderer_activatable_new ();
997         gtk_tree_view_column_pack_start (col, cell, FALSE);
998         gtk_tree_view_column_set_cell_data_func (
999                 col, cell,
1000                 (GtkTreeCellDataFunc) contact_list_view_audio_call_cell_data_func,
1001                 view, NULL);
1002
1003         g_object_set (cell,
1004                       "visible", FALSE,
1005                       NULL);
1006
1007         g_signal_connect (cell, "path-activated",
1008                           G_CALLBACK (contact_list_view_audio_call_activated_cb),
1009                           view);
1010
1011         /* Video Call Icon */
1012         cell = empathy_cell_renderer_activatable_new ();
1013         gtk_tree_view_column_pack_start (col, cell, FALSE);
1014         gtk_tree_view_column_set_cell_data_func (
1015                 col, cell,
1016                 (GtkTreeCellDataFunc) contact_list_view_video_call_cell_data_func,
1017                 view, NULL);
1018
1019         g_object_set (cell,
1020                       "visible", FALSE,
1021                       NULL);
1022
1023         g_signal_connect (cell, "path-activated",
1024                           G_CALLBACK (contact_list_view_video_call_activated_cb),
1025                           view);
1026
1027         /* Avatar */
1028         cell = gtk_cell_renderer_pixbuf_new ();
1029         gtk_tree_view_column_pack_start (col, cell, FALSE);
1030         gtk_tree_view_column_set_cell_data_func (
1031                 col, cell,
1032                 (GtkTreeCellDataFunc) contact_list_view_avatar_cell_data_func,
1033                 view, NULL);
1034
1035         g_object_set (cell,
1036                       "xpad", 0,
1037                       "ypad", 0,
1038                       "visible", FALSE,
1039                       "width", 32,
1040                       "height", 32,
1041                       NULL);
1042
1043         /* Expander */
1044         cell = empathy_cell_renderer_expander_new ();
1045         gtk_tree_view_column_pack_end (col, cell, FALSE);
1046         gtk_tree_view_column_set_cell_data_func (
1047                 col, cell,
1048                 (GtkTreeCellDataFunc) contact_list_view_expander_cell_data_func,
1049                 view, NULL);
1050
1051         /* Actually add the column now we have added all cell renderers */
1052         gtk_tree_view_append_column (GTK_TREE_VIEW (view), col);
1053
1054         /* Drag & Drop. */
1055         for (i = 0; i < G_N_ELEMENTS (drag_types_dest); ++i) {
1056                 drag_atoms_dest[i] = gdk_atom_intern (drag_types_dest[i].target,
1057                                                       FALSE);
1058         }
1059
1060         for (i = 0; i < G_N_ELEMENTS (drag_types_source); ++i) {
1061                 drag_atoms_source[i] = gdk_atom_intern (drag_types_source[i].target,
1062                                                         FALSE);
1063         }
1064 }
1065
1066 static void
1067 contact_list_view_set_list_features (EmpathyContactListView         *view,
1068                                      EmpathyContactListFeatureFlags  features)
1069 {
1070         EmpathyContactListViewPriv *priv = GET_PRIV (view);
1071         gboolean                    has_tooltip;
1072
1073         g_return_if_fail (EMPATHY_IS_CONTACT_LIST_VIEW (view));
1074
1075         priv->list_features = features;
1076
1077         /* Update DnD source/dest */
1078         if (features & EMPATHY_CONTACT_LIST_FEATURE_CONTACT_DRAG) {
1079                 gtk_drag_source_set (GTK_WIDGET (view),
1080                                      GDK_BUTTON1_MASK,
1081                                      drag_types_source,
1082                                      G_N_ELEMENTS (drag_types_source),
1083                                      GDK_ACTION_MOVE | GDK_ACTION_COPY);
1084         } else {
1085                 gtk_drag_source_unset (GTK_WIDGET (view));
1086
1087         }
1088
1089         if (features & EMPATHY_CONTACT_LIST_FEATURE_CONTACT_DROP) {
1090                 gtk_drag_dest_set (GTK_WIDGET (view),
1091                                    GTK_DEST_DEFAULT_ALL,
1092                                    drag_types_dest,
1093                                    G_N_ELEMENTS (drag_types_dest),
1094                                    GDK_ACTION_MOVE | GDK_ACTION_COPY);
1095         } else {
1096                 /* FIXME: URI could still be droped depending on FT feature */
1097                 gtk_drag_dest_unset (GTK_WIDGET (view));
1098         }
1099
1100         /* Update has-tooltip */
1101         has_tooltip = (features & EMPATHY_CONTACT_LIST_FEATURE_CONTACT_TOOLTIP) != 0;
1102         gtk_widget_set_has_tooltip (GTK_WIDGET (view), has_tooltip);
1103 }
1104
1105 static void
1106 contact_list_view_finalize (GObject *object)
1107 {
1108         EmpathyContactListViewPriv *priv;
1109
1110         priv = GET_PRIV (object);
1111
1112         if (priv->store) {
1113                 g_object_unref (priv->store);
1114         }
1115         if (priv->tooltip_widget) {
1116                 gtk_widget_destroy (priv->tooltip_widget);
1117         }
1118
1119         G_OBJECT_CLASS (empathy_contact_list_view_parent_class)->finalize (object);
1120 }
1121
1122 static void
1123 contact_list_view_get_property (GObject    *object,
1124                                 guint       param_id,
1125                                 GValue     *value,
1126                                 GParamSpec *pspec)
1127 {
1128         EmpathyContactListViewPriv *priv;
1129
1130         priv = GET_PRIV (object);
1131
1132         switch (param_id) {
1133         case PROP_STORE:
1134                 g_value_set_object (value, priv->store);
1135                 break;
1136         case PROP_LIST_FEATURES:
1137                 g_value_set_flags (value, priv->list_features);
1138                 break;
1139         case PROP_CONTACT_FEATURES:
1140                 g_value_set_flags (value, priv->contact_features);
1141                 break;
1142         default:
1143                 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
1144                 break;
1145         };
1146 }
1147
1148 static void
1149 contact_list_view_set_property (GObject      *object,
1150                                 guint         param_id,
1151                                 const GValue *value,
1152                                 GParamSpec   *pspec)
1153 {
1154         EmpathyContactListView     *view = EMPATHY_CONTACT_LIST_VIEW (object);
1155         EmpathyContactListViewPriv *priv = GET_PRIV (object);
1156
1157         switch (param_id) {
1158         case PROP_STORE:
1159                 priv->store = g_value_dup_object (value);
1160                 contact_list_view_setup (view);
1161                 break;
1162         case PROP_LIST_FEATURES:
1163                 contact_list_view_set_list_features (view, g_value_get_flags (value));
1164                 break;
1165         case PROP_CONTACT_FEATURES:
1166                 priv->contact_features = g_value_get_flags (value);
1167                 break;
1168         default:
1169                 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
1170                 break;
1171         };
1172 }
1173
1174 static void
1175 empathy_contact_list_view_class_init (EmpathyContactListViewClass *klass)
1176 {
1177         GObjectClass     *object_class = G_OBJECT_CLASS (klass);
1178         GtkWidgetClass   *widget_class = GTK_WIDGET_CLASS (klass);
1179         GtkTreeViewClass *tree_view_class = GTK_TREE_VIEW_CLASS (klass);
1180
1181         object_class->finalize = contact_list_view_finalize;
1182         object_class->get_property = contact_list_view_get_property;
1183         object_class->set_property = contact_list_view_set_property;
1184
1185         widget_class->drag_data_received = contact_list_view_drag_data_received;
1186         widget_class->drag_drop          = contact_list_view_drag_drop;
1187         widget_class->drag_begin         = contact_list_view_drag_begin;
1188         widget_class->drag_data_get      = contact_list_view_drag_data_get;
1189         widget_class->drag_end           = contact_list_view_drag_end;
1190         widget_class->drag_motion        = contact_list_view_drag_motion;
1191
1192         /* We use the class method to let user of this widget to connect to
1193          * the signal and stop emission of the signal so the default handler
1194          * won't be called. */
1195         tree_view_class->row_activated = contact_list_view_row_activated;
1196
1197         signals[DRAG_CONTACT_RECEIVED] =
1198                 g_signal_new ("drag-contact-received",
1199                               G_OBJECT_CLASS_TYPE (klass),
1200                               G_SIGNAL_RUN_LAST,
1201                               0,
1202                               NULL, NULL,
1203                               _empathy_gtk_marshal_VOID__OBJECT_STRING_STRING,
1204                               G_TYPE_NONE,
1205                               3, EMPATHY_TYPE_CONTACT, G_TYPE_STRING, G_TYPE_STRING);
1206
1207         g_object_class_install_property (object_class,
1208                                          PROP_STORE,
1209                                          g_param_spec_object ("store",
1210                                                              "The store of the view",
1211                                                              "The store of the view",
1212                                                               EMPATHY_TYPE_CONTACT_LIST_STORE,
1213                                                               G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE));
1214         g_object_class_install_property (object_class,
1215                                          PROP_LIST_FEATURES,
1216                                          g_param_spec_flags ("list-features",
1217                                                              "Features of the view",
1218                                                              "Falgs for all enabled features",
1219                                                               EMPATHY_TYPE_CONTACT_LIST_FEATURE_FLAGS,
1220                                                               EMPATHY_CONTACT_LIST_FEATURE_NONE,
1221                                                               G_PARAM_READWRITE));
1222         g_object_class_install_property (object_class,
1223                                          PROP_CONTACT_FEATURES,
1224                                          g_param_spec_flags ("contact-features",
1225                                                              "Features of the contact menu",
1226                                                              "Falgs for all enabled features for the menu",
1227                                                               EMPATHY_TYPE_CONTACT_FEATURE_FLAGS,
1228                                                               EMPATHY_CONTACT_FEATURE_NONE,
1229                                                               G_PARAM_READWRITE));
1230
1231         g_type_class_add_private (object_class, sizeof (EmpathyContactListViewPriv));
1232 }
1233
1234 static void
1235 empathy_contact_list_view_init (EmpathyContactListView *view)
1236 {
1237         EmpathyContactListViewPriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (view,
1238                 EMPATHY_TYPE_CONTACT_LIST_VIEW, EmpathyContactListViewPriv);
1239
1240         view->priv = priv;
1241         /* Get saved group states. */
1242         empathy_contact_groups_get_all ();
1243
1244         gtk_tree_view_set_row_separator_func (GTK_TREE_VIEW (view),
1245                                               empathy_contact_list_store_row_separator_func,
1246                                               NULL, NULL);
1247
1248         /* Connect to tree view signals rather than override. */
1249         g_signal_connect (view, "button-press-event",
1250                           G_CALLBACK (contact_list_view_button_press_event_cb),
1251                           NULL);
1252         g_signal_connect (view, "key-press-event",
1253                           G_CALLBACK (contact_list_view_key_press_event_cb),
1254                           NULL);
1255         g_signal_connect (view, "row-expanded",
1256                           G_CALLBACK (contact_list_view_row_expand_or_collapse_cb),
1257                           GINT_TO_POINTER (TRUE));
1258         g_signal_connect (view, "row-collapsed",
1259                           G_CALLBACK (contact_list_view_row_expand_or_collapse_cb),
1260                           GINT_TO_POINTER (FALSE));
1261         g_signal_connect (view, "query-tooltip",
1262                           G_CALLBACK (contact_list_view_query_tooltip_cb),
1263                           NULL);
1264 }
1265
1266 EmpathyContactListView *
1267 empathy_contact_list_view_new (EmpathyContactListStore        *store,
1268                                EmpathyContactListFeatureFlags  list_features,
1269                                EmpathyContactFeatureFlags      contact_features)
1270 {
1271         g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), NULL);
1272
1273         return g_object_new (EMPATHY_TYPE_CONTACT_LIST_VIEW,
1274                              "store", store,
1275                              "contact-features", contact_features,
1276                              "list-features", list_features,
1277                              NULL);
1278 }
1279
1280 EmpathyContact *
1281 empathy_contact_list_view_dup_selected (EmpathyContactListView *view)
1282 {
1283         EmpathyContactListViewPriv *priv;
1284         GtkTreeSelection          *selection;
1285         GtkTreeIter                iter;
1286         GtkTreeModel              *model;
1287         EmpathyContact             *contact;
1288
1289         g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_VIEW (view), NULL);
1290
1291         priv = GET_PRIV (view);
1292
1293         selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
1294         if (!gtk_tree_selection_get_selected (selection, &model, &iter)) {
1295                 return NULL;
1296         }
1297
1298         gtk_tree_model_get (model, &iter,
1299                             EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
1300                             -1);
1301
1302         return contact;
1303 }
1304
1305 gchar *
1306 empathy_contact_list_view_get_selected_group (EmpathyContactListView *view)
1307 {
1308         EmpathyContactListViewPriv *priv;
1309         GtkTreeSelection          *selection;
1310         GtkTreeIter                iter;
1311         GtkTreeModel              *model;
1312         gboolean                   is_group;
1313         gchar                     *name;
1314
1315         g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_VIEW (view), NULL);
1316
1317         priv = GET_PRIV (view);
1318
1319         selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
1320         if (!gtk_tree_selection_get_selected (selection, &model, &iter)) {
1321                 return NULL;
1322         }
1323
1324         gtk_tree_model_get (model, &iter,
1325                             EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
1326                             EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name,
1327                             -1);
1328
1329         if (!is_group) {
1330                 g_free (name);
1331                 return NULL;
1332         }
1333
1334         return name;
1335 }
1336
1337 static gboolean
1338 contact_list_view_remove_dialog_show (GtkWindow   *parent,
1339                                       const gchar *message,
1340                                       const gchar *secondary_text)
1341 {
1342         GtkWidget *dialog;
1343         gboolean res;
1344
1345         dialog = gtk_message_dialog_new (parent, GTK_DIALOG_MODAL,
1346                                          GTK_MESSAGE_QUESTION, GTK_BUTTONS_NONE,
1347                                          "%s", message);
1348         gtk_dialog_add_buttons (GTK_DIALOG (dialog),
1349                                 GTK_STOCK_CANCEL, GTK_RESPONSE_NO,
1350                                 GTK_STOCK_DELETE, GTK_RESPONSE_YES,
1351                                 NULL);
1352         gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog),
1353                                                   "%s", secondary_text);
1354
1355         gtk_widget_show (dialog);
1356
1357         res = gtk_dialog_run (GTK_DIALOG (dialog));
1358         gtk_widget_destroy (dialog);
1359
1360         return (res == GTK_RESPONSE_YES);
1361 }
1362
1363 static void
1364 contact_list_view_group_remove_activate_cb (GtkMenuItem            *menuitem,
1365                                             EmpathyContactListView *view)
1366 {
1367         EmpathyContactListViewPriv *priv = GET_PRIV (view);
1368         gchar                      *group;
1369
1370         group = empathy_contact_list_view_get_selected_group (view);
1371         if (group) {
1372                 gchar     *text;
1373                 GtkWindow *parent;
1374
1375                 text = g_strdup_printf (_("Do you really want to remove the group '%s'?"), group);
1376                 parent = empathy_get_toplevel_window (GTK_WIDGET (view));
1377                 if (contact_list_view_remove_dialog_show (parent, _("Removing group"), text)) {
1378                         EmpathyContactList *list;
1379
1380                         list = empathy_contact_list_store_get_list_iface (priv->store);
1381                         empathy_contact_list_remove_group (list, group);
1382                 }
1383
1384                 g_free (text);
1385         }
1386
1387         g_free (group);
1388 }
1389
1390 GtkWidget *
1391 empathy_contact_list_view_get_group_menu (EmpathyContactListView *view)
1392 {
1393         EmpathyContactListViewPriv *priv = GET_PRIV (view);
1394         gchar                      *group;
1395         GtkWidget                  *menu;
1396         GtkWidget                  *item;
1397         GtkWidget                  *image;
1398
1399         g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_VIEW (view), NULL);
1400
1401         if (!(priv->list_features & (EMPATHY_CONTACT_LIST_FEATURE_GROUPS_RENAME |
1402                                      EMPATHY_CONTACT_LIST_FEATURE_GROUPS_REMOVE))) {
1403                 return NULL;
1404         }
1405
1406         group = empathy_contact_list_view_get_selected_group (view);
1407         if (!group) {
1408                 return NULL;
1409         }
1410
1411         menu = gtk_menu_new ();
1412
1413         /* FIXME: Not implemented yet
1414         if (priv->features & EMPATHY_CONTACT_LIST_FEATURE_GROUPS_RENAME) {
1415                 item = gtk_menu_item_new_with_mnemonic (_("Re_name"));
1416                 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
1417                 gtk_widget_show (item);
1418                 g_signal_connect (item, "activate",
1419                                   G_CALLBACK (contact_list_view_group_rename_activate_cb),
1420                                   view);
1421         }*/
1422
1423         if (priv->list_features & EMPATHY_CONTACT_LIST_FEATURE_GROUPS_REMOVE) {
1424                 item = gtk_image_menu_item_new_with_mnemonic (_("_Remove"));
1425                 image = gtk_image_new_from_icon_name (GTK_STOCK_REMOVE,
1426                                                       GTK_ICON_SIZE_MENU);
1427                 gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
1428                 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
1429                 gtk_widget_show (item);
1430                 g_signal_connect (item, "activate",
1431                                   G_CALLBACK (contact_list_view_group_remove_activate_cb),
1432                                   view);
1433         }
1434
1435         g_free (group);
1436
1437         return menu;
1438 }
1439
1440 static void
1441 contact_list_view_remove_activate_cb (GtkMenuItem            *menuitem,
1442                                       EmpathyContactListView *view)
1443 {
1444         EmpathyContactListViewPriv *priv = GET_PRIV (view);
1445         EmpathyContact             *contact;
1446
1447         contact = empathy_contact_list_view_dup_selected (view);
1448
1449         if (contact) {
1450                 gchar     *text;
1451                 GtkWindow *parent;
1452
1453                 parent = empathy_get_toplevel_window (GTK_WIDGET (view));
1454                 text = g_strdup_printf (_("Do you really want to remove the contact '%s'?"),
1455                                         empathy_contact_get_name (contact));
1456                 if (contact_list_view_remove_dialog_show (parent, _("Removing contact"), text)) {
1457                         EmpathyContactList *list;
1458
1459                         list = empathy_contact_list_store_get_list_iface (priv->store);
1460                         empathy_contact_list_remove (list, contact, "");
1461                 }
1462
1463                 g_free (text);
1464                 g_object_unref (contact);
1465         }
1466 }
1467
1468 GtkWidget *
1469 empathy_contact_list_view_get_contact_menu (EmpathyContactListView *view)
1470 {
1471         EmpathyContactListViewPriv *priv = GET_PRIV (view);
1472         EmpathyContact             *contact;
1473         GtkWidget                  *menu;
1474         GtkWidget                  *item;
1475         GtkWidget                  *image;
1476
1477         g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_VIEW (view), NULL);
1478
1479         contact = empathy_contact_list_view_dup_selected (view);
1480         if (!contact) {
1481                 return NULL;
1482         }
1483
1484         menu = empathy_contact_menu_new (contact, priv->contact_features);
1485
1486         if (!(priv->list_features & EMPATHY_CONTACT_LIST_FEATURE_CONTACT_REMOVE)) {
1487                 g_object_unref (contact);
1488                 return menu;
1489         }
1490
1491         if (menu) {
1492                 /* Separator */
1493                 item = gtk_separator_menu_item_new ();
1494                 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
1495                 gtk_widget_show (item);
1496         } else {
1497                 menu = gtk_menu_new ();
1498         }
1499
1500         /* Remove contact */
1501         if (priv->list_features & EMPATHY_CONTACT_LIST_FEATURE_CONTACT_REMOVE) {
1502                 item = gtk_image_menu_item_new_with_mnemonic (_("_Remove"));
1503                 image = gtk_image_new_from_icon_name (GTK_STOCK_REMOVE,
1504                                                       GTK_ICON_SIZE_MENU);
1505                 gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
1506                 gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
1507                 gtk_widget_show (item);
1508                 g_signal_connect (item, "activate",
1509                                   G_CALLBACK (contact_list_view_remove_activate_cb),
1510                                   view);
1511         }
1512
1513         g_object_unref (contact);
1514
1515         return menu;
1516 }
1517