aboutsummaryrefslogblamecommitdiffstats
path: root/libempathy-gtk/empathy-ui-utils.c
blob: 34e74c8fd3142ebc172e2682305be1e882ea65a8 (plain) (tree)
1
2
3

                                     
                                         












                                                                     


                                                               


                                                
                                                 
                                                   
                                                           
  




                                                                             
                   
                             
 

                      
                           
                                
                                
                                                
                                                 
                                          
 
                               
                           
                          
 
                                      
                          
 

                               

                       
 
                                      
 

                  
 
                  
 

                                                                   
 






                                                                   
 

                                                                                
 



                     

 
             
                                                                  
 


































                                                                                

 
             
                                                       
 
                                    
 

                                                     
 

                                                    

 
             

                                                              

                                    
 


                                                                
 
                                                   


             
                                                           
 
                     
 
                                                            
 


                                                  
 
                                            

 




                                 

  

                                                             













































                                                                                 

 


                                                  





































                                                   




                                                

























                                                   

 
                  
                                        
 
                    


                                         
                                                           





                                          
                 
                


      
                                     

     

                                            
 










                                                   

 
                  
                                                         

                
 



                          
 

                
 


                                    
 
                                    
 

                                                               
 













                                                                                
 

                                              
 
                          
 
                


           
                                                                   

                
 
                        
 
                                                            
 
                                                
 
                                                                   
 
 

              


                             
                            

                                    
                                          










                                                                       
                                                             







                                                      

 

                                            
                                               
 
                                         
                                   
                                                            

 






                                    
 
                    
 

                                         
 
                


           


                                     
 



                                                         

                          
 






                                                                    
 

                                                                           
 
                          
 





                                                                    
 
                                           
 


                                                             

    
                                                   
 
                         
                                                       


    
                                                    



































                                                                               

 
                                   

                                                     


                                
 


                                                              
 

                                                                 
 

                                                            
 


                                                          
 



                                                              

 
           
                                                            
                           
 


                                                            
 
                                                      
 

                        
 

                                                                    

 




                                                                

                                                                           















                                                       
                                                          









































                                                                        

 
                  
                                                                     

                
 




                                                            
 
                                                  
                                           
                                                              
 




                                                                                
 
                

 













                            
    
                                    
                      
 

                       
 

                                                              
 
                                          






                                                                         
 


                                               
 

                                                                            
 

                                                   
 
                                               
 

                             
 
                    

 

                                   
                
 


                            
 

                                                  
 
                                                
 

                                                                   
 



                                              
 
                           

 



























































































































































































































                                                                           
    





























                                                                         

 
           


                                                      
 
              


                         
 
                                                                             

                                                                   
 





                                                                         
                                        
 

                            
 

                                           

 

                                                
                  
 

                                                             




                         
                        
 
                                  
 

                                                                           
 
                

 
    
                                                             
 


                    






                          


                                                  
 

                                                                 

                                       
            
 





                                                                
 
                                                            
                             
 

                                                       
                             
 
                                                                     
 

                                                                 
 

                                                         
 













                                                                       

                                                                               



                                                              
 
                           

                            
 
 

                                                         

                              
 

                            
 





                                    
 


                                      



                                                                   
 
                              
 


                                                  
 


                                
 


                                                               
 
                            
 






                                                                           
 

                                                      
 





                                                             
 
                                                
 


                                       
 
                                
 

                 
 












                                                                               
                                                    
 

                                                               
 









                                                             
 
                                           




                                                                  
                    
                              


                          








                            












                                                               
                                                          










                                                                       























                                                                             

                                                                   





                                                              


                           


                            
 



                                          
                                               
 


                                                  
 


                                  
                     
 


                                                                       
 
                         







                                                                             
                  
 
                         
 
                                                               
 






                                                                                
 
              
 





                                                                        
 
                                                       


                                                                             
        

                                                             


                     


                          



                                                                         
                                                




                                                        

                                                         
     
                                                      
                     
 





                                                                                
                                           











                                                         
                                                                  





                                
     
                         


                                                                
                
 
 


















































                                                                        










                                                                       

              














































                                                                                
                                   























































                                                                
































                                                                               


























                                                                        
                      



                                
                     










                                                           











                                                                                
 
















                                                                         

                
/*
 * Copyright (C) 2002-2007 Imendio AB
 * Copyright (C) 2007-2010 Collabora Ltd.
 *
 * This program is free software; you can redistribute it and/or
 * modify it under the terms of the GNU General Public License as
 * published by the Free Software Foundation; either version 2 of the
 * License, or (at your option) any later version.
 *
 * This program 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
 * General Public License for more details.
 *
 * You should have received a copy of the GNU General Public
 * License along with this program; if not, write to the
 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
 * Boston, MA  02110-1301  USA
 *
 * Authors: Mikael Hallendal <micke@imendio.com>
 *          Richard Hult <richard@imendio.com>
 *          Martyn Russell <martyn@imendio.com>
 *          Xavier Claessens <xclaesse@gmail.com>
 *          Jonny Lamb <jonny.lamb@collabora.co.uk>
 *          Travis Reitter <travis.reitter@collabora.co.uk>
 *
 *          Part of this file is copied from GtkSourceView (gtksourceiter.c):
 *          Paolo Maggi
 *          Jeroen Zwartepoorte
 */

#include "config.h"
#include "empathy-ui-utils.h"

#include <X11/Xatom.h>
#include <gdk/gdkx.h>
#include <glib/gi18n-lib.h>
#include <gio/gdesktopappinfo.h>
#include <gnome-autoar/autoar.h>
#include <tp-account-widgets/tpaw-live-search.h>
#include <tp-account-widgets/tpaw-pixbuf-utils.h>
#include <tp-account-widgets/tpaw-utils.h>

#include "empathy-ft-factory.h"
#include "empathy-images.h"
#include "empathy-utils.h"

#define DEBUG_FLAG EMPATHY_DEBUG_OTHER
#include "empathy-debug.h"

#define EMPATHY_RESPONSE_SEND 1

void
empathy_gtk_init (void)
{
  static gboolean initialized = FALSE;

  if (initialized)
    return;

  empathy_init ();

  gtk_icon_theme_append_search_path (gtk_icon_theme_get_default (),
      PKGDATADIR G_DIR_SEPARATOR_S "icons");

  /* Add icons from source dir if available */
  if (g_getenv ("EMPATHY_SRCDIR") != NULL)
    {
      gchar *path;

      path = g_build_filename (g_getenv ("EMPATHY_SRCDIR"), "data",
          "icons", "local-copy", NULL);

      if (g_file_test (path, G_FILE_TEST_EXISTS))
        gtk_icon_theme_append_search_path (gtk_icon_theme_get_default (), path);

      g_free (path);
    }

  initialized = TRUE;
}

const gchar *
empathy_icon_name_for_presence (TpConnectionPresenceType presence)
{
  switch (presence)
    {
      case TP_CONNECTION_PRESENCE_TYPE_AVAILABLE:
        return EMPATHY_IMAGE_AVAILABLE;
      case TP_CONNECTION_PRESENCE_TYPE_BUSY:
        return EMPATHY_IMAGE_BUSY;
      case TP_CONNECTION_PRESENCE_TYPE_AWAY:
        return EMPATHY_IMAGE_AWAY;
      case TP_CONNECTION_PRESENCE_TYPE_EXTENDED_AWAY:
        if (gtk_icon_theme_has_icon (gtk_icon_theme_get_default (),
                   EMPATHY_IMAGE_EXT_AWAY))
          return EMPATHY_IMAGE_EXT_AWAY;

        /* The 'extended-away' icon is not an official one so we fallback to
         * idle if it's not implemented */
        return EMPATHY_IMAGE_IDLE;
      case TP_CONNECTION_PRESENCE_TYPE_HIDDEN:
        if (gtk_icon_theme_has_icon (gtk_icon_theme_get_default (),
                   EMPATHY_IMAGE_HIDDEN))
          return EMPATHY_IMAGE_HIDDEN;

        /* The 'hidden' icon is not an official one so we fallback to offline if
         * it's not implemented */
        return EMPATHY_IMAGE_OFFLINE;
      case TP_CONNECTION_PRESENCE_TYPE_OFFLINE:
      case TP_CONNECTION_PRESENCE_TYPE_ERROR:
        return EMPATHY_IMAGE_OFFLINE;
      case TP_CONNECTION_PRESENCE_TYPE_UNKNOWN:
        return EMPATHY_IMAGE_PENDING;
      case TP_CONNECTION_PRESENCE_TYPE_UNSET:
      default:
        return NULL;
    }

  return NULL;
}

const gchar *
empathy_icon_name_for_contact (EmpathyContact *contact)
{
  TpConnectionPresenceType presence;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact),
      EMPATHY_IMAGE_OFFLINE);

  presence = empathy_contact_get_presence (contact);
  return empathy_icon_name_for_presence (presence);
}

const gchar *
empathy_icon_name_for_individual (FolksIndividual *individual)
{
  FolksPresenceType folks_presence;
  TpConnectionPresenceType presence;

  folks_presence = folks_presence_details_get_presence_type (
      FOLKS_PRESENCE_DETAILS (individual));
  presence = empathy_folks_presence_type_to_tp (folks_presence);

  return empathy_icon_name_for_presence (presence);
}

const gchar *
empathy_protocol_name_for_contact (EmpathyContact *contact)
{
  TpAccount *account;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);

  account = empathy_contact_get_account (contact);
  if (account == NULL)
    return NULL;

  return tp_account_get_icon_name (account);
}

struct SizeData
{
  gint width;
  gint height;
  gboolean preserve_aspect_ratio;
};

static void
pixbuf_from_avatar_size_prepared_cb (GdkPixbufLoader *loader,
    int width,
    int height,
    struct SizeData *data)
{
  g_return_if_fail (width > 0 && height > 0);

  if (data->preserve_aspect_ratio && (data->width > 0 || data->height > 0))
    {
      if (width < data->width && height < data->height)
        {
          width = width;
          height = height;
        }

      if (data->width < 0)
        {
          width = width * (double) data->height / (gdouble) height;
          height = data->height;
        }
      else if (data->height < 0)
        {
          height = height * (double) data->width / (double) width;
          width = data->width;
        }
      else if ((double) height * (double) data->width >
           (double) width * (double) data->height)
        {
          width = 0.5 + (double) width * (double) data->height / (double) height;
          height = data->height;
        }
      else
        {
          height = 0.5 + (double) height * (double) data->width / (double) width;
          width = data->width;
        }
    }
  else
    {
      if (data->width > 0)
        width = data->width;

      if (data->height > 0)
        height = data->height;
    }

  gdk_pixbuf_loader_set_size (loader, width, height);
}

static void
empathy_avatar_pixbuf_roundify (GdkPixbuf *pixbuf)
{
  gint width, height, rowstride;
  guchar *pixels;

  width = gdk_pixbuf_get_width (pixbuf);
  height = gdk_pixbuf_get_height (pixbuf);
  rowstride = gdk_pixbuf_get_rowstride (pixbuf);
  pixels = gdk_pixbuf_get_pixels (pixbuf);

  if (width < 6 || height < 6)
    return;

  /* Top left */
  pixels[3] = 0;
  pixels[7] = 0x80;
  pixels[11] = 0xC0;
  pixels[rowstride + 3] = 0x80;
  pixels[rowstride * 2 + 3] = 0xC0;

  /* Top right */
  pixels[width * 4 - 1] = 0;
  pixels[width * 4 - 5] = 0x80;
  pixels[width * 4 - 9] = 0xC0;
  pixels[rowstride + (width * 4) - 1] = 0x80;
  pixels[(2 * rowstride) + (width * 4) - 1] = 0xC0;

  /* Bottom left */
  pixels[(height - 1) * rowstride + 3] = 0;
  pixels[(height - 1) * rowstride + 7] = 0x80;
  pixels[(height - 1) * rowstride + 11] = 0xC0;
  pixels[(height - 2) * rowstride + 3] = 0x80;
  pixels[(height - 3) * rowstride + 3] = 0xC0;

  /* Bottom right */
  pixels[height * rowstride - 1] = 0;
  pixels[(height - 1) * rowstride - 1] = 0x80;
  pixels[(height - 2) * rowstride - 1] = 0xC0;
  pixels[height * rowstride - 5] = 0x80;
  pixels[height * rowstride - 9] = 0xC0;
}

static gboolean
empathy_gdk_pixbuf_is_opaque (GdkPixbuf *pixbuf)
{
  gint height, rowstride, i;
  guchar *pixels;
  guchar *row;

  height = gdk_pixbuf_get_height (pixbuf);
  rowstride = gdk_pixbuf_get_rowstride (pixbuf);
  pixels = gdk_pixbuf_get_pixels (pixbuf);

  row = pixels;
  for (i = 3; i < rowstride; i+=4)
    if (row[i] < 0xfe)
      return FALSE;

  for (i = 1; i < height - 1; i++)
    {
      row = pixels + (i*rowstride);
      if (row[3] < 0xfe || row[rowstride-1] < 0xfe)
        return FALSE;
    }

  row = pixels + ((height-1) * rowstride);
  for (i = 3; i < rowstride; i+=4)
    if (row[i] < 0xfe)
      return FALSE;

  return TRUE;
}

static GdkPixbuf *
pixbuf_round_corners (GdkPixbuf *pixbuf)
{
  GdkPixbuf *result;

  if (!gdk_pixbuf_get_has_alpha (pixbuf))
    {
      result = gdk_pixbuf_new (GDK_COLORSPACE_RGB, TRUE, 8,
          gdk_pixbuf_get_width (pixbuf),
          gdk_pixbuf_get_height (pixbuf));

      gdk_pixbuf_copy_area (pixbuf, 0, 0,
          gdk_pixbuf_get_width (pixbuf),
          gdk_pixbuf_get_height (pixbuf),
          result,
          0, 0);
    }
  else
    {
      result = g_object_ref (pixbuf);
    }

  if (empathy_gdk_pixbuf_is_opaque (result))
    empathy_avatar_pixbuf_roundify (result);

  return result;
}

static GdkPixbuf *
avatar_pixbuf_from_loader (GdkPixbufLoader *loader)
{
  GdkPixbuf *pixbuf;

  pixbuf = gdk_pixbuf_loader_get_pixbuf (loader);

  return pixbuf_round_corners (pixbuf);
}

static GdkPixbuf *
empathy_pixbuf_from_avatar_scaled (EmpathyAvatar *avatar,
    gint width,
    gint height)
{
  GdkPixbuf *pixbuf;
  GdkPixbufLoader *loader;
  struct SizeData data;
  GError *error = NULL;

  if (!avatar)
    return NULL;

  data.width = width;
  data.height = height;
  data.preserve_aspect_ratio = TRUE;

  loader = gdk_pixbuf_loader_new ();

  g_signal_connect (loader, "size-prepared",
      G_CALLBACK (pixbuf_from_avatar_size_prepared_cb), &data);

  if (avatar->len == 0)
    {
      g_warning ("Avatar has 0 length");
      return NULL;
    }
  else if (!gdk_pixbuf_loader_write (loader, avatar->data, avatar->len, &error))
    {
      g_warning ("Couldn't write avatar image:%p with "
          "length:%" G_GSIZE_FORMAT " to pixbuf loader: %s",
          avatar->data, avatar->len, error->message);

      g_error_free (error);
      return NULL;
    }

  gdk_pixbuf_loader_close (loader, NULL);
  pixbuf = avatar_pixbuf_from_loader (loader);

  g_object_unref (loader);

  return pixbuf;
}

GdkPixbuf *
empathy_pixbuf_avatar_from_contact_scaled (EmpathyContact *contact,
    gint width,
    gint height)
{
  EmpathyAvatar *avatar;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);

  avatar = empathy_contact_get_avatar (contact);

  return empathy_pixbuf_from_avatar_scaled (avatar, width, height);
}

typedef struct
{
  GSimpleAsyncResult *result;
  guint width;
  guint height;
  GCancellable *cancellable;
} PixbufAvatarFromIndividualClosure;

static PixbufAvatarFromIndividualClosure *
pixbuf_avatar_from_individual_closure_new (FolksIndividual *individual,
    GSimpleAsyncResult *result,
    gint width,
    gint height,
    GCancellable *cancellable)
{
  PixbufAvatarFromIndividualClosure *closure;

  g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
  g_return_val_if_fail (G_IS_ASYNC_RESULT (result), NULL);

  closure = g_slice_new0 (PixbufAvatarFromIndividualClosure);
  closure->result = g_object_ref (result);
  closure->width = width;
  closure->height = height;

  if (cancellable != NULL)
    closure->cancellable = g_object_ref (cancellable);

  return closure;
}

static void
pixbuf_avatar_from_individual_closure_free (
    PixbufAvatarFromIndividualClosure *closure)
{
  g_clear_object (&closure->cancellable);
  g_object_unref (closure->result);
  g_slice_free (PixbufAvatarFromIndividualClosure, closure);
}

/**
 * @pixbuf: (transfer all)
 *
 * Return: (transfer all)
 */
static GdkPixbuf *
transform_pixbuf (GdkPixbuf *pixbuf)
{
  GdkPixbuf *result;

  result = pixbuf_round_corners (pixbuf);
  g_object_unref (pixbuf);

  return result;
}

static void
avatar_icon_load_cb (GObject *object,
    GAsyncResult *result,
    gpointer user_data)
{
  GLoadableIcon *icon = G_LOADABLE_ICON (object);
  PixbufAvatarFromIndividualClosure *closure = user_data;
  GInputStream *stream;
  GError *error = NULL;
  GdkPixbuf *pixbuf;
  GdkPixbuf *final_pixbuf;

  stream = g_loadable_icon_load_finish (icon, result, NULL, &error);
  if (error != NULL)
    {
      DEBUG ("Failed to open avatar stream: %s", error->message);
      g_simple_async_result_set_from_error (closure->result, error);
      goto out;
    }

  pixbuf = gdk_pixbuf_new_from_stream_at_scale (stream,
      closure->width, closure->height, TRUE, closure->cancellable, &error);

  g_object_unref (stream);

  if (pixbuf == NULL)
    {
      DEBUG ("Failed to read avatar: %s", error->message);
      g_simple_async_result_set_from_error (closure->result, error);
      goto out;
    }

  final_pixbuf = transform_pixbuf (pixbuf);

  /* Pass ownership of final_pixbuf to the result */
  g_simple_async_result_set_op_res_gpointer (closure->result,
      final_pixbuf, g_object_unref);

out:
  g_simple_async_result_complete (closure->result);

  g_clear_error (&error);
  pixbuf_avatar_from_individual_closure_free (closure);
}

void
empathy_pixbuf_avatar_from_individual_scaled_async (
    FolksIndividual *individual,
    gint width,
    gint height,
    GCancellable *cancellable,
    GAsyncReadyCallback callback,
    gpointer user_data)
{
  GLoadableIcon *avatar_icon;
  GSimpleAsyncResult *result;
  PixbufAvatarFromIndividualClosure *closure;

  result = g_simple_async_result_new (G_OBJECT (individual),
      callback, user_data, empathy_pixbuf_avatar_from_individual_scaled_async);

  avatar_icon = folks_avatar_details_get_avatar (
      FOLKS_AVATAR_DETAILS (individual));

  if (avatar_icon == NULL)
    {
      g_simple_async_result_set_error (result, G_IO_ERROR,
        G_IO_ERROR_NOT_FOUND, "no avatar found");

      g_simple_async_result_complete (result);
      g_object_unref (result);
      return;
    }

  closure = pixbuf_avatar_from_individual_closure_new (individual, result,
      width, height, cancellable);

  g_return_if_fail (closure != NULL);

  g_loadable_icon_load_async (avatar_icon, width, cancellable,
      avatar_icon_load_cb, closure);

  g_object_unref (result);
}

/* Return a ref on the GdkPixbuf */
GdkPixbuf *
empathy_pixbuf_avatar_from_individual_scaled_finish (
    FolksIndividual *individual,
    GAsyncResult *result,
    GError **error)
{
  GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (result);
  gboolean result_valid;
  GdkPixbuf *pixbuf;

  g_return_val_if_fail (FOLKS_IS_INDIVIDUAL (individual), NULL);
  g_return_val_if_fail (G_IS_SIMPLE_ASYNC_RESULT (simple), NULL);

  if (g_simple_async_result_propagate_error (simple, error))
    return NULL;

  result_valid = g_simple_async_result_is_valid (result,
      G_OBJECT (individual),
      empathy_pixbuf_avatar_from_individual_scaled_async);

  g_return_val_if_fail (result_valid, NULL);

  pixbuf = g_simple_async_result_get_op_res_gpointer (simple);
  return pixbuf != NULL ? g_object_ref (pixbuf) : NULL;
}

GdkPixbuf *
empathy_pixbuf_contact_status_icon (EmpathyContact *contact,
    gboolean show_protocol)
{
  const gchar *icon_name;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);

  icon_name = empathy_icon_name_for_contact (contact);

  if (icon_name == NULL)
    return NULL;

  return empathy_pixbuf_contact_status_icon_with_icon_name (contact,
      icon_name, show_protocol);
}

static GdkPixbuf * empathy_pixbuf_protocol_from_contact_scaled (
    EmpathyContact *contact,
    gint width,
    gint height);

GdkPixbuf *
empathy_pixbuf_contact_status_icon_with_icon_name (EmpathyContact *contact,
    const gchar *icon_name,
    gboolean show_protocol)
{
  GdkPixbuf *pix_status;
  GdkPixbuf *pix_protocol;
  gchar *icon_filename;
  gint height, width;
  gint numerator, denominator;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact) ||
      (show_protocol == FALSE), NULL);
  g_return_val_if_fail (icon_name != NULL, NULL);

  numerator = 3;
  denominator = 4;

  icon_filename = tpaw_filename_from_icon_name (icon_name,
      GTK_ICON_SIZE_MENU);

  if (icon_filename == NULL)
    {
      DEBUG ("icon name: %s could not be found\n", icon_name);
      return NULL;
    }

  pix_status = gdk_pixbuf_new_from_file (icon_filename, NULL);

  if (pix_status == NULL)
    {
      DEBUG ("Could not open icon %s\n", icon_filename);
      g_free (icon_filename);
      return NULL;
    }

  g_free (icon_filename);

  if (!show_protocol)
    return pix_status;

  height = gdk_pixbuf_get_height (pix_status);
  width = gdk_pixbuf_get_width (pix_status);

  pix_protocol = empathy_pixbuf_protocol_from_contact_scaled (contact,
      width * numerator / denominator,
      height * numerator / denominator);

  if (pix_protocol == NULL)
    return pix_status;

  gdk_pixbuf_composite (pix_protocol, pix_status,
      0, height - height * numerator / denominator,
      width * numerator / denominator, height * numerator / denominator,
      0, height - height * numerator / denominator,
      1, 1,
      GDK_INTERP_BILINEAR, 255);

  g_object_unref (pix_protocol);

  return pix_status;
}

static GdkPixbuf *
empathy_pixbuf_protocol_from_contact_scaled (EmpathyContact *contact,
    gint width,
    gint height)
{
  TpAccount *account;
  gchar *filename;
  GdkPixbuf *pixbuf = NULL;

  g_return_val_if_fail (EMPATHY_IS_CONTACT (contact), NULL);

  account = empathy_contact_get_account (contact);
  filename = tpaw_filename_from_icon_name (
      tp_account_get_icon_name (account), GTK_ICON_SIZE_MENU);

  if (filename != NULL)
    {
      pixbuf = gdk_pixbuf_new_from_file_at_size (filename, width, height, NULL);
      g_free (filename);
    }

  return pixbuf;
}

struct SendData
{
  GFile *file;
  GFile *destination;
  GCancellable *cancellable;
  EmpathyContact *contact;
  GtkWidget *dialog;
  GtkProgressBar *progress;
  GtkLabel *progress_label;
  GtkLabel *filename_label;
  gboolean dialog_shown;
  gint64 last_notify;
};

void
empathy_url_show (GtkWidget *parent,
      const char *url)
{
  gchar  *real_url;
  GError *error = NULL;

  g_return_if_fail (parent == NULL || GTK_IS_WIDGET (parent));
  g_return_if_fail (url != NULL);

  real_url = tpaw_make_absolute_url (url);

  gtk_show_uri (parent ? gtk_widget_get_screen (parent) : NULL, real_url,
      gtk_get_current_event_time (), &error);

  if (error)
    {
      GtkWidget *dialog;

      dialog = gtk_message_dialog_new (NULL, 0,
          GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
          _("Unable to open URI"));

      gtk_message_dialog_format_secondary_text (GTK_MESSAGE_DIALOG (dialog),
          "%s", error->message);

      g_signal_connect (dialog, "response",
            G_CALLBACK (gtk_widget_destroy), NULL);

      gtk_window_present (GTK_WINDOW (dialog));

      g_clear_error (&error);
    }

  g_free (real_url);
}

static void
send_file (EmpathyContact *contact,
    GFile *file)
{
  EmpathyFTFactory *factory;
  GtkRecentManager *manager;
  gchar *uri;

  g_return_if_fail (EMPATHY_IS_CONTACT (contact));
  g_return_if_fail (G_IS_FILE (file));

  factory = empathy_ft_factory_dup_singleton ();

  empathy_ft_factory_new_transfer_outgoing (factory, contact, file,
      empathy_get_current_action_time ());

  uri = g_file_get_uri (file);
  manager = gtk_recent_manager_get_default ();
  gtk_recent_manager_add_item (manager, uri);
  g_free (uri);

  g_object_unref (factory);
}

static void
send_data_free (struct SendData *send_data)
{
  g_clear_object (&(send_data->file));
  g_clear_object (&(send_data->destination));
  g_clear_object (&(send_data->contact));
  g_clear_object (&(send_data->cancellable));
  if (send_data->dialog != NULL)
    gtk_widget_destroy (send_data->dialog);
  g_free (send_data);
}

static void
send_file_autoar_decide_dest_cb (AutoarCreate *arcreate,
    GFile *destination,
    struct SendData *send_data)
{
  send_data->destination = g_object_ref (destination);
}

static void
send_file_autoar_dialog_response_cb (GtkDialog *dialog,
    gint response_id,
    struct SendData *send_data)
{
  if (response_id == GTK_RESPONSE_CANCEL)
    g_cancellable_cancel (send_data->cancellable);
}

static void
send_file_autoar_create_dialog (struct SendData *send_data)
{
  GtkWidget *content;
  GtkBox *vbox;
  char *str, *filename, *destname;

  send_data->dialog = gtk_dialog_new_with_buttons (
      _("Creating Archives…"), NULL,
      GTK_DIALOG_DESTROY_WITH_PARENT,
      _("Cancel"), GTK_RESPONSE_CANCEL,
      NULL);
  send_data->progress = GTK_PROGRESS_BAR (gtk_progress_bar_new ());
  send_data->progress_label = GTK_LABEL (gtk_label_new (NULL));

  str = g_strdup_printf ("%s -> %s",
      filename = autoar_common_g_file_get_name (send_data->file),
      destname = autoar_common_g_file_get_name (send_data->destination));
  send_data->filename_label = GTK_LABEL (gtk_label_new (str));

  vbox = GTK_BOX (gtk_box_new (GTK_ORIENTATION_VERTICAL, 5));
  gtk_box_pack_start (vbox, GTK_WIDGET (send_data->filename_label),
      FALSE, FALSE, 0);
  gtk_box_pack_start (vbox, GTK_WIDGET (send_data->progress),
      FALSE, FALSE, 0);
  gtk_box_pack_start (vbox, GTK_WIDGET (send_data->progress_label),
      FALSE, FALSE, 0);
  content = gtk_dialog_get_content_area (GTK_DIALOG (send_data->dialog));
  gtk_container_add (GTK_CONTAINER (content), GTK_WIDGET (vbox));

  g_signal_connect (send_data->dialog, "response",
      G_CALLBACK (send_file_autoar_dialog_response_cb), send_data);

  g_free (str);
  g_free (filename);
  g_free (destname);
}

static void
send_file_autoar_progress_cb (AutoarCreate *arcreate,
    guint64 completed_size,
    guint completed_files,
    struct SendData *send_data)
{
  gint64 mtime;

  DEBUG ("%s: size = %" G_GUINT64_FORMAT ", files = %u",
     __func__, completed_size, completed_files);

  mtime = g_get_monotonic_time ();
  if (mtime - send_data->last_notify < 200000)
    return;

  send_data->last_notify = mtime;

  if (!(send_data->dialog_shown))
    {
      if (send_data->dialog == NULL)
        send_file_autoar_create_dialog (send_data);
      gtk_widget_show_all (send_data->dialog);
      send_data->dialog_shown = TRUE;
    }

  if (send_data->progress != NULL)
    gtk_progress_bar_pulse (send_data->progress);

  if (send_data->progress_label != NULL)
    {
      gchar *msg, *size;
      msg = g_strdup_printf (
          _("%u files (%s) are archived"),
          completed_files, size = g_format_size (completed_size));
      gtk_label_set_text (send_data->progress_label, msg);
      g_free (msg);
      g_free (size);
    }
}

static void
send_file_autoar_cancelled_cb (AutoarCreate *arcreate,
    struct SendData *send_data)
{
  send_data_free (send_data);
}

static void
send_file_autoar_completed_cb (AutoarCreate *arcreate,
    struct SendData *send_data)
{
  send_file (send_data->contact, send_data->destination);
  send_data_free (send_data);
}

static void
send_file_autoar_error_cb (AutoarCreate *arcreate,
    GError *error,
    struct SendData *send_data)
{
  GtkWidget *error_dialog;
  char *filename;

  error_dialog = gtk_message_dialog_new_with_markup (
      NULL, GTK_DIALOG_DESTROY_WITH_PARENT,
      GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE,
      "<big><b>%s</b></big>",
      filename = autoar_common_g_file_get_name (send_data->file));
  gtk_message_dialog_format_secondary_text (
      GTK_MESSAGE_DIALOG (error_dialog), "%s", error->message);
  gtk_widget_show_all (error_dialog);
  g_signal_connect_swapped (error_dialog, "response",
      G_CALLBACK (gtk_widget_destroy), error_dialog);

  send_data_free (send_data);
  g_free (filename);
}

static void
send_file_info_ready_cb (GFile *file,
    GAsyncResult *res,
    EmpathyContact *contact)
{
  GFileInfo *file_info;
  GFileType file_type;

  file_info = g_file_query_info_finish (file, res, NULL);
  if (file_info == NULL)
    return;

  file_type = g_file_info_get_file_type (file_info);
  g_object_unref (file_info);

  if (file_type == G_FILE_TYPE_DIRECTORY)
    {
      struct SendData *send_data;
      AutoarPref *arpref;
      AutoarCreate *arcreate;
      gchar *template;
      GFile *archive_dir;
      gchar *archive_dir_path;

      arpref = g_object_get_data (G_OBJECT (contact), "autoar-pref");
      template = g_strdup_printf ("empathy-%s-XXXXXX", g_get_user_name ());
      archive_dir_path = g_dir_make_tmp (template, NULL);
      if (archive_dir_path == NULL)
        {
          g_free (template);
          return;
        }

      DEBUG ("tmpdir = %s", archive_dir_path);
      archive_dir = g_file_new_for_path (archive_dir_path);
      arcreate = autoar_create_new_file (arpref, archive_dir, file, NULL);
      send_data = g_new0 (struct SendData, 1);
      send_data->contact = g_object_ref (contact);
      send_data->file = g_object_ref (file);
      send_data->cancellable = g_cancellable_new ();
      send_data->last_notify = g_get_monotonic_time ();

      g_signal_connect (arcreate, "decide-dest",
          G_CALLBACK (send_file_autoar_decide_dest_cb), send_data);
      g_signal_connect (arcreate, "progress",
          G_CALLBACK (send_file_autoar_progress_cb), send_data);
      g_signal_connect (arcreate, "cancelled",
          G_CALLBACK (send_file_autoar_cancelled_cb), send_data);
      g_signal_connect (arcreate, "completed",
          G_CALLBACK (send_file_autoar_completed_cb), send_data);
      g_signal_connect (arcreate, "error",
          G_CALLBACK (send_file_autoar_error_cb), send_data);
      autoar_create_start_async (arcreate, send_data->cancellable);

      g_free (template);
      g_free (archive_dir_path);
      g_object_unref (archive_dir);
      g_object_unref (arcreate);
    }
  else
    {
      send_file (contact, file);
    }
}

void
empathy_send_file (EmpathyContact *contact,
    GFile *file)
{
  g_file_query_info_async (file,
      G_FILE_ATTRIBUTE_STANDARD_TYPE,
      G_FILE_QUERY_INFO_NONE, G_PRIORITY_DEFAULT,
      NULL, (GAsyncReadyCallback) send_file_info_ready_cb, contact);
}

void
empathy_send_file_from_uri_list (EmpathyContact *contact,
    const gchar *uri_list)
{
  const gchar *nl;
  GFile *file;

  /* Only handle a single file for now. It would be wicked cool to be
     able to do multiple files, offering to zip them or whatever like
     nautilus-sendto does. Note that text/uri-list is defined to have
     each line terminated by \r\n, but we can be tolerant of applications
     that only use \n or don't terminate single-line entries.
  */
  nl = strstr (uri_list, "\r\n");
  if (!nl)
    nl = strchr (uri_list, '\n');

  if (nl)
    {
      gchar *uri = g_strndup (uri_list, nl - uri_list);
      file = g_file_new_for_uri (uri);
      g_free (uri);
    }
  else
    {
      file = g_file_new_for_uri (uri_list);
    }

  empathy_send_file (contact, file);

  g_object_unref (file);
}

static void
file_manager_send_file_response_cb (GtkDialog *widget,
    gint response_id,
    EmpathyContact *contact)
{
  GFile *file;
  GtkWidget *format;
  AutoarPref *arpref;
  int arformat, arfilter;

  if (response_id == GTK_RESPONSE_OK || response_id == EMPATHY_RESPONSE_SEND)
    {
      file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (widget));

      arpref = g_object_get_data (G_OBJECT (contact), "autoar-pref");
      format = g_object_get_data (G_OBJECT (contact), "autoar-format");
      autoar_gtk_format_filter_simple_get (format, &arformat, &arfilter);
      autoar_pref_set_default_format (arpref, arformat);
      autoar_pref_set_default_filter (arpref, arfilter);

      empathy_send_file (contact, file);

      g_object_unref (file);
    }

  g_object_unref (contact);
  gtk_widget_destroy (GTK_WIDGET (widget));
}

static gboolean
filter_cb (const GtkFileFilterInfo *filter_info,
    gpointer data)
{
  /* filter out socket files */
  return tp_strdiff (filter_info->mime_type, "inode/socket");
}

static GtkFileFilter *
create_file_filter (void)
{
  GtkFileFilter *filter;

  filter = gtk_file_filter_new ();

  gtk_file_filter_add_custom (filter, GTK_FILE_FILTER_MIME_TYPE, filter_cb,
    NULL, NULL);

  return filter;
}

void
empathy_send_file_with_file_chooser (EmpathyContact *contact)
{
  GtkWidget *widget;
  GtkWidget *button;

  GtkWidget *extra;
  GtkWidget *format_label;
  GtkWidget *format_combo;

  GSettings *settings;
  AutoarPref *arpref;

  g_return_if_fail (EMPATHY_IS_CONTACT (contact));

  DEBUG ("Creating selection file chooser");

  widget = gtk_file_chooser_dialog_new (_("Select a file"), NULL,
      GTK_FILE_CHOOSER_ACTION_OPEN,
      _("OK"), GTK_RESPONSE_OK,
      _("Cancel"), GTK_RESPONSE_CANCEL,
      NULL);

  /* send button */
  button = gtk_button_new_with_mnemonic (_("_Send"));
  gtk_button_set_image (GTK_BUTTON (button),
      gtk_image_new_from_icon_name (EMPATHY_IMAGE_DOCUMENT_SEND,
        GTK_ICON_SIZE_BUTTON));
  gtk_widget_show (button);

  gtk_dialog_add_action_widget (GTK_DIALOG (widget), button,
      EMPATHY_RESPONSE_SEND);

  gtk_widget_set_can_default (button, TRUE);
  gtk_dialog_set_default_response (GTK_DIALOG (widget),
      EMPATHY_RESPONSE_SEND);

  gtk_file_chooser_set_local_only (GTK_FILE_CHOOSER (widget), FALSE);

  gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (widget),
      g_get_home_dir ());

  gtk_file_chooser_add_filter (GTK_FILE_CHOOSER (widget),
      create_file_filter ());

  settings = g_settings_new (AUTOAR_PREF_DEFAULT_GSCHEMA_ID);
  arpref = autoar_pref_new_with_gsettings (settings);

  extra = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
  format_label = gtk_label_new (
      _("Archive selected directories using this format: "));
  format_combo = autoar_gtk_format_filter_simple_new (
      autoar_pref_get_default_format (arpref),
      autoar_pref_get_default_filter (arpref));
  gtk_box_pack_start (GTK_BOX (extra), format_label, FALSE, FALSE, 0);
  gtk_box_pack_start (GTK_BOX (extra), format_combo, FALSE, FALSE, 0);
  gtk_widget_show_all (extra);
  gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (widget), extra);

  g_signal_connect (widget, "response",
      G_CALLBACK (file_manager_send_file_response_cb), g_object_ref (contact));
  g_object_set_data_full (G_OBJECT (contact), "autoar-pref",
      g_object_ref (arpref), g_object_unref);
  g_object_set_data_full (G_OBJECT (contact), "autoar-format",
      g_object_ref (format_combo), g_object_unref);

  gtk_widget_show (widget);
  g_object_unref (settings);
  g_object_unref (arpref);
}

static void
file_manager_receive_file_response_cb (GtkDialog *dialog,
    GtkResponseType response,
    EmpathyFTHandler *handler)
{
  EmpathyFTFactory *factory;
  GFile *file;

  if (response == GTK_RESPONSE_OK)
    {
      GFile *parent;
      GFileInfo *info;
      guint64 free_space, file_size;
      GError *error = NULL;

      GtkToggleButton *extract_yes;
      GtkToggleButton *extract_delete;

      file = gtk_file_chooser_get_file (GTK_FILE_CHOOSER (dialog));
      parent = g_file_get_parent (file);
      info = g_file_query_filesystem_info (parent,
          G_FILE_ATTRIBUTE_FILESYSTEM_FREE, NULL, &error);

      g_object_unref (parent);

      if (error != NULL)
        {
          g_warning ("Error: %s", error->message);

          g_object_unref (file);
          return;
        }

      free_space = g_file_info_get_attribute_uint64 (info,
          G_FILE_ATTRIBUTE_FILESYSTEM_FREE);
      file_size = empathy_ft_handler_get_total_bytes (handler);

      g_object_unref (info);

      if (file_size > free_space)
        {
          GtkWidget *message = gtk_message_dialog_new (GTK_WINDOW (dialog),
            GTK_DIALOG_MODAL, GTK_MESSAGE_ERROR,
            GTK_BUTTONS_CLOSE,
            _("Insufficient free space to save file"));
          char *file_size_str, *free_space_str;

          file_size_str = g_format_size (file_size);
          free_space_str = g_format_size (free_space);

          gtk_message_dialog_format_secondary_text (
              GTK_MESSAGE_DIALOG (message),
              _("%s of free space are required to save this "
                "file, but only %s is available. Please "
                "choose another location."),
            file_size_str, free_space_str);

          gtk_dialog_run (GTK_DIALOG (message));

          g_free (file_size_str);
          g_free (free_space_str);
          gtk_widget_destroy (message);

          g_object_unref (file);

          return;
        }

      extract_yes = g_object_get_data (G_OBJECT (dialog), "extract-yes");
      extract_delete = g_object_get_data (G_OBJECT (dialog), "extract-delete");
      if (!gtk_toggle_button_get_active (extract_yes))
        {
          g_object_set_data (G_OBJECT (handler), "autoar-pref", NULL);
        }
      else
        {
          autoar_pref_set_delete_if_succeed (
              g_object_get_data (G_OBJECT (handler), "autoar-pref"),
              gtk_toggle_button_get_active (extract_delete));
        }

      factory = empathy_ft_factory_dup_singleton ();

      empathy_ft_factory_set_destination_for_incoming_handler (
          factory, handler, file);

      g_object_unref (factory);
      g_object_unref (file);
    }
  else
    {
      /* unref the handler, as we dismissed the file chooser,
       * and refused the transfer.
       */
      g_object_unref (handler);
    }

  gtk_widget_destroy (GTK_WIDGET (dialog));
}

void
empathy_receive_file_with_file_chooser (EmpathyFTHandler *handler)
{
  GtkWidget *widget;
  const gchar *dir, *filename;
  EmpathyContact *contact;
  gchar *title;

  GtkWidget *extra;
  GtkWidget *extract_yes;
  GtkWidget *extract_delete;

  GSettings *settings;
  AutoarPref *arpref;
  const gchar *content_type;
  gchar *mime_type;

  contact = empathy_ft_handler_get_contact (handler);
  g_assert (contact != NULL);

  title = g_strdup_printf (_("Incoming file from %s"),
    empathy_contact_get_alias (contact));

  widget = gtk_file_chooser_dialog_new (title,
      NULL, GTK_FILE_CHOOSER_ACTION_SAVE,
      GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
      GTK_STOCK_SAVE, GTK_RESPONSE_OK,
      NULL);

  gtk_file_chooser_set_current_name (GTK_FILE_CHOOSER (widget),
    filename = empathy_ft_handler_get_filename (handler));

  gtk_file_chooser_set_do_overwrite_confirmation
    (GTK_FILE_CHOOSER (widget), TRUE);

  dir = g_get_user_special_dir (G_USER_DIRECTORY_DOWNLOAD);
  if (dir == NULL)
    /* Fallback to $HOME if $XDG_DOWNLOAD_DIR is not set */
    dir = g_get_home_dir ();

  gtk_file_chooser_set_current_folder (GTK_FILE_CHOOSER (widget), dir);

  extra = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
  extract_yes = gtk_check_button_new_with_mnemonic (
     _("E_xtract files from the incoming file if it is an archive"));
  extract_delete = gtk_check_button_new_with_mnemonic (
     _("_Delete the saved incoming file after the extraction is completed"));
  gtk_box_pack_start (GTK_BOX (extra), extract_yes, FALSE, FALSE, 0);
  gtk_box_pack_start (GTK_BOX (extra), extract_delete, FALSE, FALSE, 0);

  settings = g_settings_new (AUTOAR_PREF_DEFAULT_GSCHEMA_ID);
  arpref = autoar_pref_new_with_gsettings (settings);
  content_type = empathy_ft_handler_get_content_type (handler);
  mime_type = g_content_type_get_mime_type (content_type);

  gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (extract_yes),
      autoar_pref_check_file_name (arpref, filename) ||
      autoar_pref_check_mime_type_d (arpref, mime_type));
  gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (extract_delete),
      autoar_pref_get_delete_if_succeed (arpref));
  g_object_bind_property (extract_yes, "active",
      extract_delete, "sensitive", G_BINDING_SYNC_CREATE);

  gtk_widget_show_all (extra);
  gtk_file_chooser_set_extra_widget (GTK_FILE_CHOOSER (widget), extra);

  g_signal_connect (widget, "response",
      G_CALLBACK (file_manager_receive_file_response_cb), handler);
  g_object_set_data_full (G_OBJECT (handler), "autoar-pref",
      g_object_ref (arpref), g_object_unref);
  g_object_set_data_full (G_OBJECT (widget), "extract-yes",
      g_object_ref (extract_yes), g_object_unref);
  g_object_set_data_full (G_OBJECT (widget), "extract-delete",
      g_object_ref (extract_delete), g_object_unref);

  gtk_widget_show (widget);
  g_free (title);
  g_object_unref (settings);
  g_object_unref (arpref);
  g_free (mime_type);
}

void
empathy_make_color_whiter (GdkRGBA *color)
{
  const GdkRGBA white = { 1.0, 1.0, 1.0, 1.0 };

  color->red = (color->red + white.red) / 2;
  color->green = (color->green + white.green) / 2;
  color->blue = (color->blue + white.blue) / 2;
}

static void
menu_deactivate_cb (GtkMenu *menu,
  gpointer user_data)
{
  /* FIXME: we shouldn't have to disconnect the signal (bgo #641327) */
  g_signal_handlers_disconnect_by_func (menu,
         menu_deactivate_cb, user_data);

  gtk_menu_detach (menu);
}

/* Convenient function to create a GtkMenu attached to @attach_to and detach
 * it when the menu is not displayed any more. This is useful when creating a
 * context menu that we want to get rid as soon as it as been displayed. */
GtkWidget *
empathy_context_menu_new (GtkWidget *attach_to)
{
  GtkWidget *menu;

  menu = gtk_menu_new ();

  gtk_menu_attach_to_widget (GTK_MENU (menu), attach_to, NULL);

  /* menu is initially unowned but gtk_menu_attach_to_widget () taked its
   * floating ref. We can either wait that @attach_to releases its ref when
   * it will be destroyed (when leaving Empathy most of the time) or explicitely
   * detach the menu when it's not displayed any more.
   * We go for the latter as we don't want to keep useless menus in memory
   * during the whole lifetime of Empathy. */
  g_signal_connect (menu, "deactivate", G_CALLBACK (menu_deactivate_cb), NULL);

  return menu;
}

gint64
empathy_get_current_action_time (void)
{
  return (tp_user_action_time_from_x11 (gtk_get_current_event_time ()));
}

/* @words = tpaw_live_search_strip_utf8_string (@text);
 *
 * User has to pass both so we don't have to compute @words ourself each time
 * this function is called. */
gboolean
empathy_individual_match_string (FolksIndividual *individual,
    const char *text,
    GPtrArray *words)
{
  const gchar *str;
  GeeSet *personas;
  GeeIterator *iter;
  gboolean retval = FALSE;

  /* check alias name */
  str = folks_alias_details_get_alias (FOLKS_ALIAS_DETAILS (individual));

  if (tpaw_live_search_match_words (str, words))
    return TRUE;

  personas = folks_individual_get_personas (individual);

  /* check contact id, remove the @server.com part */
  iter = gee_iterable_iterator (GEE_ITERABLE (personas));
  while (retval == FALSE && gee_iterator_next (iter))
    {
      FolksPersona *persona = gee_iterator_get (iter);
      const gchar *p;

      if (empathy_folks_persona_is_interesting (persona))
        {
          str = folks_persona_get_display_id (persona);

          /* Accept the persona if @text is a full prefix of his ID; that allows
           * user to find, say, a jabber contact by typing his JID. */
          if (g_str_has_prefix (str, text))
            {
              retval = TRUE;
            }
          else
            {
              gchar *dup_str = NULL;
              gboolean visible;

              p = strstr (str, "@");
              if (p != NULL)
                str = dup_str = g_strndup (str, p - str);

              visible = tpaw_live_search_match_words (str, words);
              g_free (dup_str);
              if (visible)
                retval = TRUE;
            }
        }
      g_clear_object (&persona);
    }
  g_clear_object (&iter);

  /* FIXME: Add more rules here, we could check phone numbers in
   * contact's vCard for example. */
  return retval;
}

void
empathy_launch_program (const gchar *dir,
    const gchar *name,
    const gchar *args)
{
  GdkDisplay *display;
  GError *error = NULL;
  gchar *path, *cmd;
  GAppInfo *app_info;
  GdkAppLaunchContext *context = NULL;

  /* Try to run from source directory if possible */
  path = g_build_filename (g_getenv ("EMPATHY_SRCDIR"), "src",
      name, NULL);

  if (!g_file_test (path, G_FILE_TEST_EXISTS))
    {
      g_free (path);
      path = g_build_filename (dir, name, NULL);
    }

  if (args != NULL)
    cmd = g_strconcat (path, " ", args, NULL);
  else
    cmd = g_strdup (path);

  app_info = g_app_info_create_from_commandline (cmd, NULL, 0, &error);
  if (app_info == NULL)
    {
      DEBUG ("Failed to create app info: %s", error->message);
      g_error_free (error);
      goto out;
    }

  display = gdk_display_get_default ();
  context = gdk_display_get_app_launch_context (display);

  if (!g_app_info_launch (app_info, NULL, (GAppLaunchContext *) context,
      &error))
    {
      g_warning ("Failed to launch %s: %s", name, error->message);
      g_error_free (error);
      goto out;
    }

out:
  tp_clear_object (&app_info);
  tp_clear_object (&context);
  g_free (path);
  g_free (cmd);
}

/* Most of the workspace manipulation code has been copied from libwnck
 * Copyright (C) 2001 Havoc Pennington
 * Copyright (C) 2005-2007 Vincent Untz
 */
static void
_wnck_activate_workspace (Screen *screen,
    int new_active_space,
    Time timestamp)
{
  Display *display;
  Window root;
  XEvent xev;

  display = DisplayOfScreen (screen);
  root = RootWindowOfScreen (screen);

  xev.xclient.type = ClientMessage;
  xev.xclient.serial = 0;
  xev.xclient.send_event = True;
  xev.xclient.display = display;
  xev.xclient.window = root;
  xev.xclient.message_type = gdk_x11_get_xatom_by_name ("_NET_CURRENT_DESKTOP");
  xev.xclient.format = 32;
  xev.xclient.data.l[0] = new_active_space;
  xev.xclient.data.l[1] = timestamp;
  xev.xclient.data.l[2] = 0;
  xev.xclient.data.l[3] = 0;
  xev.xclient.data.l[4] = 0;

  gdk_error_trap_push ();
  XSendEvent (display, root, False,
      SubstructureRedirectMask | SubstructureNotifyMask,
      &xev);
  XSync (display, False);
  gdk_error_trap_pop_ignored ();
}

static gboolean
_wnck_get_cardinal (Screen *screen,
    Window xwindow,
    Atom atom,
    int *val)
{
  Display *display;
  Atom type;
  int format;
  gulong nitems;
  gulong bytes_after;
  gulong *num;
  int err, result;

  display = DisplayOfScreen (screen);

  *val = 0;

  gdk_error_trap_push ();
  type = None;
  result = XGetWindowProperty (display, xwindow, atom,
      0, G_MAXLONG, False, XA_CARDINAL, &type, &format, &nitems,
      &bytes_after, (void *) &num);
  err = gdk_error_trap_pop ();
  if (err != Success ||
      result != Success)
    return FALSE;

  if (type != XA_CARDINAL)
    {
      XFree (num);
      return FALSE;
    }

  *val = *num;

  XFree (num);

  return TRUE;
}

static int
window_get_workspace (Screen *xscreen,
    Window win)
{
  int number;

  if (!_wnck_get_cardinal (xscreen, win,
        gdk_x11_get_xatom_by_name ("_NET_WM_DESKTOP"), &number))
    return -1;

  return number;
}

/* Ask X to move to the desktop on which @window currently is
 * and the present @window. */
void
empathy_move_to_window_desktop (GtkWindow *window,
    guint32 timestamp)
{
  GdkScreen *screen;
  Screen *xscreen;
  GdkWindow *gdk_window;
  int workspace;

  screen = gtk_window_get_screen (window);
  xscreen = gdk_x11_screen_get_xscreen (screen);
  gdk_window = gtk_widget_get_window (GTK_WIDGET (window));

  workspace = window_get_workspace (xscreen,
      gdk_x11_window_get_xid (gdk_window));
  if (workspace == -1)
    goto out;

  _wnck_activate_workspace (xscreen, workspace, timestamp);

out:
  gtk_window_present_with_time (window, timestamp);
}

void
empathy_set_css_provider (GtkWidget *widget)
{
  GtkCssProvider *provider;
  gchar *filename;
  GError *error = NULL;
  GdkScreen *screen;

  filename = empathy_file_lookup ("empathy.css", "data");

  provider = gtk_css_provider_new ();

  if (!gtk_css_provider_load_from_path (provider, filename, &error))
    {
      g_warning ("Failed to load css file '%s': %s", filename, error->message);
      g_error_free (error);
      goto out;
    }

  if (widget != NULL)
    screen = gtk_widget_get_screen (widget);
  else
    screen = gdk_screen_get_default ();

  gtk_style_context_add_provider_for_screen (screen,
      GTK_STYLE_PROVIDER (provider),
      GTK_STYLE_PROVIDER_PRIORITY_APPLICATION);

out:
  g_free (filename);
  g_object_unref (provider);
}

static gboolean
launch_app_info (GAppInfo *app_info,
    GError **error)
{
  GdkAppLaunchContext *context = NULL;
  GdkDisplay *display;
  GError *err = NULL;

  display = gdk_display_get_default ();
  context = gdk_display_get_app_launch_context (display);

  if (!g_app_info_launch (app_info, NULL, (GAppLaunchContext *) context,
        &err))
    {
      DEBUG ("Failed to launch %s: %s",
          g_app_info_get_display_name (app_info), err->message);
      g_propagate_error (error, err);
      return FALSE;
    }

  tp_clear_object (&context);
  return TRUE;
}

gboolean
empathy_launch_external_app (const gchar *desktop_file,
    const gchar *args,
    GError **error)
{
  GDesktopAppInfo *desktop_info;
  gboolean result;
  GError *err = NULL;

  desktop_info = g_desktop_app_info_new (desktop_file);
  if (desktop_info == NULL)
    {
      DEBUG ("%s not found", desktop_file);

      g_set_error (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND,
          "%s not found", desktop_file);
      return FALSE;
    }

  if (args == NULL)
    {
      result = launch_app_info (G_APP_INFO (desktop_info), error);
    }
  else
    {
      gchar *cmd;
      GAppInfo *app_info;

      /* glib doesn't have API to start a desktop file with args... (#637875) */
      cmd = g_strdup_printf ("%s %s", g_app_info_get_commandline (
            (GAppInfo *) desktop_info), args);

      app_info = g_app_info_create_from_commandline (cmd, NULL, 0, &err);
      if (app_info == NULL)
        {
          DEBUG ("Failed to launch '%s': %s", cmd, err->message);
          g_free (cmd);
          g_object_unref (desktop_info);
          g_propagate_error (error, err);
          return FALSE;
        }

      result = launch_app_info (app_info, error);

      g_object_unref (app_info);
      g_free (cmd);
    }

  g_object_unref (desktop_info);
  return result;
}