aboutsummaryrefslogblamecommitdiffstats
path: root/libempathy-gtk/empathy-new-message-dialog.c
blob: 1fe935d4e78545efb415e970d256f6514663b9bf (plain) (tree)
1
2
  
                                         























                                                                             
                           
 

                                      
                                                  
                                               
                                          

                                     


                                        
                                            
                                          



                                       




                                                      


                                                                  
                                                    
 





                                                        

                                                                

   





                       
           
                                                                        
 
                     

                          
                                                   

                                                             
                                                                
 
                                                              
 














                                                                     


                                           

 













































                                                                                
                                                                         

                                                       


                           

                                                                


                                                         



                                      



                                                   



                                      
 



                                                                           

 
























































                                                                            

                                                   

                                 
 

















                                                                       

 

                                                                 
 

                                                                          
                    
                   
 









                                                            




                                                                    
 
                                                                           
                            
                                          
 

                                                                             





                                                                            
 


                                                                    
 
                                                          





                                       
                                                      

                                                              
                                                  
 
                                                                     
 


                                                                             
 
 










                                                   
                    
 
                                                                
 




                                                        
 

                           
 
/*
 * 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 <xclaesse@gmail.com>
 */

#include <config.h>

#include <string.h>
#include <stdlib.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-dispatcher.h>
#include <libempathy/empathy-utils.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
 * 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;
}