-/* -*- 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;
}