]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-new-message-dialog.c
Merge branch 'sasl'
[empathy.git] / libempathy-gtk / empathy-new-message-dialog.c
index 66b2adb2d70d94de251383d43883ebb2702c1e8a..4747dfbb92762c1de26fda9d2192b9a28eb0a34e 100644 (file)
@@ -1,4 +1,3 @@
-/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
 /*
  * Copyright (C) 2007-2008 Collabora Ltd.
  *
 #include <gtk/gtk.h>
 #include <glib/gi18n-lib.h>
 
-#include <libempathy/empathy-call-factory.h>
+#include <telepathy-glib/interfaces.h>
+
 #include <libempathy/empathy-tp-contact-factory.h>
 #include <libempathy/empathy-contact-manager.h>
 #include <libempathy/empathy-dispatcher.h>
 #include <libempathy/empathy-utils.h>
-#include <libempathy/empathy-account.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"
 
+typedef struct {
+  EmpathyAccountChooserFilterResultCallback callback;
+  gpointer                                  user_data;
+} FilterCallbackData;
+
+static EmpathyNewMessageDialog *dialog_singleton = NULL;
+
+G_DEFINE_TYPE(EmpathyNewMessageDialog, empathy_new_message_dialog,
+               EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG)
+
 /**
  * SECTION:empathy-new-message-dialog
  * @title: EmpathyNewMessageDialog
  * @short_description: A dialog to show a new message
  * @include: libempathy-gtk/empathy-new-message-dialog.h
  *
- * #EmpathyNewMessageDialog is a dialog which allows a text chat or
- * call to be started with any contact on any enabled account.
+ * #EmpathyNewMessageDialog is a dialog which allows a text chat
+ * to be started with any contact on any enabled account.
  */
 
-typedef struct {
-       GtkWidget *dialog;
-       GtkWidget *table_contact;
-       GtkWidget *account_chooser;
-       GtkWidget *entry_id;
-       GtkWidget *button_chat;
-       GtkWidget *button_call;
-       EmpathyContactManager *contact_manager;
-} EmpathyNewMessageDialog;
-
-enum {
-       COMPLETION_COL_TEXT,
-       COMPLETION_COL_ID,
-       COMPLETION_COL_NAME,
-} CompletionCol;
-
 static void
-new_message_dialog_account_changed_cb (GtkWidget               *widget,
-                                      EmpathyNewMessageDialog *dialog)
+empathy_new_message_dialog_response (GtkDialog *dialog, int response_id)
 {
-       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 (dialog->entry_id));
-       store = GTK_LIST_STORE (gtk_entry_completion_get_model (completion));
-       gtk_list_store_clear (store);
-
-       /* Get members of the new account */
-       chooser = EMPATHY_ACCOUNT_CHOOSER (dialog->account_chooser);
-       connection = empathy_account_chooser_get_connection (chooser);
-       if (!connection) {
-               return;
-       }
-       contact_list = empathy_contact_manager_get_list (dialog->contact_manager,
-                                                        connection);
-       members = empathy_contact_list_get_members (EMPATHY_CONTACT_LIST (contact_list));
-
-       /* Add members to the completion */
-       while (members) {
-               EmpathyContact *contact = members->data;
-
-               if (empathy_contact_is_online (contact)) {
-                       DEBUG ("Adding contact ID %s, Name %s",
-                              empathy_contact_get_id (contact),
-                              empathy_contact_get_name (contact));
-
-                       tmpstr = g_strdup_printf ("%s (%s)",
-                               empathy_contact_get_name (contact),
-                               empathy_contact_get_id (contact));
-
-                       gtk_list_store_insert_with_values (store, &iter, -1,
-                               COMPLETION_COL_TEXT, tmpstr,
-                               COMPLETION_COL_ID, empathy_contact_get_id (contact),
-                               COMPLETION_COL_NAME, empathy_contact_get_name (contact),
-                               -1);
-
-                       g_free (tmpstr);
-               }
-
-               g_object_unref (contact);
-               members = g_list_delete_link (members, members);
-       }
-}
-
-static gboolean
-new_message_dialog_match_selected_cb (GtkEntryCompletion *widget,
-                                     GtkTreeModel       *model,
-                                     GtkTreeIter        *iter,
-                                     EmpathyNewMessageDialog *dialog)
-{
-       gchar *id;
+  TpAccount *account;
+  const gchar *contact_id;
 
-       if (!iter || !model) {
-               return FALSE;
-       }
+  if (response_id != GTK_RESPONSE_ACCEPT) goto out;
 
-       gtk_tree_model_get (model, iter, COMPLETION_COL_ID, &id, -1);
-       gtk_entry_set_text (GTK_ENTRY (dialog->entry_id), id);
+  contact_id = empathy_contact_selector_dialog_get_selected (
+      EMPATHY_CONTACT_SELECTOR_DIALOG (dialog), NULL, &account);
 
-       DEBUG ("Got selected match **%s**", id);
+  if (EMP_STR_EMPTY (contact_id) || account == NULL) goto out;
 
-       g_free (id);
+  empathy_dispatcher_chat_with_contact_id (account, contact_id,
+      gtk_get_current_event_time ());
 
-       return TRUE;
-}
-
-static gboolean
-new_message_dialog_match_func (GtkEntryCompletion *completion,
-                              const gchar        *key,
-                              GtkTreeIter        *iter,
-                              gpointer            user_data)
-{
-       GtkTreeModel *model;
-       gchar        *id;
-       gchar        *name;
-
-       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)) {
-               DEBUG ("Key %s is matching name **%s**", key, name);
-               g_free (name);
-               return TRUE;
-       }
-       g_free (name);
-
-       gtk_tree_model_get (model, iter, COMPLETION_COL_ID, &id, -1);
-       if (strstr (id, key)) {
-               DEBUG ("Key %s is matching ID **%s**", key, id);
-               g_free (id);
-               return TRUE;
-       }
-       g_free (id);
-
-       return FALSE;
+out:
+  gtk_widget_destroy (GTK_WIDGET (dialog));
 }
 
 static void
-new_message_dialog_call_got_contact_cb (EmpathyTpContactFactory *factory,
-                                       EmpathyContact          *contact,
-                                       const GError            *error,
-                                       gpointer                 user_data,
-                                       GObject                 *weak_object)
+conn_prepared_cb (GObject *conn,
+    GAsyncResult *result,
+    gpointer user_data)
 {
-       EmpathyCallFactory *call_factory;
+  FilterCallbackData *data = user_data;
+  GError *myerr = NULL;
+  TpCapabilities *caps;
 
-       if (error != NULL) {
-               DEBUG ("Error: %s", error->message);
-               return;
-       }
+  if (!tp_proxy_prepare_finish (conn, result, &myerr))
+    {
+      data->callback (FALSE, data->user_data);
+      g_slice_free (FilterCallbackData, data);
+    }
 
-       call_factory = empathy_call_factory_get ();
-       empathy_call_factory_new_call (call_factory, contact);
+  caps = tp_connection_get_capabilities (TP_CONNECTION (conn));
+  data->callback (tp_capabilities_supports_text_chats (caps),
+      data->user_data);
+
+  g_slice_free (FilterCallbackData, data);
 }
 
 static void
-new_message_dialog_response_cb (GtkWidget               *widget,
-                               gint                    response,
-                               EmpathyNewMessageDialog *dialog)
+empathy_new_message_account_filter (EmpathyContactSelectorDialog *dialog,
+    EmpathyAccountChooserFilterResultCallback callback,
+    gpointer callback_data,
+    TpAccount *account)
+{
+  TpConnection *connection;
+  FilterCallbackData *cb_data;
+  GQuark features[] = { TP_CONNECTION_FEATURE_CAPABILITIES, 0 };
+
+  if (tp_account_get_connection_status (account, NULL) !=
+      TP_CONNECTION_STATUS_CONNECTED)
+    {
+      callback (FALSE, callback_data);
+      return;
+    }
+
+  /* check if CM supports 1-1 text chat */
+  connection = tp_account_get_connection (account);
+  if (connection == NULL)
+    {
+      callback (FALSE, callback_data);
+      return;
+    }
+
+  cb_data = g_slice_new0 (FilterCallbackData);
+  cb_data->callback = callback;
+  cb_data->user_data = callback_data;
+  tp_proxy_prepare_async (connection, features, conn_prepared_cb, cb_data);
+}
+
+static GObject *
+empathy_new_message_dialog_constructor (GType type,
+    guint n_props,
+    GObjectConstructParam *props)
 {
-       TpConnection *connection;
-       const gchar *id;
-
-       connection = empathy_account_chooser_get_connection (
-               EMPATHY_ACCOUNT_CHOOSER (dialog->account_chooser));
-       id = gtk_entry_get_text (GTK_ENTRY (dialog->entry_id));
-       if (!connection || EMP_STR_EMPTY (id)) {
-               gtk_widget_destroy (widget);
-               return;
-       }
-
-       if (response == 1) {
-               EmpathyTpContactFactory *factory;
-
-               factory = empathy_tp_contact_factory_dup_singleton (connection);
-               empathy_tp_contact_factory_get_from_id (factory, id,
-                       new_message_dialog_call_got_contact_cb,
-                       NULL, NULL, NULL);
-               g_object_unref (factory);
-       } else if (response == 2) {
-               empathy_dispatcher_chat_with_contact_id (connection, id, NULL, NULL);
-       }
-
-       gtk_widget_destroy (widget);
+  GObject *retval;
+
+  if (dialog_singleton)
+    {
+      retval = G_OBJECT (dialog_singleton);
+      g_object_ref (retval);
+    }
+  else
+    {
+      retval = G_OBJECT_CLASS (
+      empathy_new_message_dialog_parent_class)->constructor (type,
+        n_props, props);
+
+      dialog_singleton = EMPATHY_NEW_MESSAGE_DIALOG (retval);
+      g_object_add_weak_pointer (retval, (gpointer) &dialog_singleton);
+    }
+
+  return retval;
 }
 
 static void
-new_message_change_state_button_cb  (GtkEditable             *editable,
-                                    EmpathyNewMessageDialog *dialog)
+empathy_new_message_dialog_init (EmpathyNewMessageDialog *dialog)
 {
-       const gchar *id;
-       gboolean     sensitive;
+  EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG (
+        dialog);
+  GtkWidget *image;
+
+  /* add chat button */
+  parent->button_action = gtk_button_new_with_mnemonic (_("C_hat"));
+  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_dialog_add_action_widget (GTK_DIALOG (dialog), parent->button_action,
+      GTK_RESPONSE_ACCEPT);
+  gtk_widget_show (parent->button_action);
 
-       id = gtk_entry_get_text (GTK_ENTRY (editable));
-       sensitive = !EMP_STR_EMPTY (id);
+  /* Tweak the dialog */
+  gtk_window_set_title (GTK_WINDOW (dialog), _("New Conversation"));
+  gtk_window_set_role (GTK_WINDOW (dialog), "new_message");
 
-       gtk_widget_set_sensitive (dialog->button_chat, sensitive);
-       gtk_widget_set_sensitive (dialog->button_call, sensitive);
+  gtk_widget_set_sensitive (parent->button_action, FALSE);
 }
 
 static void
-new_message_dialog_destroy_cb (GtkWidget               *widget,
-                              EmpathyNewMessageDialog *dialog)
+empathy_new_message_dialog_class_init (
+  EmpathyNewMessageDialogClass *class)
 {
-       g_object_unref (dialog->contact_manager);
-       g_free (dialog);
+  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;
 }
 
 /**
- * empathy_new_message_dialog_show:
+ * empathy_new_message_dialog_new:
  * @parent: parent #GtkWindow of the dialog
  *
- * Create a new #EmpathyNewMessageDialog and show it.
+ * Create a new #EmpathyNewMessageDialog it.
  *
  * Return value: the new #EmpathyNewMessageDialog
  */
 GtkWidget *
 empathy_new_message_dialog_show (GtkWindow *parent)
 {
-       static EmpathyNewMessageDialog *dialog = NULL;
-       GtkBuilder                     *gui;
-       gchar                          *filename;
-       GtkEntryCompletion             *completion;
-       GtkListStore                   *model;
-
-       if (dialog) {
-               gtk_window_present (GTK_WINDOW (dialog->dialog));
-               return dialog->dialog;
-       }
-
-       dialog = g_new0 (EmpathyNewMessageDialog, 1);
-
-       /* create a contact manager */
-       dialog->contact_manager = empathy_contact_manager_dup_singleton ();
-
-       filename = empathy_file_lookup ("empathy-new-message-dialog.ui",
-                                       "libempathy-gtk");
-       gui = empathy_builder_get_file (filename,
-                                       "new_message_dialog", &dialog->dialog,
-                                       "table_contact", &dialog->table_contact,
-                                       "entry_id", &dialog->entry_id,
-                                       "button_chat", &dialog->button_chat,
-                                       "button_call",&dialog->button_call,
-                                       NULL);
-       g_free (filename);
-
-       /* text completion */
-       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,
-                                            new_message_dialog_match_func,
-                                            NULL, NULL);
-       gtk_entry_completion_set_model (completion, GTK_TREE_MODEL (model));
-       gtk_entry_set_completion (GTK_ENTRY (dialog->entry_id), completion);
-       g_signal_connect (completion, "match-selected",
-                         G_CALLBACK (new_message_dialog_match_selected_cb),
-                         dialog);
-       g_object_unref (completion);
-       g_object_unref (model);
-
-       empathy_builder_connect (gui, dialog,
-                              "new_message_dialog", "destroy", new_message_dialog_destroy_cb,
-                              "new_message_dialog", "response", new_message_dialog_response_cb,
-                              "entry_id", "changed", new_message_change_state_button_cb,
-                              NULL);
-
-       g_object_add_weak_pointer (G_OBJECT (dialog->dialog), (gpointer) &dialog);
-
-       g_object_unref (gui);
-
-       /* Create account chooser */
-       dialog->account_chooser = empathy_account_chooser_new ();
-       gtk_table_attach_defaults (GTK_TABLE (dialog->table_contact),
-                                  dialog->account_chooser,
-                                  1, 2, 0, 1);
-       empathy_account_chooser_set_filter (EMPATHY_ACCOUNT_CHOOSER (dialog->account_chooser),
-                                           empathy_account_chooser_filter_is_connected,
-                                           NULL);
-       gtk_widget_show (dialog->account_chooser);
-
-       new_message_dialog_account_changed_cb (dialog->account_chooser, dialog);
-       g_signal_connect (dialog->account_chooser, "changed",
-                         G_CALLBACK (new_message_dialog_account_changed_cb),
-                         dialog);
-
-       if (parent) {
-               gtk_window_set_transient_for (GTK_WINDOW (dialog->dialog),
-                                             GTK_WINDOW (parent));
-       }
-
-       gtk_widget_set_sensitive (dialog->button_chat, FALSE);
-       gtk_widget_set_sensitive (dialog->button_call, FALSE);
-
-       gtk_widget_show (dialog->dialog);
-
-       return dialog->dialog;
+  GtkWidget *dialog;
+
+  dialog = g_object_new (EMPATHY_TYPE_NEW_MESSAGE_DIALOG, NULL);
+
+  if (parent)
+    {
+      gtk_window_set_transient_for (GTK_WINDOW (dialog),
+          GTK_WINDOW (parent));
+    }
+
+  gtk_widget_show (dialog);
+  return dialog;
 }