]> git.0d.be Git - empathy.git/blob - libempathy-gtk/empathy-contact-widget.c
Change EmpathyTpContactFactory API to look more like TpConnection. Add function to...
[empathy.git] / libempathy-gtk / empathy-contact-widget.c
1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
2 /*
3  * Copyright (C) 2007-2009 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  */
21
22 #include <config.h>
23
24 #include <string.h>
25 #include <stdlib.h>
26
27 #include <gtk/gtk.h>
28 #include <glib/gi18n-lib.h>
29
30 #include <libmissioncontrol/mc-account.h>
31 #include <telepathy-glib/util.h>
32
33 #include <libempathy/empathy-tp-contact-factory.h>
34 #include <libempathy/empathy-contact-manager.h>
35 #include <libempathy/empathy-contact-list.h>
36 #include <libempathy/empathy-utils.h>
37
38 #include "empathy-contact-widget.h"
39 #include "empathy-account-chooser.h"
40 #include "empathy-avatar-chooser.h"
41 #include "empathy-avatar-image.h"
42 #include "empathy-ui-utils.h"
43
44 #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT
45 #include <libempathy/empathy-debug.h>
46
47 /* Delay before updating the widget when the id entry changed (seconds) */
48 #define ID_CHANGED_TIMEOUT 1
49
50 typedef struct
51 {
52   EmpathyTpContactFactory *factory;
53   EmpathyContactManager *manager;
54   EmpathyContact *contact;
55   EmpathyContactWidgetFlags flags;
56   guint widget_id_timeout;
57
58   GtkWidget *vbox_contact_widget;
59
60   /* Contact */
61   GtkWidget *vbox_contact;
62   GtkWidget *widget_avatar;
63   GtkWidget *widget_account;
64   GtkWidget *widget_id;
65   GtkWidget *widget_alias;
66   GtkWidget *label_alias;
67   GtkWidget *entry_alias;
68   GtkWidget *hbox_presence;
69   GtkWidget *image_state;
70   GtkWidget *label_status;
71   GtkWidget *table_contact;
72   GtkWidget *vbox_avatar;
73
74   /* Groups */
75   GtkWidget *vbox_groups;
76   GtkWidget *entry_group;
77   GtkWidget *button_group;
78   GtkWidget *treeview_groups;
79
80   /* Details */
81   GtkWidget *vbox_details;
82   GtkWidget *table_details;
83   GtkWidget *hbox_details_requested;
84
85   /* Client */
86   GtkWidget *vbox_client;
87   GtkWidget *table_client;
88   GtkWidget *hbox_client_requested;
89 } EmpathyContactWidget;
90
91 typedef struct
92 {
93   EmpathyContactWidget *information;
94   const gchar *name;
95   gboolean found;
96   GtkTreeIter found_iter;
97 } FindName;
98
99 static void contact_widget_destroy_cb (GtkWidget *widget,
100     EmpathyContactWidget *information);
101 static void contact_widget_remove_contact (EmpathyContactWidget *information);
102 static void contact_widget_set_contact (EmpathyContactWidget *information,
103     EmpathyContact *contact);
104 static void contact_widget_contact_setup (EmpathyContactWidget *information);
105 static void contact_widget_contact_update (EmpathyContactWidget *information);
106 static void contact_widget_change_contact (EmpathyContactWidget *information);
107 static void contact_widget_avatar_changed_cb (EmpathyAvatarChooser *chooser,
108     EmpathyContactWidget *information);
109 static gboolean contact_widget_id_focus_out_cb (GtkWidget *widget,
110     GdkEventFocus *event, EmpathyContactWidget *information);
111 static gboolean contact_widget_entry_alias_focus_event_cb (
112     GtkEditable *editable, GdkEventFocus *event,
113     EmpathyContactWidget *information);
114 static void contact_widget_name_notify_cb (EmpathyContactWidget *information);
115 static void contact_widget_presence_notify_cb (
116     EmpathyContactWidget *information);
117 static void contact_widget_avatar_notify_cb (
118     EmpathyContactWidget *information);
119 static void contact_widget_groups_setup (
120     EmpathyContactWidget *information);
121 static void contact_widget_groups_update (EmpathyContactWidget *information);
122 static void contact_widget_model_setup (EmpathyContactWidget *information);
123 static void contact_widget_model_populate_columns (
124     EmpathyContactWidget *information);
125 static void contact_widget_groups_populate_data (
126     EmpathyContactWidget *information);
127 static void contact_widget_groups_notify_cb (
128     EmpathyContactWidget *information);
129 static gboolean contact_widget_model_find_name (
130     EmpathyContactWidget *information,const gchar *name, GtkTreeIter *iter);
131 static gboolean contact_widget_model_find_name_foreach (GtkTreeModel *model,
132     GtkTreePath *path, GtkTreeIter *iter, FindName *data);
133 static void contact_widget_cell_toggled (GtkCellRendererToggle *cell,
134     gchar *path_string, EmpathyContactWidget *information);
135 static void contact_widget_entry_group_changed_cb (GtkEditable *editable,
136     EmpathyContactWidget *information);
137 static void contact_widget_entry_group_activate_cb (GtkEntry *entry,
138     EmpathyContactWidget *information);
139 static void contact_widget_button_group_clicked_cb (GtkButton *button,
140     EmpathyContactWidget *information);
141 static void contact_widget_details_setup (EmpathyContactWidget *information);
142 static void contact_widget_details_update (EmpathyContactWidget *information);
143 static void contact_widget_client_setup (EmpathyContactWidget *information);
144 static void contact_widget_client_update (EmpathyContactWidget *information);
145
146 enum
147 {
148   COL_NAME,
149   COL_ENABLED,
150   COL_EDITABLE,
151   COL_COUNT
152 };
153
154 GtkWidget *
155 empathy_contact_widget_new (EmpathyContact *contact,
156                             EmpathyContactWidgetFlags flags)
157 {
158   EmpathyContactWidget *information;
159   GtkBuilder *gui;
160   TpConnection *connection;
161   gchar *filename;
162
163   g_return_val_if_fail (contact == NULL || EMPATHY_IS_CONTACT (contact), NULL);
164
165   information = g_slice_new0 (EmpathyContactWidget);
166   information->flags = flags;
167
168   filename = empathy_file_lookup ("empathy-contact-widget.ui",
169       "libempathy-gtk");
170   gui = empathy_builder_get_file (filename,
171        "vbox_contact_widget", &information->vbox_contact_widget,
172        "vbox_contact", &information->vbox_contact,
173        "hbox_presence", &information->hbox_presence,
174        "label_alias", &information->label_alias,
175        "image_state", &information->image_state,
176        "label_status", &information->label_status,
177        "table_contact", &information->table_contact,
178        "vbox_avatar", &information->vbox_avatar,
179        "vbox_groups", &information->vbox_groups,
180        "entry_group", &information->entry_group,
181        "button_group", &information->button_group,
182        "treeview_groups", &information->treeview_groups,
183        "vbox_details", &information->vbox_details,
184        "table_details", &information->table_details,
185        "hbox_details_requested", &information->hbox_details_requested,
186        "vbox_client", &information->vbox_client,
187        "table_client", &information->table_client,
188        "hbox_client_requested", &information->hbox_client_requested,
189        NULL);
190   g_free (filename);
191
192   empathy_builder_connect (gui, information,
193       "vbox_contact_widget", "destroy", contact_widget_destroy_cb,
194       "entry_group", "changed", contact_widget_entry_group_changed_cb,
195       "entry_group", "activate", contact_widget_entry_group_activate_cb,
196       "button_group", "clicked", contact_widget_button_group_clicked_cb,
197       NULL);
198
199   g_object_set_data (G_OBJECT (information->vbox_contact_widget),
200       "EmpathyContactWidget",
201       information);
202
203   /* Create widgets */
204   contact_widget_contact_setup (information);
205   contact_widget_groups_setup (information);
206   contact_widget_details_setup (information);
207   contact_widget_client_setup (information);
208
209   if (contact != NULL)
210       contact_widget_set_contact (information, contact);
211
212   return empathy_builder_unref_and_keep_widget (gui,
213     information->vbox_contact_widget);
214 }
215
216 EmpathyContact *
217 empathy_contact_widget_get_contact (GtkWidget *widget)
218 {
219   EmpathyContactWidget *information;
220
221   g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
222
223   information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
224   if (!information)
225       return NULL;
226
227   return information->contact;
228 }
229
230 void
231 empathy_contact_widget_set_contact (GtkWidget *widget,
232                                     EmpathyContact *contact)
233 {
234   EmpathyContactWidget *information;
235
236   g_return_if_fail (GTK_IS_WIDGET (widget));
237   g_return_if_fail (EMPATHY_IS_CONTACT (contact));
238
239   information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
240   if (!information)
241     return;
242
243   contact_widget_set_contact (information, contact);
244 }
245
246 void
247 empathy_contact_widget_set_account_filter (
248     GtkWidget *widget,
249     EmpathyAccountChooserFilterFunc filter,
250     gpointer user_data)
251 {
252   EmpathyContactWidget *information;
253   EmpathyAccountChooser *chooser;
254
255   g_return_if_fail (GTK_IS_WIDGET (widget));
256
257   information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
258   if (!information)
259     return;
260
261   chooser = EMPATHY_ACCOUNT_CHOOSER (information->widget_account);
262   if (chooser)
263       empathy_account_chooser_set_filter (chooser, filter, user_data);
264 }
265   
266 static void
267 contact_widget_destroy_cb (GtkWidget *widget,
268                            EmpathyContactWidget *information)
269 {
270   contact_widget_remove_contact (information);
271
272   if (information->widget_id_timeout != 0)
273     {
274       g_source_remove (information->widget_id_timeout);
275     }
276   if (information->manager)
277     {
278       g_object_unref (information->manager);
279     }   
280
281   g_slice_free (EmpathyContactWidget, information);
282 }
283
284 static void
285 contact_widget_remove_contact (EmpathyContactWidget *information)
286 {
287   if (information->contact)
288     {
289       g_signal_handlers_disconnect_by_func (information->contact,
290           contact_widget_name_notify_cb, information);
291       g_signal_handlers_disconnect_by_func (information->contact,
292           contact_widget_presence_notify_cb, information);
293       g_signal_handlers_disconnect_by_func (information->contact,
294           contact_widget_avatar_notify_cb, information);
295       g_signal_handlers_disconnect_by_func (information->contact,
296           contact_widget_groups_notify_cb, information);
297
298       g_object_unref (information->contact);
299       g_object_unref (information->factory);
300       information->contact = NULL;
301       information->factory = NULL;
302     }
303 }
304
305 static void
306 contact_widget_set_contact (EmpathyContactWidget *information,
307                             EmpathyContact *contact)
308 {
309   if (contact == information->contact)
310     return;
311
312   contact_widget_remove_contact (information);
313   if (contact)
314     {
315       TpConnection *connection;
316
317       connection = empathy_contact_get_connection (contact);
318       information->contact = g_object_ref (contact);
319       information->factory = empathy_tp_contact_factory_dup_singleton (connection);
320     }
321
322   /* Update information for widgets */
323   contact_widget_contact_update (information);
324   contact_widget_groups_update (information);
325   contact_widget_details_update (information);
326   contact_widget_client_update (information);
327 }
328
329 static gboolean
330 contact_widget_id_activate_timeout (EmpathyContactWidget *self)
331 {
332   contact_widget_change_contact (self);
333   return FALSE;
334 }
335
336 static void
337 contact_widget_id_changed_cb (GtkEntry *entry,
338                               EmpathyContactWidget *self)
339 {
340   if (self->widget_id_timeout != 0)
341     {   
342       g_source_remove (self->widget_id_timeout);
343     }
344
345   self->widget_id_timeout =
346     g_timeout_add_seconds (ID_CHANGED_TIMEOUT,
347         (GSourceFunc) contact_widget_id_activate_timeout, self);
348 }
349
350 static void
351 save_avatar_menu_activate_cb (GtkWidget *widget,
352                               EmpathyContactWidget *information)
353 {
354   GtkWidget *dialog;
355   EmpathyAvatar *avatar;
356   gchar *ext = NULL, *filename;
357
358   dialog = gtk_file_chooser_dialog_new (_("Save Avatar"),
359       NULL,
360       GTK_FILE_CHOOSER_ACTION_SAVE,
361       GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
362       GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
363       NULL);
364
365   gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog),
366       TRUE);
367
368   /* look for the avatar extension */
369   avatar = empathy_contact_get_avatar (information->contact);
370   if (avatar->format != NULL)
371     {
372       gchar **splitted;
373
374       splitted = g_strsplit (avatar->format, "/", 2);
375       if (splitted[0] != NULL && splitted[1] != NULL)
376           ext = g_strdup (splitted[1]);
377
378       g_strfreev (splitted);
379     }
380   else
381     {
382       /* Avatar was loaded from the cache so was converted to PNG */
383       ext = g_strdup ("png");
384     }
385
386   if (ext != NULL)
387     {
388       gchar *id;
389
390       id = tp_escape_as_identifier (empathy_contact_get_id (
391             information->contact));
392
393       filename = g_strdup_printf ("%s.%s", id, ext);
394       gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), filename);
395
396       g_free (id);
397       g_free (ext);
398       g_free (filename);
399     }
400
401   if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT)
402     {
403       GError *error = NULL;
404
405       filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
406
407       if (!empathy_avatar_save_to_file (avatar, filename, &error))
408         {
409           /* Save error */
410           GtkWidget *dialog;
411
412           dialog = gtk_message_dialog_new (NULL, 0,
413               GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, 
414               _("Unable to save avatar"));
415
416           gtk_message_dialog_format_secondary_text (
417               GTK_MESSAGE_DIALOG (dialog), "%s", error->message);
418
419           g_signal_connect (dialog, "response",
420               G_CALLBACK (gtk_widget_destroy), NULL);
421
422           gtk_window_present (GTK_WINDOW (dialog));
423
424           g_clear_error (&error);
425         }
426
427       g_free (filename);
428     }
429
430   gtk_widget_destroy (dialog);
431 }
432
433 static void
434 popup_avatar_menu (EmpathyContactWidget *information,
435                    GtkWidget *parent,
436                    GdkEventButton *event)
437 {
438   GtkWidget *menu, *item;
439   gint button, event_time;
440
441   if (information->contact == NULL ||
442       empathy_contact_get_avatar (information->contact) == NULL)
443       return;
444
445   menu = gtk_menu_new ();
446
447   /* Add "Save as..." entry */
448   item = gtk_image_menu_item_new_from_stock (GTK_STOCK_SAVE_AS, NULL);
449   gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
450   gtk_widget_show (item);
451
452   g_signal_connect (item, "activate",
453       G_CALLBACK (save_avatar_menu_activate_cb), information);
454
455   if (event)
456     {
457       button = event->button;
458       event_time = event->time;
459     }
460   else
461     {
462       button = 0;
463       event_time = gtk_get_current_event_time ();
464     }
465
466   gtk_menu_attach_to_widget (GTK_MENU (menu), parent, NULL);
467   gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL, 
468       button, event_time);
469 }
470
471 static gboolean
472 widget_avatar_popup_menu_cb (GtkWidget *widget,
473                              EmpathyContactWidget *information)
474 {
475   popup_avatar_menu (information, widget, NULL);
476
477   return TRUE;
478 }
479
480 static gboolean
481 widget_avatar_button_press_event_cb (GtkWidget *widget,
482                                      GdkEventButton *event,
483                                      EmpathyContactWidget *information)
484 {
485   /* Ignore double-clicks and triple-clicks */
486   if (event->button == 3 && event->type == GDK_BUTTON_PRESS)
487     {
488       popup_avatar_menu (information, widget, event);
489       return TRUE;
490     }
491
492   return FALSE;
493 }
494
495 static void
496 update_avatar_chooser_account_cb (EmpathyAccountChooser *account_chooser,
497                                   EmpathyAvatarChooser *avatar_chooser)
498 {
499   TpConnection *connection;
500
501   connection = empathy_account_chooser_get_connection (account_chooser);
502   g_object_set (avatar_chooser, "connection", connection, NULL);
503 }
504
505 static void
506 contact_widget_contact_setup (EmpathyContactWidget *information)
507 {
508   /* Setup account label/chooser */
509   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ACCOUNT)
510     {
511       information->widget_account = empathy_account_chooser_new ();
512
513       contact_widget_change_contact (information);
514       g_signal_connect_swapped (information->widget_account, "changed",
515             G_CALLBACK (contact_widget_change_contact),
516             information);
517     }
518   else
519     {
520       information->widget_account = gtk_label_new (NULL);
521       if (!(information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP)) {
522         gtk_label_set_selectable (GTK_LABEL (information->widget_account), TRUE);
523       }
524       gtk_misc_set_alignment (GTK_MISC (information->widget_account), 0, 0.5);
525     }
526   gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
527            information->widget_account,
528            1, 2, 0, 1);
529   gtk_widget_show (information->widget_account);
530
531   /* Set up avatar chooser/display */
532   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_AVATAR)
533     {
534       information->widget_avatar = empathy_avatar_chooser_new ();
535       g_signal_connect (information->widget_avatar, "changed",
536             G_CALLBACK (contact_widget_avatar_changed_cb),
537             information);
538       if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ACCOUNT)
539         {
540           g_signal_connect (information->widget_account, "changed",
541               G_CALLBACK (update_avatar_chooser_account_cb),
542               information->widget_avatar);
543           update_avatar_chooser_account_cb (
544               EMPATHY_ACCOUNT_CHOOSER (information->widget_account),
545               EMPATHY_AVATAR_CHOOSER (information->widget_avatar));
546         }
547     }
548   else
549     {
550       information->widget_avatar = empathy_avatar_image_new ();
551
552       g_signal_connect (information->widget_avatar, "popup-menu",
553           G_CALLBACK (widget_avatar_popup_menu_cb), information);
554       g_signal_connect (information->widget_avatar, "button-press-event",
555           G_CALLBACK (widget_avatar_button_press_event_cb), information);
556     }
557
558   gtk_box_pack_start (GTK_BOX (information->vbox_avatar),
559           information->widget_avatar,
560           FALSE, FALSE,
561           6);
562   gtk_widget_show (information->widget_avatar);
563
564   /* Setup id label/entry */
565   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
566     {
567       information->widget_id = gtk_entry_new ();
568       g_signal_connect (information->widget_id, "focus-out-event",
569             G_CALLBACK (contact_widget_id_focus_out_cb),
570             information);
571       g_signal_connect (information->widget_id, "changed",
572             G_CALLBACK (contact_widget_id_changed_cb),
573             information);
574     }
575   else
576     {
577       information->widget_id = gtk_label_new (NULL);
578       if (!(information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP)) {
579         gtk_label_set_selectable (GTK_LABEL (information->widget_id), TRUE);
580       }
581       gtk_misc_set_alignment (GTK_MISC (information->widget_id), 0, 0.5);
582     }
583   gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
584            information->widget_id,
585            1, 2, 1, 2);
586   gtk_widget_show (information->widget_id);
587
588   /* Setup alias label/entry */
589   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ALIAS)
590     {
591       information->widget_alias = gtk_entry_new ();
592       g_signal_connect (information->widget_alias, "focus-out-event",
593             G_CALLBACK (contact_widget_entry_alias_focus_event_cb),
594             information);
595       /* Make return activate the window default (the Close button) */
596       gtk_entry_set_activates_default (GTK_ENTRY (information->widget_alias),
597           TRUE);
598     }
599   else
600     {
601       information->widget_alias = gtk_label_new (NULL);
602       if (!(information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP)) {
603         gtk_label_set_selectable (GTK_LABEL (information->widget_alias), TRUE);
604       }
605       gtk_misc_set_alignment (GTK_MISC (information->widget_alias), 0, 0.5);
606     }
607   gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
608            information->widget_alias,
609            1, 2, 2, 3);
610   if (information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP) {
611     gtk_label_set_selectable (GTK_LABEL (information->label_status), FALSE);
612   }
613   gtk_widget_show (information->widget_alias);
614 }
615
616 static void
617 contact_widget_contact_update (EmpathyContactWidget *information)
618 {
619   McAccount *account = NULL;
620   const gchar *id = NULL;
621
622   /* Connect and get info from new contact */
623   if (information->contact)
624     {
625       g_signal_connect_swapped (information->contact, "notify::name",
626           G_CALLBACK (contact_widget_name_notify_cb), information);
627       g_signal_connect_swapped (information->contact, "notify::presence",
628           G_CALLBACK (contact_widget_presence_notify_cb), information);
629       g_signal_connect_swapped (information->contact,
630           "notify::presence-message",
631           G_CALLBACK (contact_widget_presence_notify_cb), information);
632       g_signal_connect_swapped (information->contact, "notify::avatar",
633           G_CALLBACK (contact_widget_avatar_notify_cb), information);
634
635       account = empathy_contact_get_account (information->contact);
636       id = empathy_contact_get_id (information->contact);
637     }
638
639   /* Update account widget */
640   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ACCOUNT)
641     {
642       if (account)
643         {
644           g_signal_handlers_block_by_func (information->widget_account,
645                    contact_widget_change_contact,
646                    information);
647           empathy_account_chooser_set_account (
648               EMPATHY_ACCOUNT_CHOOSER (information->widget_account), account);
649           g_signal_handlers_unblock_by_func (information->widget_account,
650               contact_widget_change_contact, information);
651         }
652     }
653   else
654     {
655       if (account)
656         {
657           const gchar *name;
658
659           name = mc_account_get_display_name (account);
660           gtk_label_set_label (GTK_LABEL (information->widget_account), name);
661         }
662     }
663
664   /* Update id widget */
665   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
666       gtk_entry_set_text (GTK_ENTRY (information->widget_id), id ? id : "");
667   else
668       gtk_label_set_label (GTK_LABEL (information->widget_id), id ? id : "");
669
670   /* Update other widgets */
671   if (information->contact)
672     {
673       contact_widget_name_notify_cb (information);
674       contact_widget_presence_notify_cb (information);
675       contact_widget_avatar_notify_cb (information);
676
677       gtk_widget_show (information->label_alias);
678       gtk_widget_show (information->widget_alias);
679       gtk_widget_show (information->hbox_presence);
680       gtk_widget_show (information->widget_avatar);
681     }
682   else
683     {
684       gtk_widget_hide (information->label_alias);
685       gtk_widget_hide (information->widget_alias);
686       gtk_widget_hide (information->hbox_presence);
687       gtk_widget_hide (information->widget_avatar);
688     }
689 }
690
691 static void
692 contact_widget_got_contact_cb (EmpathyTpContactFactory *factory,
693                                EmpathyContact *contact,
694                                const GError *error,
695                                gpointer user_data,
696                                GObject *weak_object)
697 {
698   EmpathyContactWidget *information = user_data;
699
700   if (error != NULL)
701     {
702       DEBUG ("Error: %s", error->message);
703       return;
704     }
705
706   contact_widget_set_contact (information, contact);
707 }
708
709 static void
710 contact_widget_get_self_handle_cb (TpConnection *connection,
711                                    TpHandle self_handle,
712                                    const GError *error,
713                                    gpointer information,
714                                    GObject *weak_object)
715 {
716   EmpathyTpContactFactory *factory;
717
718   if (error != NULL)
719     {
720       DEBUG ("Error: %s", error->message);
721       return;
722     }
723
724   factory = empathy_tp_contact_factory_dup_singleton (connection);
725   empathy_tp_contact_factory_get_from_handle (factory, self_handle,
726       contact_widget_got_contact_cb, information, NULL,
727       weak_object);
728   g_object_unref (factory);
729 }
730
731 static void
732 contact_widget_change_contact (EmpathyContactWidget *information)
733 {
734   TpConnection *connection;
735
736   connection = empathy_account_chooser_get_connection (
737       EMPATHY_ACCOUNT_CHOOSER (information->widget_account));
738   if (!connection)
739       return;
740
741   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
742     {
743       const gchar *id;
744
745       id = gtk_entry_get_text (GTK_ENTRY (information->widget_id));
746       if (!EMP_STR_EMPTY (id))
747         {
748           EmpathyTpContactFactory *factory;
749
750           factory = empathy_tp_contact_factory_dup_singleton (connection);
751           empathy_tp_contact_factory_get_from_id (factory, id,
752               contact_widget_got_contact_cb, information, NULL,
753               G_OBJECT (information->vbox_contact_widget));
754           g_object_unref (factory);
755         }
756     }
757   else
758     {
759       /* FIXME: TpConnection should have a SelfHandle property */
760       tp_cli_connection_call_get_self_handle (connection, -1,
761           contact_widget_get_self_handle_cb, information, NULL,
762           G_OBJECT (information->vbox_contact_widget));
763     }
764 }
765
766 static void
767 contact_widget_avatar_changed_cb (EmpathyAvatarChooser *chooser,
768                                   EmpathyContactWidget *information)
769 {
770   const gchar *data;
771   gsize size;
772   const gchar *mime_type;
773
774   empathy_avatar_chooser_get_image_data (
775       EMPATHY_AVATAR_CHOOSER (information->widget_avatar),
776       &data, &size, &mime_type);
777   empathy_tp_contact_factory_set_avatar (information->factory,
778       data, size, mime_type);
779 }
780
781 static gboolean
782 contact_widget_id_focus_out_cb (GtkWidget *widget,
783                                 GdkEventFocus *event,
784                                 EmpathyContactWidget *information)
785 {
786   contact_widget_change_contact (information);
787   return FALSE;
788 }
789
790 static gboolean
791 contact_widget_entry_alias_focus_event_cb (GtkEditable *editable,
792                                            GdkEventFocus *event,
793                                            EmpathyContactWidget *information)
794 {
795   if (information->contact)
796     {
797       const gchar *alias;
798
799       alias = gtk_entry_get_text (GTK_ENTRY (editable));
800       empathy_tp_contact_factory_set_alias (information->factory,
801           information->contact, alias);
802     }
803
804   return FALSE;
805 }
806
807 static void
808 contact_widget_name_notify_cb (EmpathyContactWidget *information)
809 {
810   if (GTK_IS_ENTRY (information->widget_alias))
811       gtk_entry_set_text (GTK_ENTRY (information->widget_alias),
812           empathy_contact_get_name (information->contact));
813   else
814       gtk_label_set_label (GTK_LABEL (information->widget_alias),
815           empathy_contact_get_name (information->contact));
816 }
817
818 static void
819 contact_widget_presence_notify_cb (EmpathyContactWidget *information)
820 {
821   gtk_label_set_text (GTK_LABEL (information->label_status),
822       empathy_contact_get_status (information->contact));
823   gtk_image_set_from_icon_name (GTK_IMAGE (information->image_state),
824       empathy_icon_name_for_contact (information->contact),
825       GTK_ICON_SIZE_BUTTON);
826 }
827
828 static void
829 contact_widget_avatar_notify_cb (EmpathyContactWidget *information)
830 {
831   EmpathyAvatar *avatar = NULL;
832
833   if (information->contact)
834       avatar = empathy_contact_get_avatar (information->contact);
835
836   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_AVATAR)
837     {
838       g_signal_handlers_block_by_func (information->widget_avatar,
839           contact_widget_avatar_changed_cb,
840           information);
841       empathy_avatar_chooser_set (
842           EMPATHY_AVATAR_CHOOSER (information->widget_avatar), avatar);
843       g_signal_handlers_unblock_by_func (information->widget_avatar,
844           contact_widget_avatar_changed_cb, information);
845     }
846   else
847       empathy_avatar_image_set (
848           EMPATHY_AVATAR_IMAGE (information->widget_avatar), avatar);
849 }
850
851 static void
852 contact_widget_groups_setup (EmpathyContactWidget *information)
853 {
854   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_GROUPS)
855     {
856       information->manager = empathy_contact_manager_dup_singleton ();
857       contact_widget_model_setup (information);
858     }
859 }
860
861 static void
862 contact_widget_groups_update (EmpathyContactWidget *information)
863 {
864   if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_GROUPS &&
865       information->contact)
866     {
867       g_signal_connect_swapped (information->contact, "notify::groups",
868           G_CALLBACK (contact_widget_groups_notify_cb), information);
869       contact_widget_groups_populate_data (information);
870
871       gtk_widget_show (information->vbox_groups);
872     }
873   else
874       gtk_widget_hide (information->vbox_groups);
875 }
876
877 static void
878 contact_widget_model_setup (EmpathyContactWidget *information)
879 {
880   GtkTreeView *view;
881   GtkListStore *store;
882   GtkTreeSelection *selection;
883
884   view = GTK_TREE_VIEW (information->treeview_groups);
885
886   store = gtk_list_store_new (COL_COUNT,
887       G_TYPE_STRING,   /* name */
888       G_TYPE_BOOLEAN,  /* enabled */
889       G_TYPE_BOOLEAN); /* editable */
890
891   gtk_tree_view_set_model (view, GTK_TREE_MODEL (store));
892
893   selection = gtk_tree_view_get_selection (view);
894   gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE);
895
896   contact_widget_model_populate_columns (information);
897
898   gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
899       COL_NAME, GTK_SORT_ASCENDING);
900
901   g_object_unref (store);
902 }
903
904 static void
905 contact_widget_model_populate_columns (EmpathyContactWidget *information)
906 {
907   GtkTreeView *view;
908   GtkTreeModel *model;
909   GtkTreeViewColumn *column;
910   GtkCellRenderer  *renderer;
911   guint col_offset;
912
913   view = GTK_TREE_VIEW (information->treeview_groups);
914   model = gtk_tree_view_get_model (view);
915
916   renderer = gtk_cell_renderer_toggle_new ();
917   g_signal_connect (renderer, "toggled",
918       G_CALLBACK (contact_widget_cell_toggled), information);
919
920   column = gtk_tree_view_column_new_with_attributes (_("Select"), renderer,
921       "active", COL_ENABLED, NULL);
922
923   gtk_tree_view_column_set_sizing (column, GTK_TREE_VIEW_COLUMN_FIXED);
924   gtk_tree_view_column_set_fixed_width (column, 50);
925   gtk_tree_view_append_column (view, column);
926
927   renderer = gtk_cell_renderer_text_new ();
928   col_offset = gtk_tree_view_insert_column_with_attributes (view,
929       -1, _("Group"),
930       renderer,
931       "text", COL_NAME,
932       /* "editable", COL_EDITABLE, */
933       NULL);
934
935   g_object_set_data (G_OBJECT (renderer),
936       "column", GINT_TO_POINTER (COL_NAME));
937
938   column = gtk_tree_view_get_column (view, col_offset - 1);
939   gtk_tree_view_column_set_sort_column_id (column, COL_NAME);
940   gtk_tree_view_column_set_resizable (column,FALSE);
941   gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (column), TRUE);
942 }
943
944 static void
945 contact_widget_groups_populate_data (EmpathyContactWidget *information)
946 {
947   GtkTreeView *view;
948   GtkListStore *store;
949   GtkTreeIter iter;
950   GList *my_groups, *l;
951   GList *all_groups;
952
953   view = GTK_TREE_VIEW (information->treeview_groups);
954   store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
955   gtk_list_store_clear (store);
956
957   all_groups = empathy_contact_list_get_all_groups (
958       EMPATHY_CONTACT_LIST (information->manager));
959   my_groups = empathy_contact_list_get_groups (
960       EMPATHY_CONTACT_LIST (information->manager),
961       information->contact);
962
963   for (l = all_groups; l; l = l->next)
964     {
965       const gchar *group_str;
966       gboolean enabled;
967
968       group_str = l->data;
969
970       enabled = g_list_find_custom (my_groups,
971           group_str, (GCompareFunc) strcmp) != NULL;
972
973       gtk_list_store_append (store, &iter);
974       gtk_list_store_set (store, &iter,
975           COL_NAME, group_str,
976           COL_EDITABLE, TRUE,
977           COL_ENABLED, enabled,
978           -1);
979     }
980
981   g_list_foreach (all_groups, (GFunc) g_free, NULL);
982   g_list_foreach (my_groups, (GFunc) g_free, NULL);
983   g_list_free (all_groups);
984   g_list_free (my_groups);
985 }
986
987 static void
988 contact_widget_groups_notify_cb (EmpathyContactWidget *information)
989 {
990   /* FIXME: not implemented */
991 }
992
993 static gboolean
994 contact_widget_model_find_name (EmpathyContactWidget *information,
995                                 const gchar *name,
996                                 GtkTreeIter *iter)
997 {
998   GtkTreeView *view;
999   GtkTreeModel *model;
1000   FindName data;
1001
1002   if (EMP_STR_EMPTY (name))
1003       return FALSE;
1004
1005   data.information = information;
1006   data.name = name;
1007   data.found = FALSE;
1008
1009   view = GTK_TREE_VIEW (information->treeview_groups);
1010   model = gtk_tree_view_get_model (view);
1011
1012   gtk_tree_model_foreach (model,
1013       (GtkTreeModelForeachFunc) contact_widget_model_find_name_foreach,
1014       &data);
1015
1016   if (data.found == TRUE)
1017     {
1018       *iter = data.found_iter;
1019       return TRUE;
1020     }
1021
1022   return FALSE;
1023 }
1024
1025 static gboolean
1026 contact_widget_model_find_name_foreach (GtkTreeModel *model,
1027                                         GtkTreePath *path,
1028                                         GtkTreeIter *iter,
1029                                         FindName *data)
1030 {
1031   gchar *name;
1032
1033   gtk_tree_model_get (model, iter,
1034       COL_NAME, &name,
1035       -1);
1036
1037   if (!name)
1038       return FALSE;
1039
1040   if (data->name && strcmp (data->name, name) == 0)
1041     {
1042       data->found = TRUE;
1043       data->found_iter = *iter;
1044
1045       g_free (name);
1046
1047       return TRUE;
1048     }
1049
1050   g_free (name);
1051
1052   return FALSE;
1053 }
1054
1055 static void
1056 contact_widget_cell_toggled (GtkCellRendererToggle *cell,
1057                              gchar *path_string,
1058                              EmpathyContactWidget *information)
1059 {
1060   GtkTreeView *view;
1061   GtkTreeModel *model;
1062   GtkListStore *store;
1063   GtkTreePath *path;
1064   GtkTreeIter iter;
1065   gboolean enabled;
1066   gchar *group;
1067
1068   view = GTK_TREE_VIEW (information->treeview_groups);
1069   model = gtk_tree_view_get_model (view);
1070   store = GTK_LIST_STORE (model);
1071
1072   path = gtk_tree_path_new_from_string (path_string);
1073
1074   gtk_tree_model_get_iter (model, &iter, path);
1075   gtk_tree_model_get (model, &iter,
1076       COL_ENABLED, &enabled,
1077       COL_NAME, &group,
1078       -1);
1079
1080   gtk_list_store_set (store, &iter, COL_ENABLED, !enabled, -1);
1081   gtk_tree_path_free (path);
1082
1083   if (group)
1084     {
1085       if (enabled)
1086         {
1087           empathy_contact_list_remove_from_group (
1088               EMPATHY_CONTACT_LIST (information->manager), information->contact,
1089               group);
1090         }
1091       else
1092         {
1093           empathy_contact_list_add_to_group (
1094               EMPATHY_CONTACT_LIST (information->manager), information->contact,
1095               group);
1096         }
1097       g_free (group);
1098     }
1099 }
1100
1101 static void
1102 contact_widget_entry_group_changed_cb (GtkEditable *editable,
1103                                        EmpathyContactWidget *information)
1104 {
1105   GtkTreeIter iter;
1106   const gchar *group;
1107
1108   group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
1109
1110   if (contact_widget_model_find_name (information, group, &iter))
1111       gtk_widget_set_sensitive (GTK_WIDGET (information->button_group), FALSE);
1112   else
1113       gtk_widget_set_sensitive (GTK_WIDGET (information->button_group),
1114           !EMP_STR_EMPTY (group));
1115 }
1116
1117 static void
1118 contact_widget_entry_group_activate_cb (GtkEntry *entry,
1119                                         EmpathyContactWidget  *information)
1120 {
1121   gtk_widget_activate (GTK_WIDGET (information->button_group));
1122 }
1123
1124 static void
1125 contact_widget_button_group_clicked_cb (GtkButton *button,
1126                                         EmpathyContactWidget *information)
1127 {
1128   GtkTreeView *view;
1129   GtkListStore *store;
1130   GtkTreeIter iter;
1131   const gchar *group;
1132
1133   view = GTK_TREE_VIEW (information->treeview_groups);
1134   store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
1135
1136   group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
1137
1138   gtk_list_store_append (store, &iter);
1139   gtk_list_store_set (store, &iter,
1140       COL_NAME, group,
1141       COL_ENABLED, TRUE,
1142       -1);
1143
1144   empathy_contact_list_add_to_group (
1145       EMPATHY_CONTACT_LIST (information->manager), information->contact,
1146       group);
1147 }
1148
1149 static void
1150 contact_widget_details_setup (EmpathyContactWidget *information)
1151 {
1152   /* FIXME: Needs new telepathy spec */
1153   gtk_widget_hide (information->vbox_details);
1154 }
1155
1156 static void
1157 contact_widget_details_update (EmpathyContactWidget *information)
1158 {
1159   /* FIXME: Needs new telepathy spec */
1160 }
1161
1162 static void
1163 contact_widget_client_setup (EmpathyContactWidget *information)
1164 {
1165   /* FIXME: Needs new telepathy spec */
1166   gtk_widget_hide (information->vbox_client);
1167 }
1168
1169 static void
1170 contact_widget_client_update (EmpathyContactWidget *information)
1171 {
1172   /* FIXME: Needs new telepathy spec */
1173 }