]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-contact-selector-dialog.c
Merge branch 'sasl'
[empathy.git] / libempathy-gtk / empathy-contact-selector-dialog.c
index c52ac6c56f6b0f5fcd01bc55365d4e59ee237e92..3ed7f30e523b3dac2a82f9a1693e8896d7cb6ed2 100644 (file)
@@ -17,6 +17,7 @@
  *
  * Authors: Xavier Claessens <xclaesse@gmail.com>
  * Authors: Guillaume Desmottes <guillaume.desmottes@collabora.co.uk>
+ * Authors: Danielle Madeley <danielle.madeley@collabora.co.uk>
  */
 
 #include <config.h>
 #include "empathy-contact-selector-dialog.h"
 #include "empathy-account-chooser.h"
 
-G_DEFINE_TYPE(EmpathyContactSelectorDialog, empathy_contact_selector_dialog,
+G_DEFINE_ABSTRACT_TYPE (EmpathyContactSelectorDialog,
+    empathy_contact_selector_dialog,
     GTK_TYPE_DIALOG)
 
 typedef struct _EmpathyContactSelectorDialogPriv \
           EmpathyContactSelectorDialogPriv;
 
 struct _EmpathyContactSelectorDialogPriv {
+  GtkListStore *store;
+  GtkWidget *account_chooser_label;
   GtkWidget *account_chooser;
   GtkWidget *entry_id;
   EmpathyContactManager *contact_manager;
+  TpAccount *filter_account;
+
+  gboolean show_account_chooser;
 };
 
 #define GET_PRIV(o) \
   (G_TYPE_INSTANCE_GET_PRIVATE ((o), EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG, \
     EmpathyContactSelectorDialogPriv))
 
+enum {
+  PROP_0,
+  PROP_SHOW_ACCOUNT_CHOOSER,
+  PROP_FILTER_ACCOUNT
+};
+
 enum {
   COMPLETION_COL_TEXT,
   COMPLETION_COL_ID,
@@ -70,17 +83,10 @@ contact_selector_dialog_account_changed_cb (GtkWidget *widget,
   EmpathyContactSelectorDialogPriv *priv = GET_PRIV (dialog);
   EmpathyAccountChooser *chooser;
   TpConnection *connection;
-  EmpathyTpContactList *contact_list;
   GList *members;
-  GtkListStore *store;
-  GtkEntryCompletion *completion;
-  GtkTreeIter iter;
-  gchar *tmpstr;
 
   /* Remove completions */
-  completion = gtk_entry_get_completion (GTK_ENTRY (priv->entry_id));
-  store = GTK_LIST_STORE (gtk_entry_completion_get_model (completion));
-  gtk_list_store_clear (store);
+  gtk_list_store_clear (priv->store);
 
   /* Get members of the new account */
   chooser = EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser);
@@ -88,28 +94,57 @@ contact_selector_dialog_account_changed_cb (GtkWidget *widget,
   if (!connection)
     return;
 
-  contact_list = empathy_contact_manager_get_list (priv->contact_manager,
-               connection);
-  members = empathy_contact_list_get_members (
-      EMPATHY_CONTACT_LIST (contact_list));
+  if (priv->show_account_chooser)
+    {
+      EmpathyTpContactList *contact_list;
+
+      contact_list = empathy_contact_manager_get_list (priv->contact_manager,
+                   connection);
+      members = empathy_contact_list_get_members (
+          EMPATHY_CONTACT_LIST (contact_list));
+    }
+  else
+    {
+      if (priv->filter_account != NULL)
+        {
+          EmpathyTpContactList *contact_list;
+
+          connection = tp_account_get_connection (priv->filter_account);
+          if (connection == NULL)
+            return;
+
+          contact_list = empathy_contact_manager_get_list (
+              priv->contact_manager, connection);
+
+          members = empathy_contact_list_get_members (
+              EMPATHY_CONTACT_LIST (contact_list));
+        }
+      else
+        {
+          members = empathy_contact_list_get_members (
+              EMPATHY_CONTACT_LIST (priv->contact_manager));
+        }
+    }
 
   /* Add members to the completion */
   while (members)
     {
       EmpathyContact *contact = members->data;
+      GtkTreeIter iter;
+      gchar *tmpstr;
 
       DEBUG ("Adding contact ID %s, Name %s",
              empathy_contact_get_id (contact),
-             empathy_contact_get_name (contact));
+             empathy_contact_get_alias (contact));
 
       tmpstr = g_strdup_printf ("%s (%s)",
-        empathy_contact_get_name (contact),
+        empathy_contact_get_alias (contact),
         empathy_contact_get_id (contact));
 
-      gtk_list_store_insert_with_values (store, &iter, -1,
+      gtk_list_store_insert_with_values (priv->store, &iter, -1,
         COMPLETION_COL_TEXT, tmpstr,
         COMPLETION_COL_ID, empathy_contact_get_id (contact),
-        COMPLETION_COL_NAME, empathy_contact_get_name (contact),
+        COMPLETION_COL_NAME, empathy_contact_get_alias (contact),
         -1);
 
       g_free (tmpstr);
@@ -148,60 +183,38 @@ contact_selector_dialog_match_func (GtkEntryCompletion *completion,
     gpointer user_data)
 {
   GtkTreeModel *model;
-  gchar *id;
-  gchar *name;
+  gchar *str, *lower;
+  gboolean v = FALSE;
 
   model = gtk_entry_completion_get_model (completion);
   if (!model || !iter)
     return FALSE;
 
-  gtk_tree_model_get (model, iter, COMPLETION_COL_NAME, &name, -1);
-  if (strstr (name, key))
+  gtk_tree_model_get (model, iter, COMPLETION_COL_NAME, &str, -1);
+  lower = g_utf8_strdown (str, -1);
+  if (strstr (lower, key))
     {
-      DEBUG ("Key %s is matching name **%s**", key, name);
-      g_free (name);
-      return TRUE;
+      DEBUG ("Key %s is matching name **%s**", key, str);
+      v = TRUE;
+      goto out;
     }
-  g_free (name);
+  g_free (str);
+  g_free (lower);
 
-  gtk_tree_model_get (model, iter, COMPLETION_COL_ID, &id, -1);
-  if (strstr (id, key))
+  gtk_tree_model_get (model, iter, COMPLETION_COL_ID, &str, -1);
+  lower = g_utf8_strdown (str, -1);
+  if (strstr (lower, key))
     {
-      DEBUG ("Key %s is matching ID **%s**", key, id);
-      g_free (id);
-      return TRUE;
+      DEBUG ("Key %s is matching ID **%s**", key, str);
+      v = TRUE;
+      goto out;
     }
-  g_free (id);
 
-  return FALSE;
-}
-
-static void
-contact_selector_dialog_response_cb (GtkWidget *widget,
-    gint response,
-    EmpathyContactSelectorDialog *dialog)
-{
-  EmpathyContactSelectorDialogPriv *priv = GET_PRIV (dialog);
-  TpConnection *connection;
-  const gchar *id;
-  EmpathyContactSelectorDialogClass *class = \
-    EMPATHY_CONTACT_SELECTOR_DIALOG_GET_CLASS (dialog);
-
-  connection = empathy_account_chooser_get_connection (
-    EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser));
-  id = gtk_entry_get_text (GTK_ENTRY (priv->entry_id));
-  if (!connection || EMP_STR_EMPTY (id))
-    {
-      gtk_widget_destroy (widget);
-      return;
-    }
-
-  if (response == GTK_RESPONSE_ACCEPT)
-    {
-      class->got_response (dialog, connection, id);
-    }
+out:
+  g_free (str);
+  g_free (lower);
 
-  gtk_widget_destroy (widget);
+  return v;
 }
 
 static void
@@ -230,6 +243,48 @@ entry_activate_cb (GtkEntry *entry,
   gtk_dialog_response (GTK_DIALOG (self), GTK_RESPONSE_ACCEPT);
 }
 
+static void
+account_chooser_filter (TpAccount *account,
+    EmpathyAccountChooserFilterResultCallback callback,
+    gpointer callback_data,
+    gpointer user_data)
+{
+  EmpathyContactSelectorDialog *self = user_data;
+  EmpathyContactSelectorDialogClass *class = \
+      EMPATHY_CONTACT_SELECTOR_DIALOG_GET_CLASS (self);
+
+  if (class->account_filter == NULL)
+    {
+      empathy_account_chooser_filter_is_connected (
+          account,callback, callback_data, user_data);
+      return;
+    }
+
+  class->account_filter (self, callback, callback_data, account);
+}
+
+static gboolean
+contact_selector_dialog_filter_visible (GtkTreeModel *model,
+                                        GtkTreeIter  *iter,
+                                        gpointer      data)
+{
+  EmpathyContactSelectorDialog *self = EMPATHY_CONTACT_SELECTOR_DIALOG (data);
+  gboolean r;
+  char *id;
+
+  gtk_tree_model_get (model, iter,
+      COMPLETION_COL_ID, &id,
+      -1);
+
+  /* this must be non-NULL for this function to get called */
+  r = EMPATHY_CONTACT_SELECTOR_DIALOG_GET_CLASS (self)->contact_filter (
+      self, id);
+
+  g_free (id);
+
+  return r;
+}
+
 static void
 empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
 {
@@ -237,8 +292,10 @@ empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
   GtkBuilder *gui;
   gchar *filename;
   GtkEntryCompletion *completion;
-  GtkListStore *model;
   GtkWidget *content_area;
+  GtkWidget *table_contact;
+
+  dialog->vbox = gtk_vbox_new (FALSE, 3);
 
   /* create a contact manager */
   priv->contact_manager = empathy_contact_manager_dup_singleton ();
@@ -246,7 +303,8 @@ empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
   filename = empathy_file_lookup ("empathy-contact-selector-dialog.ui",
           "libempathy-gtk");
   gui = empathy_builder_get_file (filename,
-                "table_contact", &dialog->table_contact,
+                "table_contact", &table_contact,
+                "account_chooser_label", &priv->account_chooser_label,
                 "entry_id", &priv->entry_id,
                 NULL);
   g_free (filename);
@@ -256,35 +314,36 @@ empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
       NULL);
 
   content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
-  gtk_container_add (GTK_CONTAINER (content_area), dialog->table_contact);
+  gtk_container_add (GTK_CONTAINER (content_area), dialog->vbox);
+  gtk_box_pack_start (GTK_BOX (dialog->vbox), table_contact, TRUE, TRUE, 0);
+  gtk_widget_show (dialog->vbox);
 
   gtk_dialog_add_button (GTK_DIALOG (dialog),
     GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL);
 
   /* Tweak the dialog */
-  gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE);
-
   gtk_window_set_resizable (GTK_WINDOW (dialog), FALSE);
   gtk_window_set_position (GTK_WINDOW (dialog), GTK_WIN_POS_CENTER_ON_PARENT);
   gtk_window_set_type_hint (GTK_WINDOW (dialog), GDK_WINDOW_TYPE_HINT_DIALOG);
 
+  gtk_container_set_border_width (GTK_CONTAINER (dialog->vbox), 6);
+  gtk_container_set_border_width (GTK_CONTAINER (dialog), 6);
+
   /* text completion */
+  priv->store = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
+
   completion = gtk_entry_completion_new ();
-  model = gtk_list_store_new (3, G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING);
   gtk_entry_completion_set_text_column (completion, COMPLETION_COL_TEXT);
   gtk_entry_completion_set_match_func (completion,
                contact_selector_dialog_match_func,
                NULL, NULL);
-  gtk_entry_completion_set_model (completion, GTK_TREE_MODEL (model));
+  gtk_entry_completion_set_model (completion, GTK_TREE_MODEL (priv->store));
   gtk_entry_set_completion (GTK_ENTRY (priv->entry_id), completion);
   g_signal_connect (completion, "match-selected",
         G_CALLBACK (contact_selector_dialog_match_selected_cb),
         dialog);
   g_object_unref (completion);
-  g_object_unref (model);
-
-  g_signal_connect (dialog, "response",
-        G_CALLBACK (contact_selector_dialog_response_cb), dialog);
+  g_object_unref (priv->store);
 
   empathy_builder_connect (gui, dialog,
              "entry_id", "changed", contact_selector_change_state_button_cb,
@@ -293,14 +352,15 @@ empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
   g_object_unref (gui);
 
   /* Create account chooser */
+  priv->show_account_chooser = TRUE;
   priv->account_chooser = empathy_account_chooser_new ();
-  gtk_table_attach_defaults (GTK_TABLE (dialog->table_contact),
+  gtk_table_attach_defaults (GTK_TABLE (table_contact),
            priv->account_chooser,
            1, 2, 0, 1);
   empathy_account_chooser_set_filter (
       EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser),
-      empathy_account_chooser_filter_is_connected,
-      NULL);
+      account_chooser_filter,
+      dialog);
   gtk_widget_show (priv->account_chooser);
 
   contact_selector_dialog_account_changed_cb (priv->account_chooser, dialog);
@@ -309,6 +369,79 @@ empathy_contact_selector_dialog_init (EmpathyContactSelectorDialog *dialog)
         dialog);
 }
 
+static void
+empathy_contact_selector_dialog_get_property (GObject *self,
+    guint prop_id,
+    GValue *value,
+    GParamSpec *pspec)
+{
+  EmpathyContactSelectorDialog *dialog = EMPATHY_CONTACT_SELECTOR_DIALOG (self);
+
+  switch (prop_id)
+    {
+      case PROP_SHOW_ACCOUNT_CHOOSER:
+        g_value_set_boolean (value,
+          empathy_contact_selector_dialog_get_show_account_chooser (dialog));
+        break;
+
+      case PROP_FILTER_ACCOUNT:
+        g_value_set_object (value,
+            empathy_contact_selector_dialog_get_filter_account (dialog));
+        break;
+
+      default:
+        G_OBJECT_WARN_INVALID_PROPERTY_ID (self, prop_id, pspec);
+        break;
+    }
+}
+
+static void
+empathy_contact_selector_dialog_set_property (GObject *self,
+    guint prop_id,
+    const GValue *value,
+    GParamSpec *pspec)
+{
+  EmpathyContactSelectorDialog *dialog = EMPATHY_CONTACT_SELECTOR_DIALOG (self);
+
+  switch (prop_id)
+    {
+      case PROP_SHOW_ACCOUNT_CHOOSER:
+        empathy_contact_selector_dialog_set_show_account_chooser (dialog,
+            g_value_get_boolean (value));
+        break;
+
+      case PROP_FILTER_ACCOUNT:
+        empathy_contact_selector_dialog_set_filter_account (dialog,
+            g_value_get_object (value));
+        break;
+
+      default:
+        G_OBJECT_WARN_INVALID_PROPERTY_ID (self, prop_id, pspec);
+        break;
+    }
+}
+
+static void
+empathy_contact_selector_dialog_constructed (GObject *dialog)
+{
+  EmpathyContactSelectorDialogPriv *priv = GET_PRIV (dialog);
+
+  if (EMPATHY_CONTACT_SELECTOR_DIALOG_GET_CLASS (dialog)->contact_filter)
+    {
+      GtkEntryCompletion *completion;
+      GtkTreeModel *filter;
+
+      completion = gtk_entry_get_completion (GTK_ENTRY (priv->entry_id));
+      filter = gtk_tree_model_filter_new (GTK_TREE_MODEL (priv->store), NULL);
+
+      gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER (filter),
+          contact_selector_dialog_filter_visible, dialog, NULL);
+
+      gtk_entry_completion_set_model (completion, filter);
+      g_object_unref (filter);
+    }
+}
+
 static void
 empathy_contact_selector_dialog_dispose (GObject *object)
 {
@@ -319,6 +452,11 @@ empathy_contact_selector_dialog_dispose (GObject *object)
     priv->contact_manager = NULL;
   }
 
+  if (priv->filter_account != NULL) {
+    g_object_unref (priv->filter_account);
+    priv->filter_account = NULL;
+  }
+
   if (G_OBJECT_CLASS (empathy_contact_selector_dialog_parent_class)->dispose)
     G_OBJECT_CLASS (empathy_contact_selector_dialog_parent_class)->dispose (
         object);
@@ -332,5 +470,120 @@ empathy_contact_selector_dialog_class_init (
 
   g_type_class_add_private (class, sizeof (EmpathyContactSelectorDialogPriv));
 
+  class->contact_filter = NULL;
+
+  object_class->constructed = empathy_contact_selector_dialog_constructed;
   object_class->dispose = empathy_contact_selector_dialog_dispose;
+  object_class->get_property = empathy_contact_selector_dialog_get_property;
+  object_class->set_property = empathy_contact_selector_dialog_set_property;
+
+  g_object_class_install_property (object_class, PROP_SHOW_ACCOUNT_CHOOSER,
+      g_param_spec_boolean ("show-account-chooser",
+        "Show Account Chooser",
+        "Whether or not this dialog should show an account chooser",
+        TRUE,
+        G_PARAM_READWRITE));
+
+  g_object_class_install_property (object_class, PROP_FILTER_ACCOUNT,
+      g_param_spec_object ("filter-account",
+        "Account to filter contacts",
+        "if 'show-account-chooser' is unset, only the contacts from this "
+        "account are displayed",
+        TP_TYPE_ACCOUNT,
+        G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
+}
+
+const gchar *
+empathy_contact_selector_dialog_get_selected (
+    EmpathyContactSelectorDialog *self,
+    TpConnection **connection,
+    TpAccount **account)
+{
+  EmpathyContactSelectorDialogPriv *priv;
+  const char *id;
+
+  g_return_val_if_fail (EMPATHY_IS_CONTACT_SELECTOR_DIALOG (self), NULL);
+
+  priv = GET_PRIV (self);
+
+  if (connection != NULL)
+    {
+      if (priv->show_account_chooser)
+        *connection = empathy_account_chooser_get_connection (
+            EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser));
+      else
+        *connection = NULL;
+    }
+
+  if (account != NULL)
+    {
+      if (priv->show_account_chooser)
+        *account = empathy_account_chooser_get_account (
+            EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser));
+      else
+        *account = NULL;
+    }
+
+
+  id = gtk_entry_get_text (GTK_ENTRY (priv->entry_id));
+  return id;
+}
+
+void
+empathy_contact_selector_dialog_set_show_account_chooser (
+    EmpathyContactSelectorDialog *self,
+    gboolean show_account_chooser)
+{
+  EmpathyContactSelectorDialogPriv *priv;
+
+  g_return_if_fail (EMPATHY_IS_CONTACT_SELECTOR_DIALOG (self));
+
+  priv = GET_PRIV (self);
+  priv->show_account_chooser = show_account_chooser;
+
+  gtk_widget_set_visible (priv->account_chooser_label, show_account_chooser);
+  gtk_widget_set_visible (priv->account_chooser, show_account_chooser);
+  contact_selector_dialog_account_changed_cb (priv->account_chooser, self);
+
+  g_object_notify (G_OBJECT (self), "show-account-chooser");
+}
+
+gboolean
+empathy_contact_selector_dialog_get_show_account_chooser (
+    EmpathyContactSelectorDialog *self)
+{
+  EmpathyContactSelectorDialogPriv *priv;
+
+  g_return_val_if_fail (EMPATHY_IS_CONTACT_SELECTOR_DIALOG (self), FALSE);
+
+  priv = GET_PRIV (self);
+  return priv->show_account_chooser;
+}
+
+void
+empathy_contact_selector_dialog_set_filter_account (
+    EmpathyContactSelectorDialog *self,
+    TpAccount *account)
+{
+  EmpathyContactSelectorDialogPriv *priv;
+
+  g_return_if_fail (EMPATHY_IS_CONTACT_SELECTOR_DIALOG (self));
+
+  priv = GET_PRIV (self);
+  priv->filter_account = g_object_ref (account);
+
+  g_object_notify (G_OBJECT (self), "filter-account");
+}
+
+TpAccount *
+empathy_contact_selector_dialog_get_filter_account (
+    EmpathyContactSelectorDialog *self)
+{
+  EmpathyContactSelectorDialogPriv *priv;
+
+  g_return_val_if_fail (EMPATHY_IS_CONTACT_SELECTOR_DIALOG (self), NULL);
+
+  priv = GET_PRIV (self);
+  return priv->filter_account;
+
 }