]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-new-message-dialog.c
switch new message dialog to use an header bar
[empathy.git] / libempathy-gtk / empathy-new-message-dialog.c
index 40c25974c91474c4a5b532ba8d90eeeeca851b8d..aa55ae01d74821bfa250afeac2d4ffe473af62e2 100644 (file)
  * Authors: Xavier Claessens <xclaesse@gmail.com>
  */
 
-#include <config.h>
-
-#include <string.h>
-#include <stdlib.h>
+#include "config.h"
+#include "empathy-new-message-dialog.h"
 
-#include <gtk/gtk.h>
 #include <glib/gi18n-lib.h>
 
-#include <telepathy-glib/interfaces.h>
-
-#include <libempathy/empathy-tp-contact-factory.h>
-#include <libempathy/empathy-contact-manager.h>
-#include <libempathy/empathy-request-util.h>
-#include <libempathy/empathy-utils.h>
+#include "empathy-request-util.h"
+#include "empathy-contact-chooser.h"
+#include "empathy-ui-utils.h"
+#include "empathy-images.h"
 
 #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT
-#include <libempathy/empathy-debug.h>
-
-#include <libempathy-gtk/empathy-ui-utils.h>
-#include <libempathy-gtk/empathy-images.h>
-
-#include "empathy-new-message-dialog.h"
-#include "empathy-account-chooser.h"
+#include "empathy-debug.h"
 
 static EmpathyNewMessageDialog *dialog_singleton = NULL;
 
 G_DEFINE_TYPE(EmpathyNewMessageDialog, empathy_new_message_dialog,
-               EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG)
+    GTK_TYPE_DIALOG)
+
+struct _EmpathyNewMessageDialogPriv {
+  GtkWidget *chooser;
+  GtkWidget *button_chat;
+  GtkWidget *button_sms;
+};
 
 /**
  * SECTION:empathy-new-message-dialog
@@ -63,121 +58,132 @@ enum
   EMP_NEW_MESSAGE_SMS,
 };
 
-static void
-empathy_new_message_dialog_response (GtkDialog *dialog, int response_id)
+static const gchar *
+get_error_display_message (GError *error)
 {
-  TpAccount *account;
-  const gchar *contact_id;
-
-  if (response_id < EMP_NEW_MESSAGE_TEXT) goto out;
-
-  contact_id = empathy_contact_selector_dialog_get_selected (
-      EMPATHY_CONTACT_SELECTOR_DIALOG (dialog), NULL, &account);
+  if (error->domain != TP_ERROR)
+    goto out;
 
-  if (EMP_STR_EMPTY (contact_id) || account == NULL) goto out;
-
-  switch (response_id)
+  switch (error->code)
     {
-      case EMP_NEW_MESSAGE_TEXT:
-        empathy_chat_with_contact_id (account, contact_id,
-            empathy_get_current_action_time (),
-            NULL, NULL);
-        break;
-
-      case EMP_NEW_MESSAGE_SMS:
-        empathy_sms_contact_id (account, contact_id,
-            empathy_get_current_action_time (),
-            NULL, NULL);
+      case TP_ERROR_NETWORK_ERROR:
+        return _("Network error");
+      case TP_ERROR_OFFLINE:
+        return _("The contact is offline");
+      case TP_ERROR_INVALID_HANDLE:
+        return _("The specified contact is either invalid or unknown");
+      case TP_ERROR_NOT_CAPABLE:
+        return _("The contact does not support this kind of conversation");
+      case TP_ERROR_NOT_IMPLEMENTED:
+        return _("The requested functionality is not implemented "
+                 "for this protocol");
+      case TP_ERROR_INVALID_ARGUMENT:
+        /* Not very user friendly to say 'invalid arguments' */
         break;
-
+      case TP_ERROR_NOT_AVAILABLE:
+        return _("Could not start a conversation with the given contact");
+      case TP_ERROR_CHANNEL_BANNED:
+        return _("You are banned from this channel");
+      case TP_ERROR_CHANNEL_FULL:
+        return _("This channel is full");
+      case TP_ERROR_CHANNEL_INVITE_ONLY:
+        return _("You must be invited to join this channel");
+      case TP_ERROR_DISCONNECTED:
+        return _("Can't proceed while disconnected");
+      case TP_ERROR_PERMISSION_DENIED:
+        return _("Permission denied");
       default:
-        g_warn_if_reached ();
+        DEBUG ("Unhandled error code: %d", error->code);
     }
 
 out:
-  gtk_widget_destroy (GTK_WIDGET (dialog));
+  return _("There was an error starting the conversation");
 }
 
 static void
-empathy_new_message_account_filter (EmpathyContactSelectorDialog *dialog,
-    EmpathyAccountChooserFilterResultCallback callback,
-    gpointer callback_data,
-    TpAccount *account)
+show_chat_error (GError *error,
+    GtkWindow *parent)
 {
-  TpConnection *connection;
-  gboolean supported = FALSE;
-  TpCapabilities *caps;
-
-  /* check if CM supports 1-1 text chat */
-  connection = tp_account_get_connection (account);
-  if (connection == NULL)
-      goto out;
+  GtkWidget *dialog;
 
-  caps = tp_connection_get_capabilities (connection);
-  if (caps == NULL)
-      goto out;
+  dialog = gtk_message_dialog_new (parent, GTK_DIALOG_MODAL,
+      GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
+      "%s",
+      get_error_display_message (error));
 
-  supported = tp_capabilities_supports_text_chats (caps);
+  g_signal_connect_swapped (dialog, "response",
+      G_CALLBACK (gtk_widget_destroy),
+      dialog);
 
-out:
-  callback (supported, callback_data);
+  gtk_widget_show (dialog);
 }
 
 static void
-empathy_new_message_dialog_update_sms_button_sensitivity (GtkWidget *widget,
-    GParamSpec *pspec,
-    GtkWidget *button)
+ensure_text_channel_cb (GObject *source,
+    GAsyncResult *result,
+    gpointer user_data)
 {
-  GtkWidget *self = gtk_widget_get_toplevel (widget);
-  EmpathyContactSelectorDialog *dialog;
-  TpConnection *conn;
-  GPtrArray *rccs;
-  gboolean sensitive = FALSE;
-  guint i;
+  GError *error = NULL;
 
-  g_return_if_fail (EMPATHY_IS_NEW_MESSAGE_DIALOG (self));
-
-  dialog = EMPATHY_CONTACT_SELECTOR_DIALOG (self);
-
-  /* if the Text widget isn't sensitive, don't bother checking the caps */
-  if (!gtk_widget_get_sensitive (dialog->button_action))
-    goto finally;
+  if (!tp_account_channel_request_ensure_channel_finish (
+        TP_ACCOUNT_CHANNEL_REQUEST (source), result, &error))
+    {
+      DEBUG ("Failed to ensure text channel: %s", error->message);
+      show_chat_error (error, user_data);
+      g_error_free (error);
+    }
+}
 
-  empathy_contact_selector_dialog_get_selected (dialog, &conn, NULL);
+static void
+empathy_new_message_dialog_response (GtkDialog *dialog,
+    int response_id)
+{
+  EmpathyNewMessageDialog *self = (EmpathyNewMessageDialog *) dialog;
+  FolksIndividual *individual = NULL;
+  EmpathyContact *contact = NULL;
 
-  if (conn == NULL)
-    goto finally;
+  if (response_id < EMP_NEW_MESSAGE_TEXT)
+    goto out;
 
-  /* iterate the rccs to find if SMS channels are supported, this should
-   * be in tp-glib */
-  rccs = tp_capabilities_get_channel_classes (
-      tp_connection_get_capabilities (conn));
+  individual = empathy_contact_chooser_dup_selected (
+      EMPATHY_CONTACT_CHOOSER (self->priv->chooser));
+  if (individual == NULL)
+    goto out;
 
-  for (i = 0; i < rccs->len; i++)
+  switch (response_id)
     {
-      GHashTable *fixed;
-      GStrv allowed;
-      const char *type;
-      gboolean sms_channel;
-
-      tp_value_array_unpack (g_ptr_array_index (rccs, i), 2,
-          &fixed,
-          &allowed);
+      case EMP_NEW_MESSAGE_TEXT:
+        contact = empathy_contact_dup_best_for_action (individual,
+            EMPATHY_ACTION_CHAT);
+        g_return_if_fail (contact != NULL);
 
-      /* SMS channels are type:Text and sms-channel:True */
-      type = tp_asv_get_string (fixed, TP_PROP_CHANNEL_CHANNEL_TYPE);
-      sms_channel = tp_asv_get_boolean (fixed,
-          TP_PROP_CHANNEL_INTERFACE_SMS_SMS_CHANNEL, NULL);
+        empathy_chat_with_contact_id (empathy_contact_get_account (contact),
+            empathy_contact_get_id (contact),
+            empathy_get_current_action_time (),
+            ensure_text_channel_cb,
+            gtk_widget_get_parent_window (GTK_WIDGET (dialog)));
+        break;
 
-      sensitive = sms_channel &&
-        !tp_strdiff (type, TP_IFACE_CHANNEL_TYPE_TEXT);
+      case EMP_NEW_MESSAGE_SMS:
+        contact = empathy_contact_dup_best_for_action (individual,
+            EMPATHY_ACTION_SMS);
+        g_return_if_fail (contact != NULL);
 
-      if (sensitive)
+        empathy_sms_contact_id (empathy_contact_get_account (contact),
+            empathy_contact_get_id (contact),
+            empathy_get_current_action_time (),
+            ensure_text_channel_cb,
+            gtk_widget_get_parent_window (GTK_WIDGET (dialog)));
         break;
+
+      default:
+        g_warn_if_reached ();
     }
 
-finally:
-  gtk_widget_set_sensitive (button, sensitive);
+out:
+  tp_clear_object (&individual);
+  tp_clear_object (&contact);
+  gtk_widget_destroy (GTK_WIDGET (dialog));
 }
 
 static GObject *
@@ -205,48 +211,125 @@ empathy_new_message_dialog_constructor (GType type,
   return retval;
 }
 
+static gboolean
+individual_supports_action (FolksIndividual *individual,
+    EmpathyActionType action)
+{
+  EmpathyContact *contact;
+
+  contact = empathy_contact_dup_best_for_action (individual, action);
+  if (contact == NULL)
+    return FALSE;
+
+  g_object_unref (contact);
+  return TRUE;
+}
+
+static gboolean
+filter_individual (EmpathyContactChooser *chooser,
+    FolksIndividual *individual,
+    gboolean is_online,
+    gboolean searching,
+    gpointer user_data)
+{
+  return individual_supports_action (individual, EMPATHY_ACTION_CHAT) ||
+    individual_supports_action (individual, EMPATHY_ACTION_SMS);
+}
+
+static void
+selection_changed_cb (GtkWidget *chooser,
+    FolksIndividual *selected,
+    EmpathyNewMessageDialog *self)
+{
+  gboolean can_chat, can_sms;
+
+  if (selected == NULL)
+    {
+      can_chat = can_sms = FALSE;
+    }
+  else
+    {
+      can_chat = individual_supports_action (selected, EMPATHY_ACTION_CHAT);
+      can_sms = individual_supports_action (selected, EMPATHY_ACTION_SMS);
+    }
+
+  gtk_widget_set_sensitive (self->priv->button_chat, can_chat);
+#if 0
+  gtk_widget_set_sensitive (self->priv->button_sms, can_sms);
+#endif
+}
+
+static void
+selection_activate_cb (GtkWidget *chooser,
+    EmpathyNewMessageDialog *self)
+{
+  gtk_dialog_response (GTK_DIALOG (self), EMP_NEW_MESSAGE_TEXT);
+}
+
 static void
-empathy_new_message_dialog_init (EmpathyNewMessageDialog *dialog)
+empathy_new_message_dialog_init (EmpathyNewMessageDialog *self)
 {
-  EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG (
-        dialog);
-  GtkWidget *button;
   GtkWidget *image;
+  GtkWidget *content;
+
+  self->priv = G_TYPE_INSTANCE_GET_PRIVATE (self,
+      EMPATHY_TYPE_NEW_MESSAGE_DIALOG, EmpathyNewMessageDialogPriv);
+
+  content = gtk_dialog_get_content_area (GTK_DIALOG (self));
+
+  /* contact chooser */
+  self->priv->chooser = empathy_contact_chooser_new ();
 
-  /* add an SMS button */
-  button = gtk_button_new_with_mnemonic (_("_SMS"));
+  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, 0);
+  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 (selection_activate_cb), self);
+
+  /* close button */
+  gtk_dialog_add_button (GTK_DIALOG (self),
+      GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL);
+
+#if 0
+  /* add SMS button */
+  self->priv->button_sms = gtk_button_new_with_mnemonic (_("_SMS"));
   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_SMS,
       GTK_ICON_SIZE_BUTTON);
-  gtk_button_set_image (GTK_BUTTON (button), image);
-
-  gtk_dialog_add_action_widget (GTK_DIALOG (dialog), button,
-      EMP_NEW_MESSAGE_SMS);
-  gtk_widget_show (button);
+  gtk_button_set_image (GTK_BUTTON (self->priv->button_sms), image);
+#endif
 
   /* add chat button */
-  parent->button_action = gtk_button_new_with_mnemonic (_("C_hat"));
+  self->priv->button_chat = gtk_button_new_with_mnemonic (_("_Done"));
   image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_NEW_MESSAGE,
       GTK_ICON_SIZE_BUTTON);
-  gtk_button_set_image (GTK_BUTTON (parent->button_action), image);
+  gtk_button_set_image (GTK_BUTTON (self->priv->button_chat), image);
 
-  gtk_dialog_add_action_widget (GTK_DIALOG (dialog), parent->button_action,
-      EMP_NEW_MESSAGE_TEXT);
-  gtk_widget_show (parent->button_action);
+#if 0
+  gtk_dialog_add_action_widget (GTK_DIALOG (self), self->priv->button_sms,
+      EMP_NEW_MESSAGE_SMS);
+  gtk_widget_show (self->priv->button_sms);
+#endif
 
-  /* the parent class will update the sensitivity of button_action, propagate
-   * it */
-  g_signal_connect (parent->button_action, "notify::sensitive",
-      G_CALLBACK (empathy_new_message_dialog_update_sms_button_sensitivity),
-      button);
-  g_signal_connect (dialog, "notify::selected-account",
-      G_CALLBACK (empathy_new_message_dialog_update_sms_button_sensitivity),
-      button);
+  gtk_dialog_add_action_widget (GTK_DIALOG (self), self->priv->button_chat,
+      EMP_NEW_MESSAGE_TEXT);
+  gtk_widget_show (self->priv->button_chat);
 
   /* Tweak the dialog */
-  gtk_window_set_title (GTK_WINDOW (dialog), _("New Conversation"));
-  gtk_window_set_role (GTK_WINDOW (dialog), "new_message");
+  gtk_window_set_title (GTK_WINDOW (self), _("New Conversation"));
+  gtk_window_set_role (GTK_WINDOW (self), "new_message");
+
+  /* Set a default height so a few contacts are displayed */
+  gtk_window_set_default_size (GTK_WINDOW (self), -1, 600);
 
-  gtk_widget_set_sensitive (parent->button_action, FALSE);
+  gtk_widget_set_sensitive (self->priv->button_chat, FALSE);
+#if 0
+  gtk_widget_set_sensitive (self->priv->button_sms, FALSE);
+#endif
 }
 
 static void
@@ -255,14 +338,12 @@ empathy_new_message_dialog_class_init (
 {
   GObjectClass *object_class = G_OBJECT_CLASS (class);
   GtkDialogClass *dialog_class = GTK_DIALOG_CLASS (class);
-  EmpathyContactSelectorDialogClass *selector_dialog_class = \
-    EMPATHY_CONTACT_SELECTOR_DIALOG_CLASS (class);
 
   object_class->constructor = empathy_new_message_dialog_constructor;
 
   dialog_class->response = empathy_new_message_dialog_response;
 
-  selector_dialog_class->account_filter = empathy_new_message_account_filter;
+  g_type_class_add_private (class, sizeof (EmpathyNewMessageDialogPriv));
 }
 
 /**
@@ -278,7 +359,9 @@ empathy_new_message_dialog_show (GtkWindow *parent)
 {
   GtkWidget *dialog;
 
-  dialog = g_object_new (EMPATHY_TYPE_NEW_MESSAGE_DIALOG, NULL);
+  dialog = g_object_new (EMPATHY_TYPE_NEW_MESSAGE_DIALOG,
+                  "use-header-bar", 1,
+                  NULL);
 
   if (parent)
     {