]> git.0d.be Git - empathy.git/blobdiff - libempathy-gtk/empathy-new-message-dialog.c
Merge commit 'staz/dnd'
[empathy.git] / libempathy-gtk / empathy-new-message-dialog.c
index c7055017e4019ea3903ea9b55d835c69eb0bc9a3..338da276077a6bb224a2966e47bda0c60fdbb82f 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.
  *
@@ -27,7 +26,8 @@
 #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>
@@ -45,7 +45,7 @@
 static EmpathyNewMessageDialog *dialog_singleton = NULL;
 
 G_DEFINE_TYPE(EmpathyNewMessageDialog, empathy_new_message_dialog,
-                                      GTK_TYPE_DIALOG)
+               EMPATHY_TYPE_CONTACT_SELECTOR_DIALOG)
 
 /**
  * SECTION:empathy-new-message-dialog
@@ -53,353 +53,124 @@ G_DEFINE_TYPE(EmpathyNewMessageDialog, empathy_new_message_dialog,
  * @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 _EmpathyNewMessageDialogPriv EmpathyNewMessageDialogPriv;
-
-struct _EmpathyNewMessageDialogPriv {
-       GtkWidget *dialog;
-       GtkWidget *table_contact;
-       GtkWidget *account_chooser;
-       GtkWidget *entry_id;
-       GtkWidget *button_chat;
-       GtkWidget *button_call;
-       EmpathyContactManager *contact_manager;
-};
-
-#define GET_PRIV(o) \
-  (G_TYPE_INSTANCE_GET_PRIVATE ((o), EMPATHY_TYPE_NEW_MESSAGE_DIALOG, \
-    EmpathyNewMessageDialogPriv))
-
-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)
 {
-       EmpathyNewMessageDialogPriv *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);
-
-       /* Get members of the new account */
-       chooser = EMPATHY_ACCOUNT_CHOOSER (priv->account_chooser);
-       connection = empathy_account_chooser_get_connection (chooser);
-       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));
-
-       /* Add members to the completion */
-       while (members) {
-               EmpathyContact *contact = members->data;
-
-               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);
-       }
-}
+  TpConnection *connection;
+  const gchar *contact_id;
 
-static gboolean
-new_message_dialog_match_selected_cb (GtkEntryCompletion *widget,
-                                     GtkTreeModel       *model,
-                                     GtkTreeIter        *iter,
-                                     EmpathyNewMessageDialog *dialog)
-{
-       EmpathyNewMessageDialogPriv *priv = GET_PRIV (dialog);
-       gchar *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 (priv->entry_id), id);
+  contact_id = empathy_contact_selector_dialog_get_selected (
+      EMPATHY_CONTACT_SELECTOR_DIALOG (dialog), &connection);
 
-       DEBUG ("Got selected match **%s**", id);
+  if (EMP_STR_EMPTY (contact_id) || connection == NULL) goto out;
 
-       g_free (id);
+  empathy_dispatcher_chat_with_contact_id (connection, contact_id, NULL, NULL);
 
-       return TRUE;
+out:
+  gtk_widget_destroy (GTK_WIDGET (dialog));
 }
 
 static gboolean
-new_message_dialog_match_func (GtkEntryCompletion *completion,
-                              const gchar        *key,
-                              GtkTreeIter        *iter,
-                              gpointer            user_data)
+empathy_new_message_account_filter (EmpathyContactSelectorDialog *dialog,
+    TpAccount *account)
 {
-       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;
-}
+  TpConnection *connection;
+  EmpathyDispatcher *dispatcher;
+  GList *classes;
 
-static void
-new_message_dialog_call_got_contact_cb (EmpathyTpContactFactory *factory,
-                                       EmpathyContact          *contact,
-                                       const GError            *error,
-                                       gpointer                 user_data,
-                                       GObject                 *weak_object)
-{
-       EmpathyCallFactory *call_factory;
+  if (tp_account_get_connection_status (account, NULL) !=
+      TP_CONNECTION_STATUS_CONNECTED)
+    return FALSE;
 
-       if (error != NULL) {
-               DEBUG ("Error: %s", error->message);
-               return;
-       }
+  /* check if CM supports 1-1 text chat */
+  connection = tp_account_get_connection (account);
+  if (connection == NULL)
+    return FALSE;
 
-       call_factory = empathy_call_factory_get ();
-       empathy_call_factory_new_call (call_factory, contact);
-}
+  dispatcher = empathy_dispatcher_dup_singleton ();
 
-static void
-new_message_dialog_response_cb (GtkWidget               *widget,
-                               gint                    response,
-                               EmpathyNewMessageDialog *dialog)
-{
-       EmpathyNewMessageDialogPriv *priv = GET_PRIV (dialog);
-       TpConnection *connection;
-       const gchar *id;
-
-       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 == 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);
-}
+  classes = empathy_dispatcher_find_requestable_channel_classes
+    (dispatcher, connection, TP_IFACE_CHANNEL_TYPE_TEXT,
+     TP_HANDLE_TYPE_CONTACT, NULL);
 
-static void
-new_message_change_state_button_cb  (GtkEditable             *editable,
-                                    EmpathyNewMessageDialog *dialog)
-{
-       EmpathyNewMessageDialogPriv *priv = GET_PRIV (dialog);
-       const gchar *id;
-       gboolean     sensitive;
+  g_object_unref (dispatcher);
 
-       id = gtk_entry_get_text (GTK_ENTRY (editable));
-       sensitive = !EMP_STR_EMPTY (id);
+  if (classes == NULL)
+    return FALSE;
 
-       gtk_widget_set_sensitive (priv->button_chat, sensitive);
-       gtk_widget_set_sensitive (priv->button_call, sensitive);
+  g_list_free (classes);
+  return TRUE;
 }
 
 static GObject *
 empathy_new_message_dialog_constructor (GType type,
-                                       guint n_props,
-                                       GObjectConstructParam *props)
+    guint n_props,
+    GObjectConstructParam *props)
 {
-       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;
+  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
 empathy_new_message_dialog_init (EmpathyNewMessageDialog *dialog)
 {
-       EmpathyNewMessageDialogPriv *priv = GET_PRIV (dialog);
-       GtkBuilder                     *gui;
-       gchar                          *filename;
-       GtkEntryCompletion             *completion;
-       GtkListStore                   *model;
-       GtkWidget                      *content_area;
-       GtkWidget                      *image;
-
-       /* create a contact manager */
-       priv->contact_manager = empathy_contact_manager_dup_singleton ();
-
-       filename = empathy_file_lookup ("empathy-new-message-dialog.ui",
-                                       "libempathy-gtk");
-       gui = empathy_builder_get_file (filename,
-                                       "table_contact", &priv->table_contact,
-                                       "entry_id", &priv->entry_id,
-                                       NULL);
-       g_free (filename);
-
-       content_area = gtk_dialog_get_content_area (GTK_DIALOG (dialog));
-       gtk_container_add (GTK_CONTAINER (content_area), priv->table_contact);
-
-       /* add buttons */
-       gtk_dialog_add_button (GTK_DIALOG (dialog),
-               GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL);
-
-       priv->button_call = gtk_button_new_with_mnemonic (_("C_all"));
-       image = gtk_image_new_from_icon_name (EMPATHY_IMAGE_VOIP,
-               GTK_ICON_SIZE_BUTTON);
-       gtk_button_set_image (GTK_BUTTON (priv->button_call), image);
-
-       gtk_dialog_add_action_widget (GTK_DIALOG (dialog), priv->button_call, 1);
-       gtk_widget_show (priv->button_call);
-
-       priv->button_chat = 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 (priv->button_chat), image);
-
-       gtk_dialog_add_action_widget (GTK_DIALOG (dialog), priv->button_chat, 2);
-       gtk_widget_show (priv->button_chat);
-
-       /* Tweak the dialog */
-       gtk_dialog_set_has_separator (GTK_DIALOG (dialog), FALSE);
-
-       gtk_window_set_title (GTK_WINDOW (dialog), _("New Conversation"));
-       gtk_window_set_role (GTK_WINDOW (dialog), "new_message");
-       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);
-
-       /* 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 (priv->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);
-
-       g_signal_connect (dialog, "response",
-                   G_CALLBACK (new_message_dialog_response_cb), dialog);
-
-       empathy_builder_connect (gui, dialog,
-                              "entry_id", "changed", new_message_change_state_button_cb,
-                              NULL);
-
-       g_object_unref (gui);
-
-       /* Create account chooser */
-       priv->account_chooser = empathy_account_chooser_new ();
-       gtk_table_attach_defaults (GTK_TABLE (priv->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);
-       gtk_widget_show (priv->account_chooser);
-
-       new_message_dialog_account_changed_cb (priv->account_chooser, dialog);
-       g_signal_connect (priv->account_chooser, "changed",
-                         G_CALLBACK (new_message_dialog_account_changed_cb),
-                         dialog);
-
-       gtk_widget_set_sensitive (priv->button_chat, FALSE);
-       gtk_widget_set_sensitive (priv->button_call, FALSE);
-}
+  EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG (
+        dialog);
+  GtkWidget *image;
 
-static void
-empathy_new_message_dialog_dispose (GObject *object)
-{
-       EmpathyNewMessageDialogPriv *priv = GET_PRIV (object);
+  /* 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);
 
-       if (priv->contact_manager != NULL) {
-               g_object_unref (priv->contact_manager);
-               priv->contact_manager = NULL;
-       }
+  /* Tweak the dialog */
+  gtk_window_set_title (GTK_WINDOW (dialog), _("New Conversation"));
+  gtk_window_set_role (GTK_WINDOW (dialog), "new_message");
 
-       if (G_OBJECT_CLASS (empathy_new_message_dialog_parent_class)->dispose)
-               G_OBJECT_CLASS (empathy_new_message_dialog_parent_class)->dispose (object);
+  gtk_widget_set_sensitive (parent->button_action, FALSE);
 }
 
 static void
 empathy_new_message_dialog_class_init (
   EmpathyNewMessageDialogClass *class)
 {
-       GObjectClass *object_class = G_OBJECT_CLASS (class);
+  GObjectClass *object_class = G_OBJECT_CLASS (class);
+  GtkDialogClass *dialog_class = GTK_DIALOG_CLASS (class);
+  EmpathyContactSelectorDialogClass *selector_dialog_class = \
+    EMPATHY_CONTACT_SELECTOR_DIALOG_CLASS (class);
 
-       g_type_class_add_private (class, sizeof (EmpathyNewMessageDialogPriv));
+  object_class->constructor = empathy_new_message_dialog_constructor;
 
-       object_class->constructor = empathy_new_message_dialog_constructor;
+  dialog_class->response = empathy_new_message_dialog_response;
 
-       object_class->dispose = empathy_new_message_dialog_dispose;
+  selector_dialog_class->account_filter = empathy_new_message_account_filter;
 }
 
 /**
@@ -413,15 +184,16 @@ empathy_new_message_dialog_class_init (
 GtkWidget *
 empathy_new_message_dialog_show (GtkWindow *parent)
 {
-       GtkWidget *dialog;
+  GtkWidget *dialog;
 
-       dialog = g_object_new (EMPATHY_TYPE_NEW_MESSAGE_DIALOG, NULL);
+  dialog = g_object_new (EMPATHY_TYPE_NEW_MESSAGE_DIALOG, NULL);
 
-       if (parent) {
-               gtk_window_set_transient_for (GTK_WINDOW (dialog),
-                                             GTK_WINDOW (parent));
-       }
+  if (parent)
+    {
+      gtk_window_set_transient_for (GTK_WINDOW (dialog),
+          GTK_WINDOW (parent));
+    }
 
-       gtk_widget_show (dialog);
-       return dialog;
+  gtk_widget_show (dialog);
+  return dialog;
 }