]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-individual-menu.c
Merge branch 'sms-651242'
[empathy.git] / libempathy-gtk / empathy-individual-menu.c
1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
2 /*
3  * Copyright (C) 2008-2010 Collabora Ltd.
4  *
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.
9  *
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.
14  *
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
18  *
19  * Authors: Xavier Claessens <xclaesse@gmail.com>
20  *          Travis Reitter <travis.reitter@collabora.co.uk>
21  */
22
23 #include "config.h"
24
25 #include <string.h>
26
27 #include <glib/gi18n-lib.h>
28 #include <gtk/gtk.h>
29 #include <telepathy-glib/util.h>
30
31 #include <folks/folks.h>
32 #include <folks/folks-telepathy.h>
33
34 #include <libempathy/empathy-request-util.h>
35 #include <libempathy/empathy-individual-manager.h>
36 #include <libempathy/empathy-chatroom-manager.h>
37 #include <libempathy/empathy-utils.h>
38
39 #include "empathy-individual-menu.h"
40 #include "empathy-images.h"
41 #include "empathy-log-window.h"
42 #include "empathy-contact-dialogs.h"
43 #include "empathy-gtk-enum-types.h"
44 #include "empathy-individual-edit-dialog.h"
45 #include "empathy-individual-information-dialog.h"
46 #include "empathy-ui-utils.h"
47 #include "empathy-share-my-desktop.h"
48 #include "empathy-linking-dialog.h"
49
50 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyIndividualMenu)
51
52 typedef struct {
53   FolksIndividual *individual; /* owned */
54   EmpathyIndividualFeatureFlags features;
55 } EmpathyIndividualMenuPriv;
56
57 enum {
58   PROP_INDIVIDUAL = 1,
59   PROP_FEATURES,
60 };
61
62 enum {
63   SIGNAL_LINK_CONTACTS_ACTIVATED,
64   LAST_SIGNAL
65 };
66
67 static guint signals[LAST_SIGNAL];
68
69 G_DEFINE_TYPE (EmpathyIndividualMenu, empathy_individual_menu, GTK_TYPE_MENU);
70
71 static void
72 individual_menu_add_personas (GtkMenuShell *menu,
73     FolksIndividual *individual,
74     EmpathyIndividualFeatureFlags features)
75 {
76   GtkWidget *item;
77   GeeSet *personas;
78   GeeIterator *iter;
79   guint persona_count = 0;
80   gboolean c;
81
82   g_return_if_fail (GTK_IS_MENU (menu));
83   g_return_if_fail (FOLKS_IS_INDIVIDUAL (individual));
84   g_return_if_fail (empathy_folks_individual_contains_contact (individual));
85
86   personas = folks_individual_get_personas (individual);
87   /* we'll re-use this iterator throughout */
88   iter = gee_iterable_iterator (GEE_ITERABLE (personas));
89
90   /* Make sure we've got enough valid entries for these menu items to add
91    * functionality */
92   while (gee_iterator_next (iter))
93     {
94       FolksPersona *persona = gee_iterator_get (iter);
95       if (empathy_folks_persona_is_interesting (persona))
96         persona_count++;
97
98       g_clear_object (&persona);
99     }
100
101   /* return early if these entries would add nothing beyond the "quick" items */
102   if (persona_count <= 1)
103     return;
104
105   /* add a separator before the list of personas */
106   item = gtk_separator_menu_item_new ();
107   gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
108   gtk_widget_show (item);
109
110   personas = folks_individual_get_personas (individual);
111   for (c = gee_iterator_first (iter); c; c = gee_iterator_next (iter))
112     {
113       GtkWidget *image;
114       GtkWidget *contact_item;
115       GtkWidget *contact_submenu;
116       TpContact *tp_contact;
117       EmpathyContact *contact;
118       TpfPersona *persona = gee_iterator_get (iter);
119       gchar *label;
120       FolksPersonaStore *store;
121       const gchar *account;
122       GtkWidget *action;
123
124       if (!empathy_folks_persona_is_interesting (FOLKS_PERSONA (persona)))
125         goto while_finish;
126
127       tp_contact = tpf_persona_get_contact (persona);
128       contact = empathy_contact_dup_from_tp_contact (tp_contact);
129
130       store = folks_persona_get_store (FOLKS_PERSONA (persona));
131       account = folks_persona_store_get_display_name (store);
132
133       /* Translators: this is used in the context menu for a contact. The first
134        * parameter is a contact ID (e.g. foo@jabber.org) and the second is one
135        * of the user's account IDs (e.g. me@hotmail.com). */
136       label = g_strdup_printf (_("%s (%s)"),
137           folks_persona_get_display_id (FOLKS_PERSONA (persona)), account);
138
139       contact_item = gtk_image_menu_item_new_with_label (label);
140       gtk_image_menu_item_set_always_show_image (GTK_IMAGE_MENU_ITEM (contact_item),
141                                                  TRUE);
142       contact_submenu = gtk_menu_new ();
143       gtk_menu_item_set_submenu (GTK_MENU_ITEM (contact_item), contact_submenu);
144       image = gtk_image_new_from_icon_name (
145           empathy_icon_name_for_contact (contact), GTK_ICON_SIZE_MENU);
146       gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (contact_item), image);
147       gtk_widget_show (image);
148
149       /* Chat */
150       if (features & EMPATHY_INDIVIDUAL_FEATURE_CHAT)
151         {
152           action = empathy_individual_chat_menu_item_new (NULL, contact);
153           gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
154           gtk_widget_show (action);
155         }
156
157       /* SMS */
158       if (features & EMPATHY_INDIVIDUAL_FEATURE_SMS)
159         {
160           action = empathy_individual_sms_menu_item_new (NULL, contact);
161           gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
162           gtk_widget_show (action);
163         }
164
165       if (features & EMPATHY_INDIVIDUAL_FEATURE_CALL)
166         {
167           /* Audio Call */
168           action = empathy_individual_audio_call_menu_item_new (NULL, contact);
169           gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
170           gtk_widget_show (action);
171
172           /* Video Call */
173           action = empathy_individual_video_call_menu_item_new (NULL, contact);
174           gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
175           gtk_widget_show (action);
176         }
177
178       /* Log */
179       if (features & EMPATHY_INDIVIDUAL_FEATURE_LOG)
180         {
181           action = empathy_individual_log_menu_item_new (NULL, contact);
182           gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
183           gtk_widget_show (action);
184         }
185
186       /* Invite */
187       action = empathy_individual_invite_menu_item_new (NULL, contact);
188       gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
189       gtk_widget_show (action);
190
191       /* File transfer */
192       action = empathy_individual_file_transfer_menu_item_new (NULL, contact);
193       gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
194       gtk_widget_show (action);
195
196       /* Share my desktop */
197       action = empathy_individual_share_my_desktop_menu_item_new (NULL,
198           contact);
199       gtk_menu_shell_append (GTK_MENU_SHELL (contact_submenu), action);
200       gtk_widget_show (action);
201
202       gtk_menu_shell_append (GTK_MENU_SHELL (menu), contact_item);
203       gtk_widget_show (contact_item);
204
205       g_free (label);
206       g_object_unref (contact);
207
208 while_finish:
209       g_clear_object (&persona);
210     }
211
212   g_clear_object (&iter);
213 }
214
215 static void
216 individual_link_menu_item_activate_cb (EmpathyIndividualMenu *self)
217 {
218   EmpathyIndividualMenuPriv *priv = GET_PRIV (self);
219   GtkWidget *dialog;
220
221   dialog = empathy_linking_dialog_show (priv->individual, NULL);
222   g_signal_emit (self, signals[SIGNAL_LINK_CONTACTS_ACTIVATED], 0, dialog);
223 }
224
225 static void
226 empathy_individual_menu_init (EmpathyIndividualMenu *self)
227 {
228   EmpathyIndividualMenuPriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (self,
229       EMPATHY_TYPE_INDIVIDUAL_MENU, EmpathyIndividualMenuPriv);
230
231   self->priv = priv;
232 }
233
234 static void
235 constructed (GObject *object)
236 {
237   EmpathyIndividualMenuPriv *priv = GET_PRIV (object);
238   GtkMenuShell *shell;
239   GtkWidget *item;
240   FolksIndividual *individual;
241   EmpathyIndividualFeatureFlags features;
242
243   /* Build the menu */
244   shell = GTK_MENU_SHELL (object);
245   individual = priv->individual;
246   features = priv->features;
247
248   /* Chat */
249   if (features & EMPATHY_INDIVIDUAL_FEATURE_CHAT)
250     {
251       item = empathy_individual_chat_menu_item_new (individual, NULL);
252       if (item != NULL)
253         {
254           gtk_menu_shell_append (shell, item);
255           gtk_widget_show (item);
256         }
257     }
258
259   /* SMS */
260   if (features & EMPATHY_INDIVIDUAL_FEATURE_SMS)
261     {
262       item = empathy_individual_sms_menu_item_new (individual, NULL);
263       if (item != NULL)
264         {
265           gtk_menu_shell_append (shell, item);
266           gtk_widget_show (item);
267         }
268     }
269
270   if (features & EMPATHY_INDIVIDUAL_FEATURE_CALL)
271     {
272       /* Audio Call */
273       item = empathy_individual_audio_call_menu_item_new (individual, NULL);
274       gtk_menu_shell_append (shell, item);
275       gtk_widget_show (item);
276
277       /* Video Call */
278       item = empathy_individual_video_call_menu_item_new (individual, NULL);
279       gtk_menu_shell_append (shell, item);
280       gtk_widget_show (item);
281     }
282
283   /* Log */
284   if (features & EMPATHY_INDIVIDUAL_FEATURE_LOG)
285     {
286       item = empathy_individual_log_menu_item_new (individual, NULL);
287       gtk_menu_shell_append (shell, item);
288       gtk_widget_show (item);
289     }
290
291   /* Invite */
292   item = empathy_individual_invite_menu_item_new (individual, NULL);
293   gtk_menu_shell_append (shell, item);
294   gtk_widget_show (item);
295
296   /* File transfer */
297   item = empathy_individual_file_transfer_menu_item_new (individual, NULL);
298   gtk_menu_shell_append (shell, item);
299   gtk_widget_show (item);
300
301   /* Share my desktop */
302   /* FIXME we should add the "Share my desktop" menu item if Vino is
303   a registered handler in MC5 */
304   item = empathy_individual_share_my_desktop_menu_item_new (individual, NULL);
305   gtk_menu_shell_append (shell, item);
306   gtk_widget_show (item);
307
308   /* Menu items to target specific contacts */
309   individual_menu_add_personas (GTK_MENU_SHELL (object), individual, features);
310
311   /* Separator */
312   if (features & (EMPATHY_INDIVIDUAL_FEATURE_EDIT |
313       EMPATHY_INDIVIDUAL_FEATURE_INFO |
314       EMPATHY_INDIVIDUAL_FEATURE_FAVOURITE |
315       EMPATHY_INDIVIDUAL_FEATURE_LINK))
316     {
317       item = gtk_separator_menu_item_new ();
318       gtk_menu_shell_append (shell, item);
319       gtk_widget_show (item);
320     }
321
322   /* Edit */
323   if (features & EMPATHY_INDIVIDUAL_FEATURE_EDIT)
324     {
325       item = empathy_individual_edit_menu_item_new (individual);
326       gtk_menu_shell_append (shell, item);
327       gtk_widget_show (item);
328     }
329
330   /* Link */
331   if (features & EMPATHY_INDIVIDUAL_FEATURE_LINK)
332     {
333       item = empathy_individual_link_menu_item_new (individual);
334       gtk_menu_shell_append (shell, item);
335
336       g_signal_connect_swapped (item, "activate",
337           (GCallback) individual_link_menu_item_activate_cb, object);
338
339       gtk_widget_show (item);
340     }
341
342   /* Info */
343   if (features & EMPATHY_INDIVIDUAL_FEATURE_INFO)
344     {
345       item = empathy_individual_info_menu_item_new (individual);
346       gtk_menu_shell_append (shell, item);
347       gtk_widget_show (item);
348     }
349
350   /* Favorite checkbox */
351   if (features & EMPATHY_INDIVIDUAL_FEATURE_FAVOURITE)
352     {
353       item = empathy_individual_favourite_menu_item_new (individual);
354       gtk_menu_shell_append (shell, item);
355       gtk_widget_show (item);
356     }
357 }
358
359 static void
360 get_property (GObject *object,
361     guint param_id,
362     GValue *value,
363     GParamSpec *pspec)
364 {
365   EmpathyIndividualMenuPriv *priv;
366
367   priv = GET_PRIV (object);
368
369   switch (param_id)
370     {
371       case PROP_INDIVIDUAL:
372         g_value_set_object (value, priv->individual);
373         break;
374       case PROP_FEATURES:
375         g_value_set_flags (value, priv->features);
376         break;
377       default:
378         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
379         break;
380     }
381 }
382
383 static void
384 set_property (GObject *object,
385     guint param_id,
386     const GValue *value,
387     GParamSpec *pspec)
388 {
389   EmpathyIndividualMenuPriv *priv;
390
391   priv = GET_PRIV (object);
392
393   switch (param_id)
394     {
395       case PROP_INDIVIDUAL:
396         priv->individual = g_value_dup_object (value);
397         break;
398       case PROP_FEATURES:
399         priv->features = g_value_get_flags (value);
400         break;
401       default:
402         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
403         break;
404     }
405 }
406
407 static void
408 dispose (GObject *object)
409 {
410   EmpathyIndividualMenuPriv *priv = GET_PRIV (object);
411
412   tp_clear_object (&priv->individual);
413
414   G_OBJECT_CLASS (empathy_individual_menu_parent_class)->dispose (object);
415 }
416
417 static void
418 empathy_individual_menu_class_init (EmpathyIndividualMenuClass *klass)
419 {
420   GObjectClass *object_class = G_OBJECT_CLASS (klass);
421
422   object_class->constructed = constructed;
423   object_class->get_property = get_property;
424   object_class->set_property = set_property;
425   object_class->dispose = dispose;
426
427   /**
428    * EmpathyIndividualMenu:individual:
429    *
430    * The #FolksIndividual the menu is for.
431    */
432   g_object_class_install_property (object_class, PROP_INDIVIDUAL,
433       g_param_spec_object ("individual",
434           "Individual",
435           "The #FolksIndividual the menu is for.",
436           FOLKS_TYPE_INDIVIDUAL,
437           G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
438
439   /**
440    * EmpathyIndividualMenu:features:
441    *
442    * A set of feature flags controlling which entries are shown.
443    */
444   g_object_class_install_property (object_class, PROP_FEATURES,
445       g_param_spec_flags ("features",
446           "Features",
447           "A set of feature flags controlling which entries are shown.",
448           EMPATHY_TYPE_INDIVIDUAL_FEATURE_FLAGS,
449           EMPATHY_INDIVIDUAL_FEATURE_NONE,
450           G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
451
452   signals[SIGNAL_LINK_CONTACTS_ACTIVATED] =
453       g_signal_new ("link-contacts-activated", G_OBJECT_CLASS_TYPE (klass),
454           G_SIGNAL_RUN_LAST, 0, NULL, NULL,
455           g_cclosure_marshal_VOID__OBJECT,
456           G_TYPE_NONE, 1, EMPATHY_TYPE_LINKING_DIALOG);
457
458   g_type_class_add_private (object_class, sizeof (EmpathyIndividualMenuPriv));
459 }
460
461 GtkWidget *
462 empathy_individual_menu_new (FolksIndividual *individual,
463     EmpathyIndividualFeatureFlags features)
464 {
465   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
466   g_return_val_if_fail (features != EMPATHY_INDIVIDUAL_FEATURE_NONE, NULL);
467
468   return g_object_new (EMPATHY_TYPE_INDIVIDUAL_MENU,
469       "individual", individual,
470       "features", features,
471       NULL);
472 }
473
474 /* Like menu_item_set_first_contact(), but always operates upon the given
475  * contact. If the contact is non-NULL, it is assumed that the menu entry should
476  * be sensitive. */
477 static gboolean
478 menu_item_set_contact (GtkWidget *item,
479     EmpathyContact *contact,
480     GCallback activate_callback,
481     EmpathyActionType action_type)
482 {
483   gboolean can_do_action = FALSE;
484
485   if (contact != NULL)
486     can_do_action = empathy_contact_can_do_action (contact, action_type);
487   gtk_widget_set_sensitive (item, can_do_action);
488
489   if (can_do_action == TRUE)
490     {
491       /* We want to make sure that the EmpathyContact stays alive while the
492        * signal is connected. */
493       g_signal_connect_data (item, "activate", G_CALLBACK (activate_callback),
494           g_object_ref (contact), (GClosureNotify) g_object_unref, 0);
495     }
496
497   return can_do_action;
498 }
499
500 /**
501  * Set the given menu @item to call @activate_callback using the TpContact
502  * (associated with @individual) with the highest availability who is also valid
503  * whenever @item is activated.
504  *
505  * @action_type is the type of action performed by the menu entry; this is used
506  * so that only contacts which can perform that action (e.g. are capable of
507  * receiving video calls) are selected, as appropriate.
508  */
509 static GtkWidget *
510 menu_item_set_first_contact (GtkWidget *item,
511     FolksIndividual *individual,
512     GCallback activate_callback,
513     EmpathyActionType action_type)
514 {
515   EmpathyContact *best_contact;
516
517   best_contact = empathy_contact_dup_best_for_action (individual, action_type);
518   menu_item_set_contact (item, best_contact, G_CALLBACK (activate_callback),
519       action_type);
520   tp_clear_object (&best_contact);
521
522   return item;
523 }
524
525 static void
526 empathy_individual_chat_menu_item_activated (GtkMenuItem *item,
527   EmpathyContact *contact)
528 {
529   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
530
531   empathy_chat_with_contact (contact, empathy_get_current_action_time ());
532 }
533
534 GtkWidget *
535 empathy_individual_chat_menu_item_new (FolksIndividual *individual,
536     EmpathyContact *contact)
537 {
538   GtkWidget *item;
539   GtkWidget *image;
540
541   g_return_val_if_fail ((FOLKS_IS_INDIVIDUAL (individual) &&
542       empathy_folks_individual_contains_contact (individual)) ||
543       EMPATHY_IS_CONTACT (contact),
544       NULL);
545
546   item = gtk_image_menu_item_new_with_mnemonic (_("_Chat"));
547   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_MESSAGE,
548       GTK_ICON_SIZE_MENU);
549   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
550   gtk_widget_show (image);
551
552   if (contact != NULL)
553     {
554       menu_item_set_contact (item, contact,
555           G_CALLBACK (empathy_individual_chat_menu_item_activated),
556           EMPATHY_ACTION_CHAT);
557     }
558   else
559     {
560       menu_item_set_first_contact (item, individual,
561           G_CALLBACK (empathy_individual_chat_menu_item_activated),
562           EMPATHY_ACTION_CHAT);
563     }
564
565   return item;
566 }
567
568 static void
569 empathy_individual_sms_menu_item_activated (GtkMenuItem *item,
570   EmpathyContact *contact)
571 {
572   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
573
574   empathy_sms_contact_id (
575       empathy_contact_get_account (contact),
576       empathy_contact_get_id (contact),
577       empathy_get_current_action_time ());
578 }
579
580 GtkWidget *
581 empathy_individual_sms_menu_item_new (FolksIndividual *individual,
582     EmpathyContact *contact)
583 {
584   GtkWidget *item;
585   GtkWidget *image;
586
587   g_return_val_if_fail ((FOLKS_IS_INDIVIDUAL (individual) &&
588       empathy_folks_individual_contains_contact (individual)) ||
589       EMPATHY_IS_CONTACT (contact),
590       NULL);
591
592   item = gtk_image_menu_item_new_with_mnemonic (_("_SMS"));
593   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_SMS,
594       GTK_ICON_SIZE_MENU);
595   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
596   gtk_widget_show (image);
597
598   if (contact != NULL)
599     {
600       menu_item_set_contact (item, contact,
601           G_CALLBACK (empathy_individual_sms_menu_item_activated),
602           EMPATHY_ACTION_SMS);
603     }
604   else
605     {
606       menu_item_set_first_contact (item, individual,
607           G_CALLBACK (empathy_individual_sms_menu_item_activated),
608           EMPATHY_ACTION_SMS);
609     }
610
611   return item;
612 }
613
614 static void
615 empathy_individual_audio_call_menu_item_activated (GtkMenuItem *item,
616   EmpathyContact *contact)
617 {
618   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
619
620   empathy_call_new_with_streams (contact, TRUE, FALSE,
621       empathy_get_current_action_time ());
622 }
623
624 GtkWidget *
625 empathy_individual_audio_call_menu_item_new (FolksIndividual *individual,
626     EmpathyContact *contact)
627 {
628   GtkWidget *item;
629   GtkWidget *image;
630
631   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
632       EMPATHY_IS_CONTACT (contact),
633       NULL);
634
635   item = gtk_image_menu_item_new_with_mnemonic (C_("menu item", "_Audio Call"));
636   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_VOIP, GTK_ICON_SIZE_MENU);
637   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
638   gtk_widget_show (image);
639
640   if (contact != NULL)
641     {
642       menu_item_set_contact (item, contact,
643           G_CALLBACK (empathy_individual_audio_call_menu_item_activated),
644           EMPATHY_ACTION_AUDIO_CALL);
645     }
646   else
647     {
648       menu_item_set_first_contact (item, individual,
649           G_CALLBACK (empathy_individual_audio_call_menu_item_activated),
650           EMPATHY_ACTION_AUDIO_CALL);
651     }
652
653   return item;
654 }
655
656 static void
657 empathy_individual_video_call_menu_item_activated (GtkMenuItem *item,
658   EmpathyContact *contact)
659 {
660   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
661
662   empathy_call_new_with_streams (contact, TRUE, TRUE,
663       empathy_get_current_action_time ());
664 }
665
666 GtkWidget *
667 empathy_individual_video_call_menu_item_new (FolksIndividual *individual,
668     EmpathyContact *contact)
669 {
670   GtkWidget *item;
671   GtkWidget *image;
672
673   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
674       EMPATHY_IS_CONTACT (contact),
675       NULL);
676
677   item = gtk_image_menu_item_new_with_mnemonic (C_("menu item", "_Video Call"));
678   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_VIDEO_CALL,
679       GTK_ICON_SIZE_MENU);
680   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
681   gtk_widget_show (image);
682
683   if (contact != NULL)
684     {
685       menu_item_set_contact (item, contact,
686           G_CALLBACK (empathy_individual_video_call_menu_item_activated),
687           EMPATHY_ACTION_VIDEO_CALL);
688     }
689   else
690     {
691       menu_item_set_first_contact (item, individual,
692           G_CALLBACK (empathy_individual_video_call_menu_item_activated),
693           EMPATHY_ACTION_VIDEO_CALL);
694     }
695
696   return item;
697 }
698
699 static void
700 empathy_individual_log_menu_item_activated (GtkMenuItem *item,
701   EmpathyContact *contact)
702 {
703   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
704
705   empathy_log_window_show (empathy_contact_get_account (contact),
706       empathy_contact_get_id (contact), FALSE, NULL);
707 }
708
709 GtkWidget *
710 empathy_individual_log_menu_item_new (FolksIndividual *individual,
711     EmpathyContact *contact)
712 {
713   GtkWidget *item;
714   GtkWidget *image;
715
716   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
717       EMPATHY_IS_CONTACT (contact),
718       NULL);
719
720   item = gtk_image_menu_item_new_with_mnemonic (_("_Previous Conversations"));
721   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_LOG, GTK_ICON_SIZE_MENU);
722   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
723   gtk_widget_show (image);
724
725   if (contact != NULL)
726     {
727       menu_item_set_contact (item, contact,
728           G_CALLBACK (empathy_individual_log_menu_item_activated),
729           EMPATHY_ACTION_VIEW_LOGS);
730     }
731   else
732     {
733       menu_item_set_first_contact (item, individual,
734           G_CALLBACK (empathy_individual_log_menu_item_activated),
735           EMPATHY_ACTION_VIEW_LOGS);
736     }
737
738   return item;
739 }
740
741 static void
742 empathy_individual_file_transfer_menu_item_activated (GtkMenuItem *item,
743     EmpathyContact *contact)
744 {
745   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
746
747   empathy_send_file_with_file_chooser (contact);
748 }
749
750 GtkWidget *
751 empathy_individual_file_transfer_menu_item_new (FolksIndividual *individual,
752     EmpathyContact *contact)
753 {
754   GtkWidget *item;
755   GtkWidget *image;
756
757   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
758       EMPATHY_IS_CONTACT (contact),
759       NULL);
760
761   item = gtk_image_menu_item_new_with_mnemonic (_("Send File"));
762   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_DOCUMENT_SEND,
763       GTK_ICON_SIZE_MENU);
764   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
765   gtk_widget_show (image);
766
767   if (contact != NULL)
768     {
769       menu_item_set_contact (item, contact,
770           G_CALLBACK (empathy_individual_file_transfer_menu_item_activated),
771           EMPATHY_ACTION_SEND_FILE);
772     }
773   else
774     {
775       menu_item_set_first_contact (item, individual,
776           G_CALLBACK (empathy_individual_file_transfer_menu_item_activated),
777           EMPATHY_ACTION_SEND_FILE);
778     }
779
780   return item;
781 }
782
783 static void
784 empathy_individual_share_my_desktop_menu_item_activated (GtkMenuItem *item,
785     EmpathyContact *contact)
786 {
787   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
788
789   empathy_share_my_desktop_share_with_contact (contact);
790 }
791
792 GtkWidget *
793 empathy_individual_share_my_desktop_menu_item_new (FolksIndividual *individual,
794     EmpathyContact *contact)
795 {
796   GtkWidget *item;
797   GtkWidget *image;
798
799   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
800       EMPATHY_IS_CONTACT (contact),
801       NULL);
802
803   item = gtk_image_menu_item_new_with_mnemonic (_("Share My Desktop"));
804   image = gtk_image_new_from_icon_name (GTK_STOCK_NETWORK, GTK_ICON_SIZE_MENU);
805   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
806   gtk_widget_show (image);
807
808   if (contact != NULL)
809     {
810       menu_item_set_contact (item, contact,
811           G_CALLBACK (empathy_individual_share_my_desktop_menu_item_activated),
812           EMPATHY_ACTION_SHARE_MY_DESKTOP);
813     }
814   else
815     {
816       menu_item_set_first_contact (item, individual,
817           G_CALLBACK (empathy_individual_share_my_desktop_menu_item_activated),
818           EMPATHY_ACTION_SHARE_MY_DESKTOP);
819     }
820
821   return item;
822 }
823
824 static void
825 favourite_menu_item_toggled_cb (GtkCheckMenuItem *item,
826   FolksIndividual *individual)
827 {
828   folks_favourite_details_set_is_favourite (
829       FOLKS_FAVOURITE_DETAILS (individual),
830       gtk_check_menu_item_get_active (item));
831 }
832
833 GtkWidget *
834 empathy_individual_favourite_menu_item_new (FolksIndividual *individual)
835 {
836   GtkWidget *item;
837
838   item = gtk_check_menu_item_new_with_label (_("Favorite"));
839
840   gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (item),
841       folks_favourite_details_get_is_favourite (
842           FOLKS_FAVOURITE_DETAILS (individual)));
843
844   g_signal_connect (item, "toggled",
845       G_CALLBACK (favourite_menu_item_toggled_cb), individual);
846
847   return item;
848 }
849
850 static void
851 individual_info_menu_item_activate_cb (FolksIndividual *individual)
852 {
853   empathy_individual_information_dialog_show (individual, NULL);
854 }
855
856 GtkWidget *
857 empathy_individual_info_menu_item_new (FolksIndividual *individual)
858 {
859   GtkWidget *item;
860   GtkWidget *image;
861
862   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
863   g_return_val_if_fail (empathy_folks_individual_contains_contact (individual),
864       NULL);
865
866   item = gtk_image_menu_item_new_with_mnemonic (_("Infor_mation"));
867   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_CONTACT_INFORMATION,
868                 GTK_ICON_SIZE_MENU);
869   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
870   gtk_widget_show (image);
871
872   g_signal_connect_swapped (item, "activate",
873           G_CALLBACK (individual_info_menu_item_activate_cb),
874           individual);
875
876   return item;
877 }
878
879 static void
880 individual_edit_menu_item_activate_cb (FolksIndividual *individual)
881 {
882   empathy_individual_edit_dialog_show (individual, NULL);
883 }
884
885 GtkWidget *
886 empathy_individual_edit_menu_item_new (FolksIndividual *individual)
887 {
888   EmpathyIndividualManager *manager;
889   GtkWidget *item;
890   GtkWidget *image;
891   gboolean enable = FALSE;
892   EmpathyContact *contact;
893
894   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
895
896   contact = empathy_contact_dup_from_folks_individual (individual);
897
898   g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);
899
900   if (empathy_individual_manager_initialized ())
901     {
902       TpConnection *connection;
903
904       manager = empathy_individual_manager_dup_singleton ();
905       connection = empathy_contact_get_connection (contact);
906
907       enable = (empathy_connection_can_alias_personas (connection) &&
908                 empathy_connection_can_group_personas (connection));
909
910       g_object_unref (manager);
911     }
912
913   item = gtk_image_menu_item_new_with_mnemonic (
914       C_("Edit individual (contextual menu)", "_Edit"));
915   image = gtk_image_new_from_icon_name (GTK_STOCK_EDIT, GTK_ICON_SIZE_MENU);
916   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
917   gtk_widget_show (image);
918
919   gtk_widget_set_sensitive (item, enable);
920
921   g_signal_connect_swapped (item, "activate",
922       G_CALLBACK (individual_edit_menu_item_activate_cb), individual);
923
924   g_object_unref (contact);
925
926   return item;
927 }
928
929 GtkWidget *
930 empathy_individual_link_menu_item_new (FolksIndividual *individual)
931 {
932   GtkWidget *item;
933   /*GtkWidget *image;*/
934
935   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
936
937   item = gtk_image_menu_item_new_with_mnemonic (
938       /* Translators: this is a verb meaning "to connect two contacts together
939        * to form a meta-contact". */
940       C_("Link individual (contextual menu)", "_Link Contacts…"));
941   /* TODO */
942   /*image = gtk_image_new_from_icon_name (GTK_STOCK_EDIT, GTK_ICON_SIZE_MENU);
943   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
944   gtk_widget_show (image);*/
945
946   /* Only allow trusted Individuals to be linked */
947   gtk_widget_set_sensitive (item,
948       folks_individual_get_trust_level (individual) ==
949           FOLKS_TRUST_LEVEL_PERSONAS);
950
951   return item;
952 }
953
954 typedef struct
955 {
956   FolksIndividual *individual;
957   EmpathyContact *contact;
958   EmpathyChatroom *chatroom;
959 } RoomSubMenuData;
960
961 static RoomSubMenuData *
962 room_sub_menu_data_new (FolksIndividual *individual,
963     EmpathyContact *contact,
964     EmpathyChatroom *chatroom)
965 {
966   RoomSubMenuData *data;
967
968   data = g_slice_new0 (RoomSubMenuData);
969   if (individual != NULL)
970     data->individual = g_object_ref (individual);
971   if (contact != NULL)
972     data->contact = g_object_ref (contact);
973   data->chatroom = g_object_ref (chatroom);
974
975   return data;
976 }
977
978 static void
979 room_sub_menu_data_free (RoomSubMenuData *data)
980 {
981   tp_clear_object (&data->individual);
982   tp_clear_object (&data->contact);
983   g_object_unref (data->chatroom);
984   g_slice_free (RoomSubMenuData, data);
985 }
986
987 static void
988 room_sub_menu_activate_cb (GtkWidget *item,
989          RoomSubMenuData *data)
990 {
991   EmpathyTpChat *chat;
992   EmpathyChatroomManager *mgr;
993   EmpathyContact *contact = NULL;
994
995   chat = empathy_chatroom_get_tp_chat (data->chatroom);
996   if (chat == NULL)
997     {
998       /* channel was invalidated. Ignoring */
999       return;
1000     }
1001
1002   mgr = empathy_chatroom_manager_dup_singleton (NULL);
1003
1004   if (data->contact != NULL)
1005     contact = g_object_ref (data->contact);
1006   else
1007     {
1008       GeeSet *personas;
1009       GeeIterator *iter;
1010
1011       /* find the first of this Individual's contacts who can join this room */
1012       personas = folks_individual_get_personas (data->individual);
1013
1014       iter = gee_iterable_iterator (GEE_ITERABLE (personas));
1015       while (gee_iterator_next (iter) && (contact == NULL))
1016         {
1017           TpfPersona *persona = gee_iterator_get (iter);
1018           TpContact *tp_contact;
1019           GList *rooms;
1020
1021           if (empathy_folks_persona_is_interesting (FOLKS_PERSONA (persona)))
1022             {
1023               tp_contact = tpf_persona_get_contact (persona);
1024               contact = empathy_contact_dup_from_tp_contact (tp_contact);
1025
1026               rooms = empathy_chatroom_manager_get_chatrooms (mgr,
1027                   empathy_contact_get_account (contact));
1028
1029               if (g_list_find (rooms, data->chatroom) == NULL)
1030                 g_clear_object (&contact);
1031
1032               /* if contact != NULL here, we've found our match */
1033
1034               g_list_free (rooms);
1035             }
1036           g_clear_object (&persona);
1037         }
1038       g_clear_object (&iter);
1039     }
1040
1041   g_object_unref (mgr);
1042
1043   if (contact == NULL)
1044     {
1045       /* contact disappeared. Ignoring */
1046       goto out;
1047     }
1048
1049   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
1050
1051   /* send invitation */
1052   empathy_contact_list_add (EMPATHY_CONTACT_LIST (chat),
1053       contact, _("Inviting you to this room"));
1054
1055 out:
1056   g_object_unref (contact);
1057 }
1058
1059 static GtkWidget *
1060 create_room_sub_menu (FolksIndividual *individual,
1061                       EmpathyContact *contact,
1062                       EmpathyChatroom *chatroom)
1063 {
1064   GtkWidget *item;
1065   RoomSubMenuData *data;
1066
1067   item = gtk_menu_item_new_with_label (empathy_chatroom_get_name (chatroom));
1068   data = room_sub_menu_data_new (individual, contact, chatroom);
1069   g_signal_connect_data (item, "activate",
1070       G_CALLBACK (room_sub_menu_activate_cb), data,
1071       (GClosureNotify) room_sub_menu_data_free, 0);
1072
1073   return item;
1074 }
1075
1076 GtkWidget *
1077 empathy_individual_invite_menu_item_new (FolksIndividual *individual,
1078     EmpathyContact *contact)
1079 {
1080   GtkWidget *item;
1081   GtkWidget *image;
1082   GtkWidget *room_item;
1083   EmpathyChatroomManager *mgr;
1084   GList *rooms = NULL;
1085   GList *names = NULL;
1086   GList *l;
1087   GtkWidget *submenu = NULL;
1088   /* map of chat room names to their objects; just a utility to remove
1089    * duplicates and to make construction of the alphabetized list easier */
1090   GHashTable *name_room_map;
1091
1092   g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual) ||
1093       EMPATHY_IS_CONTACT (contact),
1094       NULL);
1095
1096   name_room_map = g_hash_table_new_full (g_str_hash, g_str_equal, NULL,
1097       g_object_unref);
1098
1099   item = gtk_image_menu_item_new_with_mnemonic (_("_Invite to Chat Room"));
1100   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_GROUP_MESSAGE,
1101       GTK_ICON_SIZE_MENU);
1102   gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (item), image);
1103
1104   mgr = empathy_chatroom_manager_dup_singleton (NULL);
1105
1106   if (contact != NULL)
1107     {
1108       rooms = empathy_chatroom_manager_get_chatrooms (mgr,
1109           empathy_contact_get_account (contact));
1110     }
1111   else
1112     {
1113       GeeSet *personas;
1114       GeeIterator *iter;
1115
1116       /* find the first of this Individual's contacts who can join this room */
1117       personas = folks_individual_get_personas (individual);
1118       iter = gee_iterable_iterator (GEE_ITERABLE (personas));
1119       while (gee_iterator_next (iter))
1120         {
1121           TpfPersona *persona = gee_iterator_get (iter);
1122           GList *rooms_cur;
1123           TpContact *tp_contact;
1124           EmpathyContact *contact_cur;
1125
1126           if (empathy_folks_persona_is_interesting (FOLKS_PERSONA (persona)))
1127             {
1128               tp_contact = tpf_persona_get_contact (persona);
1129               contact_cur = empathy_contact_dup_from_tp_contact (tp_contact);
1130
1131               rooms_cur = empathy_chatroom_manager_get_chatrooms (mgr,
1132                   empathy_contact_get_account (contact_cur));
1133               rooms = g_list_concat (rooms, rooms_cur);
1134
1135               g_object_unref (contact_cur);
1136             }
1137           g_clear_object (&persona);
1138         }
1139       g_clear_object (&iter);
1140     }
1141
1142   /* alphabetize the rooms */
1143   for (l = rooms; l != NULL; l = g_list_next (l))
1144     {
1145       EmpathyChatroom *chatroom = l->data;
1146       gboolean existed;
1147
1148       if (empathy_chatroom_get_tp_chat (chatroom) != NULL)
1149         {
1150           const gchar *name;
1151
1152           name = empathy_chatroom_get_name (chatroom);
1153           existed = (g_hash_table_lookup (name_room_map, name) != NULL);
1154           g_hash_table_insert (name_room_map, (gpointer) name,
1155               g_object_ref (chatroom));
1156
1157           /* this will take care of duplicates in rooms */
1158           if (!existed)
1159             {
1160               names = g_list_insert_sorted (names, (gpointer) name,
1161                   (GCompareFunc) g_strcmp0);
1162             }
1163         }
1164     }
1165
1166   for (l = names; l != NULL; l = g_list_next (l))
1167     {
1168       const gchar *name = l->data;
1169       EmpathyChatroom *chatroom;
1170
1171       if (G_UNLIKELY (submenu == NULL))
1172         submenu = gtk_menu_new ();
1173
1174       chatroom = g_hash_table_lookup (name_room_map, name);
1175       room_item = create_room_sub_menu (individual, contact, chatroom);
1176       gtk_menu_shell_append ((GtkMenuShell *) submenu, room_item);
1177       gtk_widget_show (room_item);
1178     }
1179
1180   if (submenu)
1181     gtk_menu_item_set_submenu (GTK_MENU_ITEM (item), submenu);
1182   else
1183     gtk_widget_set_sensitive (item, FALSE);
1184
1185   gtk_widget_show (image);
1186
1187   g_hash_table_destroy (name_room_map);
1188   g_object_unref (mgr);
1189   g_list_free (names);
1190   g_list_free (rooms);
1191
1192   return item;
1193 }