/* * Copyright (C) 2007-2008 Collabora Ltd. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * * Authors: Xavier Claessens */ #include #include #include #include #include #include #include #include #include #include #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT #include #include #include #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 * to be started with any contact on any enabled account. */ enum { EMP_NEW_MESSAGE_TEXT, EMP_NEW_MESSAGE_SMS, }; static void empathy_new_message_dialog_response (GtkDialog *dialog, int response_id) { 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 (EMP_STR_EMPTY (contact_id) || account == NULL) goto out; switch (response_id) { case EMP_NEW_MESSAGE_TEXT: empathy_dispatcher_chat_with_contact_id (account, contact_id, gtk_get_current_event_time ()); break; case EMP_NEW_MESSAGE_SMS: empathy_dispatcher_sms_contact_id (account, contact_id, gtk_get_current_event_time ()); break; default: g_warn_if_reached (); } out: gtk_widget_destroy (GTK_WIDGET (dialog)); } static void conn_prepared_cb (GObject *conn, GAsyncResult *result, gpointer user_data) { FilterCallbackData *data = user_data; GError *myerr = NULL; TpCapabilities *caps; GPtrArray *classes; guint i; if (!tp_proxy_prepare_finish (conn, result, &myerr)) goto out; caps = tp_connection_get_capabilities (TP_CONNECTION (conn)); classes = tp_capabilities_get_channel_classes (caps); for (i = 0; i < classes->len; i++) { GHashTable *fixed; GStrv allowed; const gchar *chan_type; tp_value_array_unpack (g_ptr_array_index (classes, i), 2, &fixed, &allowed); chan_type = tp_asv_get_string (fixed, TP_PROP_CHANNEL_CHANNEL_TYPE); if (tp_strdiff (chan_type, TP_IFACE_CHANNEL_TYPE_TEXT)) continue; if (tp_asv_get_uint32 (fixed, TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, NULL) != TP_HANDLE_TYPE_CONTACT) continue; data->callback (TRUE, data->user_data); g_slice_free (FilterCallbackData, data); return; } out: data->callback (FALSE, data->user_data); g_slice_free (FilterCallbackData, data); } static void 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 void empathy_new_message_dialog_update_sms_button_sensitivity (GtkWidget *widget, GParamSpec *pspec, GtkWidget *button) { GtkWidget *self = gtk_widget_get_toplevel (widget); EmpathyContactSelectorDialog *dialog; TpConnection *conn; GPtrArray *rccs; gboolean sensitive = FALSE; guint i; 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; empathy_contact_selector_dialog_get_selected (dialog, &conn, NULL); if (conn == NULL) goto finally; /* 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)); for (i = 0; i < rccs->len; i++) { GHashTable *fixed; GStrv allowed; const char *type; gboolean sms_channel; tp_value_array_unpack (g_ptr_array_index (rccs, i), 2, &fixed, &allowed); /* 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); sensitive = sms_channel && !tp_strdiff (type, TP_IFACE_CHANNEL_TYPE_TEXT); if (sensitive) break; } finally: gtk_widget_set_sensitive (button, sensitive); } static GObject * empathy_new_message_dialog_constructor (GType type, 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; } static void empathy_new_message_dialog_init (EmpathyNewMessageDialog *dialog) { EmpathyContactSelectorDialog *parent = EMPATHY_CONTACT_SELECTOR_DIALOG ( dialog); GtkWidget *button; GtkWidget *image; /* add an SMS button */ button = 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); /* 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, EMP_NEW_MESSAGE_TEXT); gtk_widget_show (parent->button_action); /* 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); /* 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 (parent->button_action, FALSE); } static void empathy_new_message_dialog_class_init ( EmpathyNewMessageDialogClass *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); 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_new: * @parent: parent #GtkWindow of the dialog * * Create a new #EmpathyNewMessageDialog it. * * Return value: the new #EmpathyNewMessageDialog */ GtkWidget * empathy_new_message_dialog_show (GtkWindow *parent) { 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; }