]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-contact-widget.c
Updated Basque translation.
[empathy.git] / libempathy-gtk / empathy-contact-widget.c
index b8ed7efb0c9ce3cc754034e32f852ee74f5c89d6..66c03b4344e6d4fe212840409d959fd74bb72e68 100644 (file)
@@ -1,21 +1,20 @@
 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
 /*
- * Copyright (C) 2007 Collabora Ltd.
+ * Copyright (C) 2007-2008 Collabora Ltd.
  *
- * This program is free software; you can redistribute it and/or
- * modify it under the terms of the GNU General Public License as
- * published by the Free Software Foundation; either version 2 of the
- * License, or (at your option) any later version.
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
  *
- * This program is distributed in the hope that it will be useful,
+ * This library is distributed in the hope that it will be useful,
  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
- * General Public License for more details.
+ * Lesser General Public License for more details.
  *
- * You should have received a copy of the GNU General Public
- * License along with this program; if not, write to the
- * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
- * Boston, MA 02111-1307, USA.
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
  *
  * Authors: Xavier Claessens <xclaesse@gmail.com>
  */
 #include <glade/glade.h>
 #include <glib/gi18n.h>
 
+#include <libmissioncontrol/mc-account.h>
+#include <telepathy-glib/util.h>
+
+#include <libempathy/empathy-contact-factory.h>
 #include <libempathy/empathy-contact-manager.h>
+#include <libempathy/empathy-contact-list.h>
+#include <libempathy/empathy-utils.h>
 
 #include "empathy-contact-widget.h"
-#include "gossip-ui-utils.h"
-
-typedef struct {
-       GossipContact   *contact;
-       gboolean         is_user;
-       gboolean         editable;
-       gboolean         changes_made;
-       GtkCellRenderer *renderer;
-
-       GtkWidget       *vbox_contact_widget;
-
-       GtkWidget       *vbox_contact;
-       GtkWidget       *widget_avatar;
-       GtkWidget       *label_id;
-       GtkWidget       *entry_alias;
-       GtkWidget       *widget_alias;
-       GtkWidget       *image_state;
-       GtkWidget       *label_status;
-       GtkWidget       *table_contact;
-       GtkWidget       *hbox_contact;
-
-       GtkWidget       *vbox_groups;
-       GtkWidget       *entry_group;
-       GtkWidget       *button_group;
-       GtkWidget       *treeview_groups;
-
-       GtkWidget       *vbox_details;
-       GtkWidget       *table_details;
-       GtkWidget       *hbox_details_requested;
-
-       GtkWidget       *vbox_client;
-       GtkWidget       *table_client;
-       GtkWidget       *hbow_client_requested;
+#include "empathy-account-chooser.h"
+#include "empathy-avatar-chooser.h"
+#include "empathy-avatar-image.h"
+#include "empathy-ui-utils.h"
+
+/* Delay before updating the widget when the id entry changed (seconds) */
+#define ID_CHANGED_TIMEOUT 1
+
+typedef struct
+{
+  EmpathyContactFactory *factory;
+  EmpathyContactManager *manager;
+  EmpathyContact *contact;
+  EmpathyContactWidgetFlags flags;
+  GtkCellRenderer *renderer;
+  guint widget_id_timeout;
+
+  GtkWidget *vbox_contact_widget;
+
+  /* Contact */
+  GtkWidget *vbox_contact;
+  GtkWidget *widget_avatar;
+  GtkWidget *widget_account;
+  GtkWidget *widget_id;
+  GtkWidget *widget_alias;
+  GtkWidget *label_alias;
+  GtkWidget *entry_alias;
+  GtkWidget *hbox_presence;
+  GtkWidget *image_state;
+  GtkWidget *label_status;
+  GtkWidget *table_contact;
+  GtkWidget *vbox_avatar;
+
+  /* Groups */
+  GtkWidget *vbox_groups;
+  GtkWidget *entry_group;
+  GtkWidget *button_group;
+  GtkWidget *treeview_groups;
+
+  /* Details */
+  GtkWidget *vbox_details;
+  GtkWidget *table_details;
+  GtkWidget *hbox_details_requested;
+
+  /* Client */
+  GtkWidget *vbox_client;
+  GtkWidget *table_client;
+  GtkWidget *hbox_client_requested;
 } EmpathyContactWidget;
 
-typedef struct {
-       EmpathyContactWidget *information;
-       const gchar          *name;
-       gboolean              found;
-       GtkTreeIter           found_iter;
+typedef struct
+{
+  EmpathyContactWidget *information;
+  const gchar *name;
+  gboolean found;
+  GtkTreeIter found_iter;
 } FindName;
 
-typedef struct {
-       EmpathyContactWidget *information;
-       GList                *list;
-} FindSelected;
-
-static void     contact_widget_destroy_cb                  (GtkWidget             *widget,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_contact_setup               (EmpathyContactWidget  *information);
-static void     contact_widget_name_notify_cb              (EmpathyContactWidget  *information);
-static void     contact_widget_presence_notify_cb          (EmpathyContactWidget  *information);
-static void     contact_widget_avatar_notify_cb            (EmpathyContactWidget  *information);
-static void     contact_widget_groups_setup                (EmpathyContactWidget  *information);
-static void     contact_widget_model_setup                 (EmpathyContactWidget  *information);
-static void     contact_widget_model_populate_columns      (EmpathyContactWidget  *information);
-static void     contact_widget_groups_populate_data        (EmpathyContactWidget  *information);
-static void     contact_widget_groups_notify_cb            (EmpathyContactWidget  *information);
-static gboolean contact_widget_model_find_name             (EmpathyContactWidget  *information,
-                                                           const gchar           *name,
-                                                           GtkTreeIter           *iter);
-static gboolean contact_widget_model_find_name_foreach     (GtkTreeModel          *model,
-                                                           GtkTreePath           *path,
-                                                           GtkTreeIter           *iter,
-                                                           FindName              *data);
-static GList *  contact_widget_model_find_selected         (EmpathyContactWidget  *information);
-static gboolean contact_widget_model_find_selected_foreach (GtkTreeModel          *model,
-                                                           GtkTreePath           *path,
-                                                           GtkTreeIter           *iter,
-                                                           FindSelected          *data);
-static void     contact_widget_cell_toggled                (GtkCellRendererToggle *cell,
-                                                           gchar                 *path_string,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_entry_alias_changed_cb      (GtkEditable           *editable,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_entry_group_changed_cb      (GtkEditable           *editable,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_entry_group_activate_cb     (GtkEntry              *entry,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_button_group_clicked_cb     (GtkButton             *button,
-                                                           EmpathyContactWidget  *information);
-static void     contact_widget_details_setup               (EmpathyContactWidget  *information);
-static void     contact_widget_client_setup                (EmpathyContactWidget  *information);
-
-enum {
-       COL_NAME,
-       COL_ENABLED,
-       COL_EDITABLE,
-       COL_COUNT
+static void contact_widget_destroy_cb (GtkWidget *widget,
+    EmpathyContactWidget *information);
+static void contact_widget_remove_contact (EmpathyContactWidget *information);
+static void contact_widget_set_contact (EmpathyContactWidget *information,
+    EmpathyContact *contact);
+static void contact_widget_contact_setup (EmpathyContactWidget *information);
+static void contact_widget_contact_update (EmpathyContactWidget *information);
+static void contact_widget_change_contact (EmpathyContactWidget *information);
+static void contact_widget_avatar_changed_cb (EmpathyAvatarChooser *chooser,
+    EmpathyContactWidget *information);
+static void contact_widget_account_changed_cb (GtkComboBox *widget,
+    EmpathyContactWidget *information);
+static gboolean contact_widget_id_focus_out_cb (GtkWidget *widget,
+    GdkEventFocus *event, EmpathyContactWidget *information);
+static gboolean contact_widget_entry_alias_focus_event_cb (
+    GtkEditable *editable, GdkEventFocus *event,
+    EmpathyContactWidget *information);
+static void contact_widget_name_notify_cb (EmpathyContactWidget *information);
+static void contact_widget_presence_notify_cb (
+    EmpathyContactWidget *information);
+static void contact_widget_avatar_notify_cb (
+    EmpathyContactWidget *information);
+static void contact_widget_groups_setup (
+    EmpathyContactWidget *information);
+static void contact_widget_groups_update (EmpathyContactWidget *information);
+static void contact_widget_model_setup (EmpathyContactWidget *information);
+static void contact_widget_model_populate_columns (
+    EmpathyContactWidget *information);
+static void contact_widget_groups_populate_data (
+    EmpathyContactWidget *information);
+static void contact_widget_groups_notify_cb (
+    EmpathyContactWidget *information);
+static gboolean contact_widget_model_find_name (
+    EmpathyContactWidget *information,const gchar *name, GtkTreeIter *iter);
+static gboolean contact_widget_model_find_name_foreach (GtkTreeModel *model,
+    GtkTreePath *path, GtkTreeIter *iter, FindName *data);
+static void contact_widget_cell_toggled (GtkCellRendererToggle *cell,
+    gchar *path_string, EmpathyContactWidget *information);
+static void contact_widget_entry_group_changed_cb (GtkEditable *editable,
+    EmpathyContactWidget *information);
+static void contact_widget_entry_group_activate_cb (GtkEntry *entry,
+    EmpathyContactWidget *information);
+static void contact_widget_button_group_clicked_cb (GtkButton *button,
+    EmpathyContactWidget *information);
+static void contact_widget_details_setup (EmpathyContactWidget *information);
+static void contact_widget_details_update (EmpathyContactWidget *information);
+static void contact_widget_client_setup (EmpathyContactWidget *information);
+static void contact_widget_client_update (EmpathyContactWidget *information);
+
+enum
+{
+  COL_NAME,
+  COL_ENABLED,
+  COL_EDITABLE,
+  COL_COUNT
 };
 
 GtkWidget *
-empathy_contact_widget_new (GossipContact *contact,
-                           gboolean       editable)
-{
-       EmpathyContactWidget *information;
-       GladeXML             *glade;
-       GossipContact        *user_contact;
-
-       g_return_val_if_fail (GOSSIP_IS_CONTACT (contact), NULL);
-
-       information = g_slice_new0 (EmpathyContactWidget);
-       information->contact = g_object_ref (contact);
-       user_contact = gossip_contact_get_user (contact);
-       information->is_user = gossip_contact_equal (contact, user_contact);
-       information->editable = editable;
-
-       glade = gossip_glade_get_file ("empathy-contact-widget.glade",
-                                      "vbox_contact_widget",
-                                      NULL,
-                                      "vbox_contact_widget", &information->vbox_contact_widget,
-                                      "vbox_contact", &information->vbox_contact,
-                                      "label_id", &information->label_id,
-                                      "image_state", &information->image_state,
-                                      "label_status", &information->label_status,
-                                      "table_contact", &information->table_contact,
-                                      "hbox_contact", &information->hbox_contact,
-                                      "vbox_groups", &information->vbox_groups,
-                                      "entry_group", &information->entry_group,
-                                      "button_group", &information->button_group,
-                                      "treeview_groups", &information->treeview_groups,
-                                      "vbox_details", &information->vbox_details,
-                                      "table_details", &information->table_details,
-                                      "hbox_details_requested", &information->hbox_details_requested,
-                                      "vbox_client", &information->vbox_client,
-                                      "table_client", &information->table_client,
-                                      "hbox_client_requested", &information->hbow_client_requested,
-                                      NULL);
-
-       gossip_glade_connect (glade,
-                             information,
-                             "vbox_contact_widget", "destroy", contact_widget_destroy_cb,
-                             "entry_group", "changed", contact_widget_entry_group_changed_cb,
-                             "entry_group", "activate", contact_widget_entry_group_activate_cb,
-                             "button_group", "clicked", contact_widget_button_group_clicked_cb,
-                             NULL);
-
-       g_object_unref (glade);
-
-       g_object_set_data (G_OBJECT (information->vbox_contact_widget),
-                          "EmpathyContactWidget",
-                          information);
-
-       contact_widget_contact_setup (information);
-       contact_widget_groups_setup (information);
-       contact_widget_details_setup (information);
-       contact_widget_client_setup (information);
-
-       gtk_widget_show (information->vbox_contact_widget);
-
-       return information->vbox_contact_widget;
+empathy_contact_widget_new (EmpathyContact *contact,
+                            EmpathyContactWidgetFlags flags)
+{
+  EmpathyContactWidget *information;
+  GladeXML *glade;
+  gchar *filename;
+
+  information = g_slice_new0 (EmpathyContactWidget);
+  information->flags = flags;
+  information->factory = empathy_contact_factory_new ();
+
+  filename = empathy_file_lookup ("empathy-contact-widget.glade",
+      "libempathy-gtk");
+  glade = empathy_glade_get_file (filename,
+      "vbox_contact_widget",
+       NULL,
+       "vbox_contact_widget", &information->vbox_contact_widget,
+       "vbox_contact", &information->vbox_contact,
+       "hbox_presence", &information->hbox_presence,
+       "label_alias", &information->label_alias,
+       "image_state", &information->image_state,
+       "label_status", &information->label_status,
+       "table_contact", &information->table_contact,
+       "vbox_avatar", &information->vbox_avatar,
+       "vbox_groups", &information->vbox_groups,
+       "entry_group", &information->entry_group,
+       "button_group", &information->button_group,
+       "treeview_groups", &information->treeview_groups,
+       "vbox_details", &information->vbox_details,
+       "table_details", &information->table_details,
+       "hbox_details_requested", &information->hbox_details_requested,
+       "vbox_client", &information->vbox_client,
+       "table_client", &information->table_client,
+       "hbox_client_requested", &information->hbox_client_requested,
+       NULL);
+  g_free (filename);
+
+  empathy_glade_connect (glade,
+      information,
+      "vbox_contact_widget", "destroy", contact_widget_destroy_cb,
+      "entry_group", "changed", contact_widget_entry_group_changed_cb,
+      "entry_group", "activate", contact_widget_entry_group_activate_cb,
+      "button_group", "clicked", contact_widget_button_group_clicked_cb,
+      NULL);
+
+  g_object_unref (glade);
+
+  g_object_set_data (G_OBJECT (information->vbox_contact_widget),
+      "EmpathyContactWidget",
+      information);
+
+  /* Create widgets */
+  contact_widget_contact_setup (information);
+  contact_widget_groups_setup (information);
+  contact_widget_details_setup (information);
+  contact_widget_client_setup (information);
+
+  contact_widget_set_contact (information, contact);
+
+  gtk_widget_show (information->vbox_contact_widget);
+
+  return information->vbox_contact_widget;
 }
 
-void
-empathy_contact_widget_save (GtkWidget *widget)
+EmpathyContact *
+empathy_contact_widget_get_contact (GtkWidget *widget)
 {
-       EmpathyContactWidget *information;
-       const gchar          *name;
-       GList                *groups;
+  EmpathyContactWidget *information;
 
-       g_return_if_fail (GTK_IS_WIDGET (widget));
+  g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
 
-       information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
-       if (!information ||
-           !information->editable ||
-           !information->changes_made) {
-               return;
-       }
+  information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
+  if (!information)
+      return NULL;
 
-       name = gtk_entry_get_text (GTK_ENTRY (information->widget_alias));
-       groups = contact_widget_model_find_selected (information);
+  return information->contact;
+}
 
-       gossip_contact_set_name (information->contact, name);
-       gossip_contact_set_groups (information->contact, groups);
+void
+empathy_contact_widget_set_contact (GtkWidget *widget,
+                                    EmpathyContact *contact)
+{
+  EmpathyContactWidget *information;
 
-       g_list_foreach (groups, (GFunc) g_free, NULL);
-       g_list_free (groups);
+  g_return_if_fail (GTK_IS_WIDGET (widget));
+  g_return_if_fail (EMPATHY_IS_CONTACT (contact));
+
+  information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
+  if (!information)
+    return;
+
+  contact_widget_set_contact (information, contact);
 }
 
-GossipContact *
-empathy_contact_widget_get_contact (GtkWidget *widget)
+void
+empathy_contact_widget_set_account_filter (
+    GtkWidget *widget,
+    EmpathyAccountChooserFilterFunc filter,
+    gpointer user_data)
 {
-       EmpathyContactWidget *information;
+  EmpathyContactWidget *information;
+  EmpathyAccountChooser *chooser;
 
-       g_return_val_if_fail (GTK_IS_WIDGET (widget), NULL);
+  g_return_if_fail (GTK_IS_WIDGET (widget));
 
-       information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
-       if (!information) {
-               return NULL;
-       }
+  information = g_object_get_data (G_OBJECT (widget), "EmpathyContactWidget");
+  if (!information)
+    return;
 
-       return information->contact;
+  chooser = EMPATHY_ACCOUNT_CHOOSER (information->widget_account);
+  if (chooser)
+      empathy_account_chooser_set_filter (chooser, filter, user_data);
 }
-       
+  
 static void
-contact_widget_destroy_cb (GtkWidget            *widget,
-                          EmpathyContactWidget *information)
-{
-       g_signal_handlers_disconnect_by_func (information->contact,
-                                             contact_widget_name_notify_cb,
-                                             information);
-       g_signal_handlers_disconnect_by_func (information->contact,
-                                             contact_widget_presence_notify_cb,
-                                             information);
-       g_signal_handlers_disconnect_by_func (information->contact,
-                                             contact_widget_avatar_notify_cb,
-                                             information);
-       g_signal_handlers_disconnect_by_func (information->contact,
-                                             contact_widget_groups_notify_cb,
-                                             information);
-
-       g_object_unref (information->contact);
-       g_slice_free (EmpathyContactWidget, information);
+contact_widget_destroy_cb (GtkWidget *widget,
+                           EmpathyContactWidget *information)
+{
+  contact_widget_remove_contact (information);
+
+  if (information->widget_id_timeout != 0)
+    {
+      g_source_remove (information->widget_id_timeout);
+    }
+  if (information->factory)
+    {
+      g_object_unref (information->factory);
+    }   
+  if (information->manager)
+    {
+      g_object_unref (information->manager);
+    }   
+
+  g_slice_free (EmpathyContactWidget, information);
 }
 
 static void
-contact_widget_contact_setup (EmpathyContactWidget *information)
+contact_widget_remove_contact (EmpathyContactWidget *information)
 {
-       g_signal_connect_swapped (information->contact, "notify::name",
-                                 G_CALLBACK (contact_widget_name_notify_cb),
-                                 information);
-       g_signal_connect_swapped (information->contact, "notify::presence",
-                                 G_CALLBACK (contact_widget_presence_notify_cb),
-                                 information);
-       g_signal_connect_swapped (information->contact, "notify::avatar",
-                                 G_CALLBACK (contact_widget_avatar_notify_cb),
-                                 information);
-
-       /* FIXME: Use GossipAvatarImage if (editable && is_user)  */
-       information->widget_avatar = gtk_image_new ();
-       gtk_box_pack_end (GTK_BOX (information->hbox_contact),
-                         information->widget_avatar,
-                         FALSE, FALSE,
-                         6);
-
-       /* Setup alias entry or label */
-       if (information->editable) {
-               information->widget_alias = gtk_entry_new ();
-               g_signal_connect (information->widget_alias, "changed",
-                                 G_CALLBACK (contact_widget_entry_alias_changed_cb),
-                                 information);
-       } else {
-               information->widget_alias = gtk_label_new (NULL);
-               gtk_label_set_selectable (GTK_LABEL (information->widget_alias), TRUE);
-       }
-       gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
-                                  information->widget_alias,
-                                  1, 2, 1, 2);
-       gtk_widget_show (information->widget_alias);
-
-       /* Setup id label */
-       gtk_label_set_text (GTK_LABEL (information->label_id),
-                           gossip_contact_get_id (information->contact));
-
-       /* Update all widgets */
-       contact_widget_name_notify_cb (information);
-       contact_widget_presence_notify_cb (information);
-       contact_widget_avatar_notify_cb (information);
+  if (information->contact)
+    {
+      g_signal_handlers_disconnect_by_func (information->contact,
+          contact_widget_name_notify_cb, information);
+      g_signal_handlers_disconnect_by_func (information->contact,
+          contact_widget_presence_notify_cb, information);
+      g_signal_handlers_disconnect_by_func (information->contact,
+          contact_widget_avatar_notify_cb, information);
+      g_signal_handlers_disconnect_by_func (information->contact,
+          contact_widget_groups_notify_cb, information);
+
+      g_object_unref (information->contact);
+      information->contact = NULL;
+    }
 }
 
 static void
-contact_widget_name_notify_cb (EmpathyContactWidget *information)
+contact_widget_set_contact (EmpathyContactWidget *information,
+                            EmpathyContact *contact)
 {
-       if (information->editable) {
-               gtk_entry_set_text (GTK_ENTRY (information->widget_alias),
-                                   gossip_contact_get_name (information->contact));
-       } else {
-               gtk_label_set_label (GTK_LABEL (information->widget_alias),
-                                    gossip_contact_get_name (information->contact));
-       }
+  if (contact == information->contact)
+    return;
+
+  contact_widget_remove_contact (information);
+  if (contact)
+      information->contact = g_object_ref (contact);
+
+  /* Update information for widgets */
+  contact_widget_contact_update (information);
+  contact_widget_groups_update (information);
+  contact_widget_details_update (information);
+  contact_widget_client_update (information);
 }
 
-static void
-contact_widget_presence_notify_cb (EmpathyContactWidget *information)
+static gboolean
+contact_widget_id_activate_timeout (EmpathyContactWidget *self)
 {
-       gtk_label_set_text (GTK_LABEL (information->label_status),
-                           gossip_contact_get_status (information->contact));
-       gtk_image_set_from_icon_name (GTK_IMAGE (information->image_state),
-                                     gossip_icon_name_for_contact (information->contact),
-                                     GTK_ICON_SIZE_BUTTON);
-
+  contact_widget_change_contact (self);
+  return FALSE;
 }
 
 static void
-contact_widget_avatar_notify_cb (EmpathyContactWidget *information)
+contact_widget_id_changed_cb (GtkEntry *entry,
+                              EmpathyContactWidget *self)
 {
-       GdkPixbuf *avatar_pixbuf;
-
-       avatar_pixbuf = gossip_pixbuf_avatar_from_contact_scaled (information->contact,
-                                                                 48, 48);
-
-       if (avatar_pixbuf) {
-               gtk_image_set_from_pixbuf (GTK_IMAGE (information->widget_avatar),
-                                          avatar_pixbuf);
-               gtk_widget_show  (information->widget_avatar);
-               g_object_unref (avatar_pixbuf);
-       } else {
-               gtk_widget_hide  (information->widget_avatar);
-       }
+  if (self->widget_id_timeout != 0)
+    {   
+      g_source_remove (self->widget_id_timeout);
+    }
+
+  self->widget_id_timeout =
+    g_timeout_add_seconds (ID_CHANGED_TIMEOUT,
+        (GSourceFunc) contact_widget_id_activate_timeout, self);
 }
 
 static void
-contact_widget_groups_setup (EmpathyContactWidget *information)
+save_avatar_menu_activate_cb (GtkWidget *widget,
+                              EmpathyContactWidget *information)
 {
-       if (information->editable) {
-               contact_widget_model_setup (information);
+  GtkWidget *dialog;
+  EmpathyAvatar *avatar;
+  gchar *ext = NULL, *filename;
 
-               g_signal_connect_swapped (information->contact, "notify::groups",
-                                         G_CALLBACK (contact_widget_groups_notify_cb),
-                                         information);
-               contact_widget_groups_populate_data (information);
+  dialog = gtk_file_chooser_dialog_new (_("Save Avatar"),
+      NULL,
+      GTK_FILE_CHOOSER_ACTION_SAVE,
+      GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
+      GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
+      NULL);
 
-               gtk_widget_show (information->vbox_groups);
-       }
-}
+  gtk_file_chooser_set_do_overwrite_confirmation (GTK_FILE_CHOOSER (dialog),
+      TRUE);
 
-static void
-contact_widget_model_setup (EmpathyContactWidget *information)
-{
-       GtkTreeView      *view;
-       GtkListStore     *store;
-       GtkTreeSelection *selection;
-
-       view = GTK_TREE_VIEW (information->treeview_groups);
+  /* look for the avatar extension */
+  avatar = empathy_contact_get_avatar (information->contact);
+  if (avatar->format != NULL)
+    {
+      gchar **splitted;
 
-       store = gtk_list_store_new (COL_COUNT,
-                                   G_TYPE_STRING,   /* name */
-                                   G_TYPE_BOOLEAN,  /* enabled */
-                                   G_TYPE_BOOLEAN); /* editable */
+      splitted = g_strsplit (avatar->format, "/", 2);
+      if (splitted[0] != NULL && splitted[1] != NULL)
+          ext = g_strdup (splitted[1]);
 
-       gtk_tree_view_set_model (view, GTK_TREE_MODEL (store));
+      g_strfreev (splitted);
+    }
+  else
+    {
+      /* Avatar was loaded from the cache so was converted to PNG */
+      ext = g_strdup ("png");
+    }
 
-       selection = gtk_tree_view_get_selection (view);
-       gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE);
+  if (ext != NULL)
+    {
+      gchar *id;
 
-       contact_widget_model_populate_columns (information);
+      id = tp_escape_as_identifier (empathy_contact_get_id (
+            information->contact));
 
-       gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
-                                             COL_NAME, GTK_SORT_ASCENDING);
+      filename = g_strdup_printf ("%s.%s", id, ext);
+      gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (dialog), filename);
 
-       g_object_unref (store);
-}
+      g_free (id);
+      g_free (ext);
+      g_free (filename);
+    }
 
-static void
-contact_widget_model_populate_columns (EmpathyContactWidget *information)
-{
-       GtkTreeView       *view;
-       GtkTreeModel      *model;
-       GtkTreeViewColumn *column;
-       GtkCellRenderer   *renderer;
-       guint              col_offset;
+  if (gtk_dialog_run (GTK_DIALOG (dialog)) == GTK_RESPONSE_ACCEPT)
+    {
+      GError *error = NULL;
 
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       model = gtk_tree_view_get_model (view);
+      filename = gtk_file_chooser_get_filename (GTK_FILE_CHOOSER (dialog));
 
-       renderer = gtk_cell_renderer_toggle_new ();
-       g_signal_connect (renderer, "toggled",
-                         G_CALLBACK (contact_widget_cell_toggled),
-                         information);
+      if (!empathy_avatar_save_to_file (avatar, filename, &error))
+        {
+          /* Save error */
+          GtkWidget *dialog;
 
-       column = gtk_tree_view_column_new_with_attributes (_("Select"), renderer,
-                                                          "active", COL_ENABLED,
-                                                          NULL);
+          dialog = gtk_message_dialog_new (NULL, 0,
+              GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, 
+              _("Unable to save avatar"));
 
-       gtk_tree_view_column_set_sizing (column, GTK_TREE_VIEW_COLUMN_FIXED);
-       gtk_tree_view_column_set_fixed_width (column, 50);
-       gtk_tree_view_append_column (view, column);
+          gtk_message_dialog_format_secondary_text (
+              GTK_MESSAGE_DIALOG (dialog), "%s", error->message);
 
-       renderer = gtk_cell_renderer_text_new ();
-       col_offset = gtk_tree_view_insert_column_with_attributes (view,
-                                                                 -1, _("Group"),
-                                                                 renderer,
-                                                                 "text", COL_NAME,
-                                                                 /* "editable", COL_EDITABLE, */
-                                                                 NULL);
+          g_signal_connect (dialog, "response",
+              G_CALLBACK (gtk_widget_destroy), NULL);
 
-       g_object_set_data (G_OBJECT (renderer),
-                          "column", GINT_TO_POINTER (COL_NAME));
+          gtk_window_present (GTK_WINDOW (dialog));
 
-       column = gtk_tree_view_get_column (view, col_offset - 1);
-       gtk_tree_view_column_set_sort_column_id (column, COL_NAME);
-       gtk_tree_view_column_set_resizable (column,FALSE);
-       gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (column), TRUE);
+          g_clear_error (&error);
+        }
 
-       if (information->renderer) {
-               g_object_unref (information->renderer);
-       }
+      g_free (filename);
+    }
 
-       information->renderer = g_object_ref (renderer);
+  gtk_widget_destroy (dialog);
 }
 
 static void
-contact_widget_groups_populate_data (EmpathyContactWidget *information)
+popup_avatar_menu (EmpathyContactWidget *information,
+                   GtkWidget *parent,
+                   GdkEventButton *event)
 {
-       EmpathyContactManager *manager;
-       GtkTreeView           *view;
-       GtkListStore          *store;
-       GtkTreeIter            iter;
-       GList                 *groups, *l;
-       GList                 *my_groups = NULL;
-       GList                 *all_groups;
-
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
-
-       manager = empathy_contact_manager_new ();
-       all_groups = empathy_contact_manager_get_groups (manager);
-       groups = gossip_contact_get_groups (information->contact);
-       g_object_unref (manager);
-
-       for (l = groups; l; l = l->next) {
-               const gchar *group_str;
-
-               group_str = l->data;
-               if (strcmp (group_str, _("Unsorted")) == 0) {
-                       continue;
-               }
+  GtkWidget *menu, *item;
+  gint button, event_time;
+
+  if (information->contact == NULL ||
+      empathy_contact_get_avatar (information->contact) == NULL)
+      return;
+
+  menu = gtk_menu_new ();
+
+  /* Add "Save as..." entry */
+  item = gtk_image_menu_item_new_from_stock (GTK_STOCK_SAVE_AS, NULL);
+  gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
+  gtk_widget_show (item);
+
+  g_signal_connect (item, "activate",
+      G_CALLBACK (save_avatar_menu_activate_cb), information);
+
+  if (event)
+    {
+      button = event->button;
+      event_time = event->time;
+    }
+  else
+    {
+      button = 0;
+      event_time = gtk_get_current_event_time ();
+    }
+
+  gtk_menu_attach_to_widget (GTK_MENU (menu), parent, NULL);
+  gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL, NULL, 
+      button, event_time);
+}
 
-               my_groups = g_list_append (my_groups, g_strdup (group_str));
-       }
+static gboolean
+widget_avatar_popup_menu_cb (GtkWidget *widget,
+                             EmpathyContactWidget *information)
+{
+  popup_avatar_menu (information, widget, NULL);
 
-       for (l = all_groups; l; l = l->next) {
-               const gchar *group_str;
-               gboolean     enabled;
+  return TRUE;
+}
 
-               group_str = l->data;
-               if (strcmp (group_str, _("Unsorted")) == 0) {
-                       continue;
-               }
+static gboolean
+widget_avatar_button_press_event_cb (GtkWidget *widget,
+                                     GdkEventButton *event,
+                                     EmpathyContactWidget *information)
+{
+  /* Ignore double-clicks and triple-clicks */
+  if (event->button == 3 && event->type == GDK_BUTTON_PRESS)
+    {
+      popup_avatar_menu (information, widget, event);
+      return TRUE;
+    }
+
+  return FALSE;
+}
 
-               enabled = g_list_find_custom (my_groups,
-                                             group_str,
-                                             (GCompareFunc) strcmp) != NULL;
+static void
+contact_widget_contact_setup (EmpathyContactWidget *information)
+{
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_AVATAR)
+    {
+      information->widget_avatar = empathy_avatar_chooser_new ();
+      g_signal_connect (information->widget_avatar, "changed",
+            G_CALLBACK (contact_widget_avatar_changed_cb),
+            information);
+    }
+  else
+    {
+      information->widget_avatar = empathy_avatar_image_new ();
+
+      g_signal_connect (information->widget_avatar, "popup-menu",
+          G_CALLBACK (widget_avatar_popup_menu_cb), information);
+      g_signal_connect (information->widget_avatar, "button-press-event",
+          G_CALLBACK (widget_avatar_button_press_event_cb), information);
+    }
+  
+  gtk_box_pack_start (GTK_BOX (information->vbox_avatar),
+          information->widget_avatar,
+          FALSE, FALSE,
+          6);
+  gtk_widget_show (information->widget_avatar);
+
+  /* Setup account label/chooser */
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ACCOUNT)
+    {
+      information->widget_account = empathy_account_chooser_new ();
+
+      g_signal_connect (information->widget_account, "changed",
+            G_CALLBACK (contact_widget_account_changed_cb),
+            information);
+    }
+  else
+    {
+      information->widget_account = gtk_label_new (NULL);
+      if (!information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP) {
+        gtk_label_set_selectable (GTK_LABEL (information->widget_account), TRUE);
+      }
+      gtk_misc_set_alignment (GTK_MISC (information->widget_account), 0, 0.5);
+    }
+  gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
+           information->widget_account,
+           1, 2, 0, 1);
+  gtk_widget_show (information->widget_account);
+
+  /* Setup id label/entry */
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
+    {
+      information->widget_id = gtk_entry_new ();
+      g_signal_connect (information->widget_id, "focus-out-event",
+            G_CALLBACK (contact_widget_id_focus_out_cb),
+            information);
+      g_signal_connect (information->widget_id, "changed",
+            G_CALLBACK (contact_widget_id_changed_cb),
+            information);
+    }
+  else
+    {
+      information->widget_id = gtk_label_new (NULL);
+      if (!information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP) {
+        gtk_label_set_selectable (GTK_LABEL (information->widget_id), TRUE);
+      }
+      gtk_misc_set_alignment (GTK_MISC (information->widget_id), 0, 0.5);
+    }
+  gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
+           information->widget_id,
+           1, 2, 1, 2);
+  gtk_widget_show (information->widget_id);
+
+  /* Setup alias label/entry */
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ALIAS)
+    {
+      information->widget_alias = gtk_entry_new ();
+      g_signal_connect (information->widget_alias, "focus-out-event",
+            G_CALLBACK (contact_widget_entry_alias_focus_event_cb),
+            information);
+      /* Make return activate the window default (the Close button) */
+      gtk_entry_set_activates_default (GTK_ENTRY (information->widget_alias),
+          TRUE);
+    }
+  else
+    {
+      information->widget_alias = gtk_label_new (NULL);
+      if (!information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP) {
+        gtk_label_set_selectable (GTK_LABEL (information->widget_alias), TRUE);
+      }
+      gtk_misc_set_alignment (GTK_MISC (information->widget_alias), 0, 0.5);
+    }
+  gtk_table_attach_defaults (GTK_TABLE (information->table_contact),
+           information->widget_alias,
+           1, 2, 2, 3);
+  if (information->flags & EMPATHY_CONTACT_WIDGET_FOR_TOOLTIP) {
+    gtk_label_set_selectable (GTK_LABEL (information->label_status), FALSE);
+  }
+  gtk_widget_show (information->widget_alias);
+}
 
-               gtk_list_store_append (store, &iter);
-               gtk_list_store_set (store, &iter,
-                                   COL_NAME, group_str,
-                                   COL_EDITABLE, TRUE,
-                                   COL_ENABLED, enabled,
-                                   -1);
-       }
+static void
+contact_widget_contact_update (EmpathyContactWidget *information)
+{
+  McAccount *account = NULL;
+  const gchar *id = NULL;
+
+  /* Connect and get info from new contact */
+  if (information->contact)
+    {
+      g_signal_connect_swapped (information->contact, "notify::name",
+          G_CALLBACK (contact_widget_name_notify_cb), information);
+      g_signal_connect_swapped (information->contact, "notify::presence",
+          G_CALLBACK (contact_widget_presence_notify_cb), information);
+      g_signal_connect_swapped (information->contact,
+          "notify::presence-message",
+          G_CALLBACK (contact_widget_presence_notify_cb), information);
+      g_signal_connect_swapped (information->contact, "notify::avatar",
+          G_CALLBACK (contact_widget_avatar_notify_cb), information);
+
+      account = empathy_contact_get_account (information->contact);
+      id = empathy_contact_get_id (information->contact);
+    }
+
+  /* Update account widget */
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ACCOUNT)
+    {
+      if (account)
+        {
+          g_signal_handlers_block_by_func (information->widget_account,
+                   contact_widget_account_changed_cb,
+                   information);
+          empathy_account_chooser_set_account (
+              EMPATHY_ACCOUNT_CHOOSER (information->widget_account), account);
+          g_signal_handlers_unblock_by_func (information->widget_account,
+              contact_widget_account_changed_cb, information);
+        }
+    }
+  else
+    {
+      if (account)
+        {
+          const gchar *name;
+
+          name = mc_account_get_display_name (account);
+          gtk_label_set_label (GTK_LABEL (information->widget_account), name);
+        }
+    }
+
+  /* Update id widget */
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
+      gtk_entry_set_text (GTK_ENTRY (information->widget_id), id ? id : "");
+  else
+      gtk_label_set_label (GTK_LABEL (information->widget_id), id ? id : "");
+
+  /* Update other widgets */
+  if (information->contact)
+    {
+      contact_widget_name_notify_cb (information);
+      contact_widget_presence_notify_cb (information);
+      contact_widget_avatar_notify_cb (information);
+
+      gtk_widget_show (information->label_alias);
+      gtk_widget_show (information->widget_alias);
+      gtk_widget_show (information->hbox_presence);
+      gtk_widget_show (information->widget_avatar);
+    }
+  else
+    {
+      gtk_widget_hide (information->label_alias);
+      gtk_widget_hide (information->widget_alias);
+      gtk_widget_hide (information->hbox_presence);
+      gtk_widget_hide (information->widget_avatar);
+    }
+}
 
-       g_list_foreach (my_groups, (GFunc) g_free, NULL);
-       g_list_free (my_groups);
+static void
+contact_widget_change_contact (EmpathyContactWidget *information)
+{
+  EmpathyContact *contact;
+  McAccount *account;
+
+  account = empathy_account_chooser_get_account (
+      EMPATHY_ACCOUNT_CHOOSER (information->widget_account));
+  if (!account)
+      return;
+
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_ID)
+    {
+      const gchar *id;
+
+      id = gtk_entry_get_text (GTK_ENTRY (information->widget_id));
+      if (G_STR_EMPTY (id))
+          return;
+
+      contact = empathy_contact_factory_get_from_id (information->factory,
+          account, id);
+    }
+  else
+    {
+      contact = empathy_contact_factory_get_user (information->factory,
+          account);
+    }
+
+  if (contact)
+    {
+      empathy_contact_run_until_ready (contact,
+          EMPATHY_CONTACT_READY_HANDLE |
+          EMPATHY_CONTACT_READY_ID,
+          NULL);
+      contact_widget_set_contact (information, contact);
+      g_object_unref (contact);
+    }
+}
 
-       g_list_free (all_groups);
+static void
+contact_widget_avatar_changed_cb (EmpathyAvatarChooser *chooser,
+                                  EmpathyContactWidget *information)
+{
+  if (information->contact && empathy_contact_is_user (information->contact))
+    {
+      McAccount *account;
+      const gchar *data;
+      gsize size;
+      const gchar *mime_type;
+
+      account = empathy_contact_get_account (information->contact);
+      empathy_avatar_chooser_get_image_data (
+          EMPATHY_AVATAR_CHOOSER (information->widget_avatar),
+          &data, &size, &mime_type);
+      empathy_contact_factory_set_avatar (information->factory, account,
+          data, size, mime_type);
+    }
 }
 
 static void
-contact_widget_groups_notify_cb (EmpathyContactWidget *information)
+contact_widget_account_changed_cb (GtkComboBox *widget,
+                                   EmpathyContactWidget *information)
 {
-       /* FIXME: not implemented */
+  contact_widget_change_contact (information);
 }
 
 static gboolean
-contact_widget_model_find_name (EmpathyContactWidget *information,
-                               const gchar          *name,
-                               GtkTreeIter          *iter)
+contact_widget_id_focus_out_cb (GtkWidget *widget,
+                                GdkEventFocus *event,
+                                EmpathyContactWidget *information)
 {
-       GtkTreeView  *view;
-       GtkTreeModel *model;
-       FindName      data;
+  contact_widget_change_contact (information);
+  return FALSE;
+}
 
-       if (G_STR_EMPTY (name)) {
-               return FALSE;
-       }
+static gboolean
+contact_widget_entry_alias_focus_event_cb (GtkEditable *editable,
+                                           GdkEventFocus *event,
+                                           EmpathyContactWidget *information)
+{
+  if (information->contact)
+    {
+      const gchar *alias;
 
-       data.information = information;
-       data.name = name;
-       data.found = FALSE;
+      alias = gtk_entry_get_text (GTK_ENTRY (editable));
+      empathy_contact_factory_set_alias (information->factory,
+          information->contact, alias);
+    }
 
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       model = gtk_tree_view_get_model (view);
+  return FALSE;
+}
 
-       gtk_tree_model_foreach (model,
-                               (GtkTreeModelForeachFunc) contact_widget_model_find_name_foreach,
-                               &data);
+static void
+contact_widget_name_notify_cb (EmpathyContactWidget *information)
+{
+  if (GTK_IS_ENTRY (information->widget_alias))
+      gtk_entry_set_text (GTK_ENTRY (information->widget_alias),
+          empathy_contact_get_name (information->contact));
+  else
+      gtk_label_set_label (GTK_LABEL (information->widget_alias),
+          empathy_contact_get_name (information->contact));
+}
 
-       if (data.found == TRUE) {
-               *iter = data.found_iter;
-               return TRUE;
-       }
+static void
+contact_widget_presence_notify_cb (EmpathyContactWidget *information)
+{
+  gtk_label_set_text (GTK_LABEL (information->label_status),
+      empathy_contact_get_status (information->contact));
+  gtk_image_set_from_icon_name (GTK_IMAGE (information->image_state),
+      empathy_icon_name_for_contact (information->contact),
+      GTK_ICON_SIZE_BUTTON);
+}
 
-       return FALSE;
+static void
+contact_widget_avatar_notify_cb (EmpathyContactWidget *information)
+{
+  EmpathyAvatar *avatar = NULL;
+
+  if (information->contact)
+      avatar = empathy_contact_get_avatar (information->contact);
+
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_AVATAR)
+    {
+      g_signal_handlers_block_by_func (information->widget_avatar,
+          contact_widget_avatar_changed_cb,
+          information);
+      empathy_avatar_chooser_set (
+          EMPATHY_AVATAR_CHOOSER (information->widget_avatar), avatar);
+      g_signal_handlers_unblock_by_func (information->widget_avatar,
+          contact_widget_avatar_changed_cb, information);
+    }
+  else
+      empathy_avatar_image_set (
+          EMPATHY_AVATAR_IMAGE (information->widget_avatar), avatar);
 }
 
-static gboolean
-contact_widget_model_find_name_foreach (GtkTreeModel *model,
-                                       GtkTreePath  *path,
-                                       GtkTreeIter  *iter,
-                                       FindName     *data)
+static void
+contact_widget_groups_setup (EmpathyContactWidget *information)
 {
-       gchar *name;
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_GROUPS)
+    {
+      information->manager = empathy_contact_manager_new ();
+      contact_widget_model_setup (information);
+    }
+}
 
-       gtk_tree_model_get (model, iter,
-                           COL_NAME, &name,
-                           -1);
+static void
+contact_widget_groups_update (EmpathyContactWidget *information)
+{
+  if (information->flags & EMPATHY_CONTACT_WIDGET_EDIT_GROUPS &&
+      information->contact)
+    {
+      g_signal_connect_swapped (information->contact, "notify::groups",
+          G_CALLBACK (contact_widget_groups_notify_cb), information);
+      contact_widget_groups_populate_data (information);
+
+      gtk_widget_show (information->vbox_groups);
+    }
+  else
+      gtk_widget_hide (information->vbox_groups);
+}
 
-       if (!name) {
-               return FALSE;
-       }
+static void
+contact_widget_model_setup (EmpathyContactWidget *information)
+{
+  GtkTreeView *view;
+  GtkListStore *store;
+  GtkTreeSelection *selection;
 
-       if (data->name && strcmp (data->name, name) == 0) {
-               data->found = TRUE;
-               data->found_iter = *iter;
+  view = GTK_TREE_VIEW (information->treeview_groups);
 
-               g_free (name);
+  store = gtk_list_store_new (COL_COUNT,
+      G_TYPE_STRING,   /* name */
+      G_TYPE_BOOLEAN,  /* enabled */
+      G_TYPE_BOOLEAN); /* editable */
 
-               return TRUE;
-       }
+  gtk_tree_view_set_model (view, GTK_TREE_MODEL (store));
 
-       g_free (name);
+  selection = gtk_tree_view_get_selection (view);
+  gtk_tree_selection_set_mode (selection, GTK_SELECTION_SINGLE);
 
-       return FALSE;
-}
+  contact_widget_model_populate_columns (information);
 
-static GList *
-contact_widget_model_find_selected (EmpathyContactWidget *information)
-{
-       GtkTreeView  *view;
-       GtkTreeModel *model;
-       FindSelected  data;
+  gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
+      COL_NAME, GTK_SORT_ASCENDING);
 
-       data.information = information;
-       data.list = NULL;
+  g_object_unref (store);
+}
 
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       model = gtk_tree_view_get_model (view);
+static void
+contact_widget_model_populate_columns (EmpathyContactWidget *information)
+{
+  GtkTreeView *view;
+  GtkTreeModel *model;
+  GtkTreeViewColumn *column;
+  GtkCellRenderer  *renderer;
+  guint col_offset;
+
+  view = GTK_TREE_VIEW (information->treeview_groups);
+  model = gtk_tree_view_get_model (view);
+
+  renderer = gtk_cell_renderer_toggle_new ();
+  g_signal_connect (renderer, "toggled",
+      G_CALLBACK (contact_widget_cell_toggled), information);
+
+  column = gtk_tree_view_column_new_with_attributes (_("Select"), renderer,
+      "active", COL_ENABLED, NULL);
+
+  gtk_tree_view_column_set_sizing (column, GTK_TREE_VIEW_COLUMN_FIXED);
+  gtk_tree_view_column_set_fixed_width (column, 50);
+  gtk_tree_view_append_column (view, column);
+
+  renderer = gtk_cell_renderer_text_new ();
+  col_offset = gtk_tree_view_insert_column_with_attributes (view,
+      -1, _("Group"),
+      renderer,
+      "text", COL_NAME,
+      /* "editable", COL_EDITABLE, */
+      NULL);
+
+  g_object_set_data (G_OBJECT (renderer),
+      "column", GINT_TO_POINTER (COL_NAME));
+
+  column = gtk_tree_view_get_column (view, col_offset - 1);
+  gtk_tree_view_column_set_sort_column_id (column, COL_NAME);
+  gtk_tree_view_column_set_resizable (column,FALSE);
+  gtk_tree_view_column_set_clickable (GTK_TREE_VIEW_COLUMN (column), TRUE);
+
+  if (information->renderer)
+      g_object_unref (information->renderer);
+
+  information->renderer = g_object_ref (renderer);
+}
 
-       gtk_tree_model_foreach (model,
-                               (GtkTreeModelForeachFunc) contact_widget_model_find_selected_foreach,
-                               &data);
+static void
+contact_widget_groups_populate_data (EmpathyContactWidget *information)
+{
+  GtkTreeView *view;
+  GtkListStore *store;
+  GtkTreeIter iter;
+  GList *my_groups, *l;
+  GList *all_groups;
+
+  view = GTK_TREE_VIEW (information->treeview_groups);
+  store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
+  gtk_list_store_clear (store);
+
+  all_groups = empathy_contact_list_get_all_groups (
+      EMPATHY_CONTACT_LIST (information->manager));
+  my_groups = empathy_contact_list_get_groups (
+      EMPATHY_CONTACT_LIST (information->manager),
+      information->contact);
+
+  for (l = all_groups; l; l = l->next)
+    {
+      const gchar *group_str;
+      gboolean enabled;
+
+      group_str = l->data;
+
+      enabled = g_list_find_custom (my_groups,
+          group_str, (GCompareFunc) strcmp) != NULL;
+
+      gtk_list_store_append (store, &iter);
+      gtk_list_store_set (store, &iter,
+          COL_NAME, group_str,
+          COL_EDITABLE, TRUE,
+          COL_ENABLED, enabled,
+          -1);
+    }
+
+  g_list_foreach (all_groups, (GFunc) g_free, NULL);
+  g_list_foreach (my_groups, (GFunc) g_free, NULL);
+  g_list_free (all_groups);
+  g_list_free (my_groups);
+}
 
-       return data.list;
+static void
+contact_widget_groups_notify_cb (EmpathyContactWidget *information)
+{
+  /* FIXME: not implemented */
 }
 
 static gboolean
-contact_widget_model_find_selected_foreach (GtkTreeModel *model,
-                                           GtkTreePath  *path,
-                                           GtkTreeIter  *iter,
-                                           FindSelected *data)
+contact_widget_model_find_name (EmpathyContactWidget *information,
+                                const gchar *name,
+                                GtkTreeIter *iter)
 {
-       gchar    *name;
-       gboolean  selected;
+  GtkTreeView *view;
+  GtkTreeModel *model;
+  FindName data;
+
+  if (G_STR_EMPTY (name))
+      return FALSE;
 
-       gtk_tree_model_get (model, iter,
-                           COL_NAME, &name,
-                           COL_ENABLED, &selected,
-                           -1);
+  data.information = information;
+  data.name = name;
+  data.found = FALSE;
 
-       if (!name) {
-               return FALSE;
-       }
+  view = GTK_TREE_VIEW (information->treeview_groups);
+  model = gtk_tree_view_get_model (view);
 
-       if (selected) {
-               data->list = g_list_append (data->list, name);
-               return FALSE;
-       }
+  gtk_tree_model_foreach (model,
+      (GtkTreeModelForeachFunc) contact_widget_model_find_name_foreach,
+      &data);
 
-       g_free (name);
+  if (data.found == TRUE)
+    {
+      *iter = data.found_iter;
+      return TRUE;
+    }
 
-       return FALSE;
+  return FALSE;
 }
 
-static void
-contact_widget_cell_toggled (GtkCellRendererToggle *cell,
-                            gchar                 *path_string,
-                            EmpathyContactWidget  *information)
+static gboolean
+contact_widget_model_find_name_foreach (GtkTreeModel *model,
+                                        GtkTreePath *path,
+                                        GtkTreeIter *iter,
+                                        FindName *data)
 {
-       GtkTreeView  *view;
-       GtkTreeModel *model;
-       GtkListStore *store;
-       GtkTreePath  *path;
-       GtkTreeIter   iter;
-       gboolean      enabled;
+  gchar *name;
 
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       model = gtk_tree_view_get_model (view);
-       store = GTK_LIST_STORE (model);
+  gtk_tree_model_get (model, iter,
+      COL_NAME, &name,
+      -1);
 
-       path = gtk_tree_path_new_from_string (path_string);
+  if (!name)
+      return FALSE;
 
-       gtk_tree_model_get_iter (model, &iter, path);
-       gtk_tree_model_get (model, &iter, COL_ENABLED, &enabled, -1);
+  if (data->name && strcmp (data->name, name) == 0)
+    {
+      data->found = TRUE;
+      data->found_iter = *iter;
 
-       enabled ^= 1;
+      g_free (name);
 
-       gtk_list_store_set (store, &iter, COL_ENABLED, enabled, -1);
-       gtk_tree_path_free (path);
+      return TRUE;
+    }
 
-       information->changes_made = TRUE;
+  g_free (name);
+
+  return FALSE;
 }
 
 static void
-contact_widget_entry_alias_changed_cb (GtkEditable           *editable,
-                                      EmpathyContactWidget  *information)
+contact_widget_cell_toggled (GtkCellRendererToggle *cell,
+                             gchar *path_string,
+                             EmpathyContactWidget *information)
 {
-       information->changes_made = TRUE;
+  GtkTreeView *view;
+  GtkTreeModel *model;
+  GtkListStore *store;
+  GtkTreePath *path;
+  GtkTreeIter iter;
+  gboolean enabled;
+  gchar *group;
+
+  view = GTK_TREE_VIEW (information->treeview_groups);
+  model = gtk_tree_view_get_model (view);
+  store = GTK_LIST_STORE (model);
+
+  path = gtk_tree_path_new_from_string (path_string);
+
+  gtk_tree_model_get_iter (model, &iter, path);
+  gtk_tree_model_get (model, &iter,
+      COL_ENABLED, &enabled,
+      COL_NAME, &group,
+      -1);
+
+  gtk_list_store_set (store, &iter, COL_ENABLED, !enabled, -1);
+  gtk_tree_path_free (path);
+
+  if (group)
+    {
+      if (enabled)
+        {
+          empathy_contact_list_remove_from_group (
+              EMPATHY_CONTACT_LIST (information->manager), information->contact,
+              group);
+        }
+      else
+        {
+          empathy_contact_list_add_to_group (
+              EMPATHY_CONTACT_LIST (information->manager), information->contact,
+              group);
+        }
+      g_free (group);
+    }
 }
 
 static void
-contact_widget_entry_group_changed_cb (GtkEditable           *editable,
-                                      EmpathyContactWidget  *information)
+contact_widget_entry_group_changed_cb (GtkEditable *editable,
+                                       EmpathyContactWidget *information)
 {
-       GtkTreeIter  iter;
-       const gchar *group;
-
-       group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
+  GtkTreeIter iter;
+  const gchar *group;
 
-       if (contact_widget_model_find_name (information, group, &iter)) {
-               gtk_widget_set_sensitive (GTK_WIDGET (information->button_group), FALSE);
+  group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
 
-       } else {
-               gtk_widget_set_sensitive (GTK_WIDGET (information->button_group),
-                                         !G_STR_EMPTY (group));
-       }
+  if (contact_widget_model_find_name (information, group, &iter))
+      gtk_widget_set_sensitive (GTK_WIDGET (information->button_group), FALSE);
+  else
+      gtk_widget_set_sensitive (GTK_WIDGET (information->button_group),
+          !G_STR_EMPTY (group));
 }
 
 static void
-contact_widget_entry_group_activate_cb (GtkEntry              *entry,
-                                       EmpathyContactWidget  *information)
+contact_widget_entry_group_activate_cb (GtkEntry *entry,
+                                        EmpathyContactWidget  *information)
 {
-       gtk_widget_activate (GTK_WIDGET (information->button_group));
+  gtk_widget_activate (GTK_WIDGET (information->button_group));
 }
 
 static void
-contact_widget_button_group_clicked_cb (GtkButton             *button,
-                                       EmpathyContactWidget  *information)
+contact_widget_button_group_clicked_cb (GtkButton *button,
+                                        EmpathyContactWidget *information)
 {
-       GtkTreeView  *view;
-       GtkListStore *store;
-       GtkTreeIter   iter;
-       const gchar  *group;
+  GtkTreeView *view;
+  GtkListStore *store;
+  GtkTreeIter iter;
+  const gchar *group;
 
-       view = GTK_TREE_VIEW (information->treeview_groups);
-       store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
+  view = GTK_TREE_VIEW (information->treeview_groups);
+  store = GTK_LIST_STORE (gtk_tree_view_get_model (view));
 
-       group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
+  group = gtk_entry_get_text (GTK_ENTRY (information->entry_group));
 
-       gtk_list_store_append (store, &iter);
-       gtk_list_store_set (store, &iter,
-                           COL_NAME, group,
-                           COL_ENABLED, TRUE,
-                           -1);
+  gtk_list_store_append (store, &iter);
+  gtk_list_store_set (store, &iter,
+      COL_NAME, group,
+      COL_ENABLED, TRUE,
+      -1);
 
-       information->changes_made = TRUE;
+  empathy_contact_list_add_to_group (
+      EMPATHY_CONTACT_LIST (information->manager), information->contact,
+      group);
 }
 
 static void
 contact_widget_details_setup (EmpathyContactWidget *information)
 {
-       /* FIXME: Needs new telepathy spec */
+  /* FIXME: Needs new telepathy spec */
+  gtk_widget_hide (information->vbox_details);
+}
+
+static void
+contact_widget_details_update (EmpathyContactWidget *information)
+{
+  /* FIXME: Needs new telepathy spec */
 }
 
 static void
 contact_widget_client_setup (EmpathyContactWidget *information)
 {
-       /* FIXME: Needs new telepathy spec */
+  /* FIXME: Needs new telepathy spec */
+  gtk_widget_hide (information->vbox_client);
 }
 
+static void
+contact_widget_client_update (EmpathyContactWidget *information)
+{
+  /* FIXME: Needs new telepathy spec */
+}