]> git.0d.be Git - empathy.git/blobdiff - src/empathy-invite-participant-dialog.c
roster-window: fix crash if empathy_chatroom_manager_find() failed
[empathy.git] / src / empathy-invite-participant-dialog.c
index f67a7feded4448f1d0f0310413bd347ac3ff5ae6..437d8e6b7358bab9242923906201f0c59df889d3 100644 (file)
  *    Danielle Madeley <danielle.madeley@collabora.co.uk>
  */
 
-#include <glib/gi18n.h>
-
+#include "config.h"
 #include "empathy-invite-participant-dialog.h"
 
-#define GET_PRIVATE(obj)    (G_TYPE_INSTANCE_GET_PRIVATE ((obj), EMPATHY_TYPE_INVITE_PARTICIPANT_DIALOG, EmpathyInviteParticipantDialogPrivate))
+#include <glib/gi18n.h>
+#include <telepathy-glib/telepathy-glib-dbus.h>
+
+#include "empathy-contact-chooser.h"
+#include "empathy-utils.h"
 
 G_DEFINE_TYPE (EmpathyInviteParticipantDialog,
-    empathy_invite_participant_dialog, EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG);
+    empathy_invite_participant_dialog, GTK_TYPE_DIALOG);
+
+enum
+{
+  PROP_TP_CHAT = 1
+};
 
-typedef struct _EmpathyInviteParticipantDialogPrivate EmpathyInviteParticipantDialogPrivate;
 struct _EmpathyInviteParticipantDialogPrivate
 {
-  int dum;
+  EmpathyTpChat *tp_chat;
+
+  GtkWidget *chooser;
+  GtkWidget *invite_button;
 };
 
 static void
-empathy_invite_participant_dialog_class_init (EmpathyInviteParticipantDialogClass *klass)
+invite_participant_dialog_get_property (GObject *object,
+    guint param_id,
+    GValue *value,
+    GParamSpec *pspec)
 {
-  GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
+  EmpathyInviteParticipantDialog *self = (EmpathyInviteParticipantDialog *)
+    object;
 
-  g_type_class_add_private (gobject_class, sizeof (EmpathyInviteParticipantDialogPrivate));
+  switch (param_id)
+    {
+    case PROP_TP_CHAT:
+      g_value_set_object (value, self->priv->tp_chat);
+      break;
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
+      break;
+    };
 }
 
 static void
-empathy_invite_participant_dialog_init (EmpathyInviteParticipantDialog *self)
+invite_participant_dialog_set_property (GObject *object,
+    guint param_id,
+    const GValue *value,
+    GParamSpec *pspec)
+{
+  EmpathyInviteParticipantDialog *self = (EmpathyInviteParticipantDialog *)
+    object;
+
+  switch (param_id)
+    {
+    case PROP_TP_CHAT:
+      g_assert (self->priv->tp_chat == NULL); /* construct-only */
+      self->priv->tp_chat = g_value_dup_object (value);
+      break;
+    default:
+      G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
+      break;
+    };
+}
+
+static void
+invite_participant_dialog_dispose (GObject *object)
+{
+  EmpathyInviteParticipantDialog *self = (EmpathyInviteParticipantDialog *)
+    object;
+
+  tp_clear_object (&self->priv->tp_chat);
+
+  G_OBJECT_CLASS (empathy_invite_participant_dialog_parent_class)->dispose (
+      object);
+}
+
+static void
+selection_changed_cb (GtkWidget *treeview,
+    FolksIndividual *selected,
+    EmpathyInviteParticipantDialog *self)
+{
+  gtk_widget_set_sensitive (self->priv->invite_button, selected != NULL);
+}
+
+static void
+activate_cb (GtkWidget *chooser,
+    EmpathyInviteParticipantDialog *self)
+{
+  gtk_dialog_response (GTK_DIALOG (self), GTK_RESPONSE_ACCEPT);
+}
+
+/* Return the TpContact of @individual which is on the same connection as the
+ * EmpathyTpChat */
+static TpContact *
+get_tp_contact_for_chat (EmpathyInviteParticipantDialog *self,
+    FolksIndividual *individual)
+{
+  TpConnection *chat_conn;
+
+  chat_conn = tp_channel_get_connection (TP_CHANNEL (self->priv->tp_chat));
+  if (chat_conn == NULL)
+    return NULL;
+
+  return empathy_get_tp_contact_for_individual (individual, chat_conn);
+}
+
+static gboolean
+filter_individual (EmpathyContactChooser *chooser,
+    FolksIndividual *individual,
+    gboolean is_online,
+    gboolean searching,
+    gpointer user_data)
+{
+  EmpathyInviteParticipantDialog *self = user_data;
+  GList *members, *l;
+  TpContact *contact;
+  gboolean display = TRUE;
+
+  /* Filter out offline contacts if we are not searching */
+  if (!searching && !is_online)
+    return FALSE;
+
+  /* Filter out individuals not having a persona on the same connection as the
+   * EmpathyTpChat. */
+  contact = get_tp_contact_for_chat (self, individual);
+
+  if (contact == NULL)
+    return FALSE;
+
+  /* Filter out contacts which are already in the chat */
+  members = empathy_tp_chat_get_members (self->priv->tp_chat);
+
+  for (l = members; l != NULL; l = g_list_next (l))
+    {
+      EmpathyContact *member = l->data;
+      TpContact *owner;
+
+      /* Try to get the non-channel specific contact. */
+      owner = tp_channel_group_get_contact_owner (
+          TP_CHANNEL (self->priv->tp_chat),
+          empathy_contact_get_tp_contact (member));
+
+      if (owner == NULL)
+        owner = empathy_contact_get_tp_contact (member);
+
+      if (owner == contact)
+        {
+          display = FALSE;
+          break;
+        }
+    }
+
+  g_list_free_full (members, g_object_unref);
+
+  return display;
+}
+
+static gboolean
+has_contact_list (EmpathyInviteParticipantDialog *self)
 {
-  EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG (self);
-  // EmpathyInviteParticipantDialogPrivate *priv = GET_PRIVATE (self);
+  TpConnection *conn;
+
+  conn = tp_channel_get_connection (TP_CHANNEL (self->priv->tp_chat));
+
+  return tp_proxy_has_interface_by_id (conn,
+      TP_IFACE_QUARK_CONNECTION_INTERFACE_CONTACT_LIST);
+}
+
+static void
+invite_participant_dialog_constructed (GObject *object)
+{
+  EmpathyInviteParticipantDialog *self =
+    (EmpathyInviteParticipantDialog *) object;
+  GtkDialog *dialog = GTK_DIALOG (self);
   GtkWidget *label;
   char *str;
+  GtkWidget *content;
+
+  content = gtk_dialog_get_content_area (dialog);
 
   label = gtk_label_new (NULL);
   str = g_strdup_printf (
@@ -48,25 +199,81 @@ empathy_invite_participant_dialog_init (EmpathyInviteParticipantDialog *self)
   gtk_label_set_markup (GTK_LABEL (label), str);
   g_free (str);
 
-  gtk_box_pack_start (GTK_BOX (parent->vbox), label, FALSE, TRUE, 0);
-  /* move to the top -- wish there was a better way to do this */
-  gtk_box_reorder_child (GTK_BOX (parent->vbox), label, 0);
+  gtk_box_pack_start (GTK_BOX (content), label, FALSE, TRUE, 6);
   gtk_widget_show (label);
 
-  parent->button_action = gtk_dialog_add_button (GTK_DIALOG (self),
-      "Invite", GTK_RESPONSE_ACCEPT);
-  gtk_widget_set_sensitive (parent->button_action, FALSE);
+  gtk_dialog_add_button (dialog, GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL);
+
+  /* contact chooser */
+  self->priv->chooser = empathy_contact_chooser_new ();
+
+  empathy_contact_chooser_set_filter_func (
+      EMPATHY_CONTACT_CHOOSER (self->priv->chooser), filter_individual, self);
+
+  gtk_box_pack_start (GTK_BOX (content), self->priv->chooser, TRUE, TRUE, 6);
+  gtk_widget_show (self->priv->chooser);
+
+  g_signal_connect (self->priv->chooser, "selection-changed",
+      G_CALLBACK (selection_changed_cb), self);
+  g_signal_connect (self->priv->chooser, "activate",
+      G_CALLBACK (activate_cb), self);
+
+  self->priv->invite_button = gtk_dialog_add_button (dialog, _("Invite"),
+      GTK_RESPONSE_ACCEPT);
+  gtk_widget_set_sensitive (self->priv->invite_button, FALSE);
 
   gtk_window_set_title (GTK_WINDOW (self), _("Invite Participant"));
   gtk_window_set_role (GTK_WINDOW (self), "invite_participant");
-  empathy_contact_selector_dialog_set_show_account_chooser (
-      EMPATHY_CONTACT_SELECTOR_DIALOG (self), FALSE);
+
+  if (has_contact_list (self))
+    {
+      /* Set a default height so a few contacts are displayed */
+      gtk_window_set_default_size (GTK_WINDOW (self), -1, 400);
+    }
+  else
+    {
+      /* No need to display an empty treeview (ie IRC) */
+      empathy_contact_chooser_show_tree_view (
+          EMPATHY_CONTACT_CHOOSER (self->priv->chooser), FALSE);
+    }
+}
+
+static void
+empathy_invite_participant_dialog_class_init (
+    EmpathyInviteParticipantDialogClass *klass)
+{
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+  object_class->get_property = invite_participant_dialog_get_property;
+  object_class->set_property = invite_participant_dialog_set_property;
+  object_class->constructed = invite_participant_dialog_constructed;
+  object_class->dispose = invite_participant_dialog_dispose;
+
+  g_type_class_add_private (object_class,
+      sizeof (EmpathyInviteParticipantDialogPrivate));
+
+  g_object_class_install_property (object_class,
+      PROP_TP_CHAT,
+      g_param_spec_object ("tp-chat", "EmpathyTpChat", "EmpathyTpChat",
+          EMPATHY_TYPE_TP_CHAT,
+          G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
+}
+
+static void
+empathy_invite_participant_dialog_init (EmpathyInviteParticipantDialog *self)
+{
+  self->priv = G_TYPE_INSTANCE_GET_PRIVATE (
+      self, EMPATHY_TYPE_INVITE_PARTICIPANT_DIALOG,
+      EmpathyInviteParticipantDialogPrivate);
 }
 
 GtkWidget *
-empathy_invite_participant_dialog_new (GtkWindow *parent)
+empathy_invite_participant_dialog_new (GtkWindow *parent,
+    EmpathyTpChat *tp_chat)
 {
-  GtkWidget *self = g_object_new (EMPATHY_TYPE_INVITE_PARTICIPANT_DIALOG, NULL);
+  GtkWidget *self = g_object_new (EMPATHY_TYPE_INVITE_PARTICIPANT_DIALOG,
+      "tp-chat", tp_chat,
+      NULL);
 
   if (parent != NULL)
     {
@@ -75,3 +282,21 @@ empathy_invite_participant_dialog_new (GtkWindow *parent)
 
   return self;
 }
+
+TpContact *
+empathy_invite_participant_dialog_get_selected (
+    EmpathyInviteParticipantDialog *self)
+{
+  FolksIndividual *individual;
+  TpContact *contact;
+
+  individual = empathy_contact_chooser_dup_selected (
+      EMPATHY_CONTACT_CHOOSER (self->priv->chooser));
+  if (individual == NULL)
+    return NULL;
+
+  contact = get_tp_contact_for_chat (self, individual);
+
+  g_object_unref (individual);
+  return contact;
+}