aboutsummaryrefslogblamecommitdiffstats
path: root/libempathy-gtk/empathy-account-widget-skype.c
blob: b6fcbff9ecf55ba6ce7103fd8f43f57be73c2d78 (plain) (tree)


















                                                                     
                                                                   










                                     
                                                   
 
                                         







                                                                  




                     
                    




                                              

                         





                                                                 
                            










































                                                                               
                                                                             



                                                                 
                                      















                                                                            
                                                                 






















                                                                      
                              
                    




                                                               
                 


                                      
                 









                                                                         
                 
 


                                                                           
 
                              
                 
 



                                                                
                                                                        
 
        



                                               
                        














                                                                              
                                                         























                                                                               

























































































































                                                                                





                                                                   
                                           













                                                                   




                                                                      
                                                 
                                           

                                                                      





                                                     


















                                                                           
                                 

 



















                                                                         

                          










                                                                             















                                                                               

                                                                        



                                                        
 








                                                                                

 
               

                                                                           
                               
 

                                                                             
 


                                                                             
 



                                                                    
 
                                                         




                                                 

               


           
















                                                                             












                                                                 
                          

                            































                                                                                
                    



                                                              







                                                               

































































                                                                               

                          



















                                                                            
                                                         

































































                                                                              





                                                    




























                                                                    












                                                                     



                                                               
                                            
                                                   
                                                                             
                                                                 
 



                                                
                                             
     

                            



                                                                              
                                         
                                                  
                                                  
                                                        

                









                                                                           







                                                                       

                                                 





                                                                    
                                                           



                                                                      
                                         
                                                                        
                                            
                                                  




                                                           

                                                             







                                                                       



                                                                   
 

                                                                       


                                                 



                                                              
 










                                                                              
 


                                                          
                                        
                                                               





                                                                     
                                                              

                                                                    
                                                                    
     



                                                                     
                                                                          
 










































































                                                                             



                                        
                      
 



                             
                                        


                                          



                                                                        














                                                                         
                                              









                                      

                                                                            

















                                                                         
      







                                                                 
                                                  











                                                                  
/*
 * Copyright (C) 2011 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: Danielle Madeley <danielle.madeley@collabora.co.uk>
 *          Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
 */

#include <config.h>

#include <string.h>

#include <glib/gi18n-lib.h>

#include <extensions/extensions.h>

#include <libempathy/empathy-utils.h>
#include <libempathy/empathy-server-sasl-handler.h>

#include "empathy-account-widget-skype.h"
#include "empathy-account-widget-private.h"
#include "empathy-ui-utils.h"

#define DEBUG_FLAG EMPATHY_DEBUG_ACCOUNT
#include <libempathy/empathy-debug.h>

#define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyAccountWidget)

typedef struct
{
  TpAccount *account;
  TpChannel *channel;
  char *password;
  gboolean remember;
} ObserveChannelsData;

static ObserveChannelsData *
observe_channels_data_new (TpAccount *account,
    TpChannel *channel,
    const char *password,
    gboolean remember)
{
  ObserveChannelsData *data = g_slice_new0 (ObserveChannelsData);

  data->account = g_object_ref (account);
  data->channel = g_object_ref (channel);
  data->password = g_strdup (password);
  data->remember = remember;

  return data;
}

static void
observe_channels_data_free (ObserveChannelsData *data)
{
  g_object_unref (data->account);
  g_object_unref (data->channel);
  g_free (data->password);

  g_slice_free (ObserveChannelsData, data);
}

static void
auth_observer_sasl_handler_invalidated (EmpathyServerSASLHandler *sasl_handler,
    gpointer user_data)
{
  DEBUG ("SASL Handler done");

  g_object_unref (sasl_handler);
}

static void
auth_observer_new_sasl_handler_cb (GObject *obj,
    GAsyncResult *result,
    gpointer user_data)
{
  ObserveChannelsData *data = user_data;
  EmpathyServerSASLHandler *sasl_handler;
  GError *error = NULL;

  sasl_handler = empathy_server_sasl_handler_new_finish (result, &error);
  if (error != NULL)
    {
      DEBUG ("Failed to create SASL handler: %s", error->message);

      tp_channel_close_async (data->channel, NULL, NULL);

      g_error_free (error);
      goto finally;
    }

  DEBUG ("providing password, remember = %s", data->remember ? "yes" : "no");

  g_signal_connect (sasl_handler, "invalidated",
      G_CALLBACK (auth_observer_sasl_handler_invalidated), NULL);
  empathy_server_sasl_handler_provide_password (sasl_handler,
      data->password, data->remember);

finally:
  observe_channels_data_free (data);
}

static void
auth_observer_claim_cb (GObject *dispatch_operation,
    GAsyncResult *result,
    gpointer user_data)
{
  ObserveChannelsData *data = user_data;
  GError *error = NULL;

  if (!tp_channel_dispatch_operation_claim_finish (
        TP_CHANNEL_DISPATCH_OPERATION (dispatch_operation), result, &error))
    {
      DEBUG ("Failed to claim auth channel: %s", error->message);

      g_error_free (error);
      observe_channels_data_free (data);
      return;
    }

  empathy_server_sasl_handler_new_async (data->account, data->channel,
      auth_observer_new_sasl_handler_cb, data);
}

static void
auth_observer_observe_channels (TpSimpleObserver *auth_observer,
    TpAccount *account,
    TpConnection *connection,
    GList *channels,
    TpChannelDispatchOperation *dispatch_operation,
    GList *requests,
    TpObserveChannelsContext *context,
    gpointer user_data)
{
  TpChannel *channel;
  GHashTable *props;
  GStrv available_mechanisms;
  const char *password = NULL;
  gboolean remember;

  /* we only do this for Psyke */
  if (tp_strdiff (
        tp_connection_get_connection_manager_name (connection),
        "psyke"))
    goto finally;

  /* can only deal with one channel */
  if (g_list_length (channels) != 1)
    goto finally;

  channel = channels->data;
  props = tp_channel_borrow_immutable_properties (channel);
  available_mechanisms = tp_asv_get_boxed (props,
      TP_PROP_CHANNEL_INTERFACE_SASL_AUTHENTICATION_AVAILABLE_MECHANISMS,
      G_TYPE_STRV);

  /* must support X-TELEPATHY-PASSWORD */
  if (!tp_strv_contains ((const char * const *) available_mechanisms,
        "X-TELEPATHY-PASSWORD"))
    goto finally;

  password = g_object_get_data (G_OBJECT (auth_observer), "password");
  remember = GPOINTER_TO_UINT (g_object_get_data (G_OBJECT (auth_observer),
        "remember-password"));

  if (tp_str_empty (password))
    goto finally;

  DEBUG ("claiming auth channel");

  tp_channel_dispatch_operation_claim_async (dispatch_operation,
      auth_observer_claim_cb,
      observe_channels_data_new (account, channel, password, remember));

finally:
  tp_observe_channels_context_accept (context);
}

static TpBaseClient *
auth_observer_new (void)
{
  TpDBusDaemon *dbus;
  TpBaseClient *auth_observer;
  GError *error = NULL;

  dbus = tp_dbus_daemon_dup (&error);

  if (error != NULL)
    {
      g_warning ("Failed to get DBus daemon: %s", error->message);
      g_error_free (error);
      return NULL;
    }

  auth_observer = tp_simple_observer_new (dbus, FALSE, "Empathy.PsykePreAuth",
      FALSE, auth_observer_observe_channels, NULL, NULL);

  tp_base_client_set_observer_delay_approvers (auth_observer, TRUE);
  tp_base_client_take_observer_filter (auth_observer, tp_asv_new (
        TP_PROP_CHANNEL_CHANNEL_TYPE,
        G_TYPE_STRING,
        TP_IFACE_CHANNEL_TYPE_SERVER_AUTHENTICATION,

        TP_PROP_CHANNEL_TYPE_SERVER_AUTHENTICATION_AUTHENTICATION_METHOD,
        G_TYPE_STRING,
        TP_IFACE_CHANNEL_INTERFACE_SASL_AUTHENTICATION,

        NULL));

  if (!tp_base_client_register (auth_observer, &error))
    {
      DEBUG ("Failed to register Psyke pre-auth observer: %s", error->message);
      g_error_free (error);
    }

  g_object_unref (dbus);

  return auth_observer;
}

enum {
    PS_COL_ENUM_VALUE,
    PS_COL_DISPLAY_NAME,
    NUM_PS_COLS
};

static void
account_widget_skype_combo_changed_cb (GtkComboBox *combo,
    EmpathyAccountWidget *self)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  const char *prop_name = g_object_get_data (G_OBJECT (combo), "dbus-property");
  TpConnection *conn;
  EmpPrivacySetting prop_value;
  GtkTreeIter iter;
  GValue value = { 0, };

  gtk_combo_box_get_active_iter (combo, &iter);
  gtk_tree_model_get (gtk_combo_box_get_model (combo), &iter,
      PS_COL_ENUM_VALUE, &prop_value,
      -1);

  DEBUG ("Combo changed: %s", prop_name);

  g_value_init (&value, G_TYPE_UINT);
  g_value_set_uint (&value, prop_value);

  conn = tp_account_get_connection (
      empathy_account_settings_get_account (priv->settings));
  tp_cli_dbus_properties_call_set (conn, -1,
      EMP_IFACE_CONNECTION_INTERFACE_PRIVACY_SETTINGS,
      prop_name, &value, NULL, NULL, NULL, NULL);

  g_value_unset (&value);
}

static void
account_widget_skype_toggle_changed_cb (GtkToggleButton *toggle,
    EmpathyAccountWidget *self)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  const char *prop_name = g_object_get_data (G_OBJECT (toggle),
      "dbus-property");
  TpConnection *conn;
  GValue value = { 0, };

  DEBUG ("Toggle changed: %s", prop_name);

  g_value_init (&value, G_TYPE_BOOLEAN);
  g_value_set_boolean (&value, gtk_toggle_button_get_active (toggle));

  conn = tp_account_get_connection (
      empathy_account_settings_get_account (priv->settings));
  tp_cli_dbus_properties_call_set (conn, -1,
      EMP_IFACE_CONNECTION_INTERFACE_PRIVACY_SETTINGS,
      prop_name, &value, NULL, NULL, NULL, NULL);

  g_value_unset (&value);
}

static void
account_widget_skype_set_value (EmpathyAccountWidget *self,
    GtkWidget *widget,
    GValue *value)
{
  g_return_if_fail (value != NULL);

  if (GTK_IS_COMBO_BOX (widget))
    {
      GtkTreeModel *model =
        gtk_combo_box_get_model (GTK_COMBO_BOX (widget));
      guint prop_value;
      GtkTreeIter iter;
      gboolean valid;

      g_return_if_fail (G_VALUE_HOLDS_UINT (value));

      prop_value = g_value_get_uint (value);

      g_signal_handlers_block_by_func (widget,
          account_widget_skype_combo_changed_cb, self);

      for (valid = gtk_tree_model_get_iter_first (model, &iter);
           valid;
           valid = gtk_tree_model_iter_next (model, &iter))
        {
          guint v;

          gtk_tree_model_get (model, &iter,
              PS_COL_ENUM_VALUE, &v,
              -1);

          if (v == prop_value)
            {
              gtk_combo_box_set_active_iter (GTK_COMBO_BOX (widget), &iter);
              break;
            }
        }

      g_signal_handlers_unblock_by_func (widget,
          account_widget_skype_combo_changed_cb, self);
    }
  else if (GTK_IS_TOGGLE_BUTTON (widget))
    {
      g_return_if_fail (G_VALUE_HOLDS_BOOLEAN (value));

      g_signal_handlers_block_by_func (widget,
          account_widget_skype_toggle_changed_cb, self);

      gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (widget),
          g_value_get_boolean (value));

      g_signal_handlers_unblock_by_func (widget,
          account_widget_skype_toggle_changed_cb, self);
    }
  else
    {
      g_assert_not_reached ();
    }
}

static void
account_widget_build_skype_get_password_saved_cb (TpProxy *account,
    const GValue *value,
    const GError *in_error,
    gpointer user_data,
    GObject *password_entry)
{
  GtkWidget *remember_password = user_data;
  gboolean password_saved;

  if (in_error != NULL)
    {
      DEBUG ("Failed to get PasswordSaved: %s", in_error->message);
      return;
    }

  g_return_if_fail (G_VALUE_HOLDS_BOOLEAN (value));

  password_saved = g_value_get_boolean (value);

  DEBUG ("PasswordSaved: %s", password_saved ? "yes" : "no");

  /* don't wipe the password if there's a real value in there */
  if (g_object_get_data (password_entry, "fake-password") == NULL &&
      !tp_str_empty (gtk_entry_get_text (GTK_ENTRY (password_entry))))
    return;

  gtk_entry_set_text (GTK_ENTRY (password_entry),
      password_saved ? "xxxxxxxxxxxx": "");
  gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (remember_password),
      password_saved);

  g_object_set_data (password_entry, "fake-password",
      GUINT_TO_POINTER (password_saved));
}

static void
account_widget_build_skype_account_properties_changed_cb (TpProxy *account,
    const char *iface,
    GHashTable *changed,
    const char **invalidated,
    gpointer user_data,
    GObject *password_entry)
{
  GValue *value;

  if (tp_strdiff (iface,
        EMP_IFACE_ACCOUNT_INTERFACE_EXTERNAL_PASSWORD_STORAGE))
    return;

  value = g_hash_table_lookup (changed, "PasswordSaved");

  if (value == NULL)
    return;

  account_widget_build_skype_get_password_saved_cb (account, value, NULL,
      user_data, password_entry);
}

static void
account_widget_skype_additional_apply_forget_passwd_cb (TpProxy *account,
    const GError *in_error,
    gpointer user_data,
    GObject *weak_obj)
{
  GSimpleAsyncResult *simple = G_SIMPLE_ASYNC_RESULT (weak_obj);

  if (in_error != NULL)
    {
      DEBUG ("Did not forget password: %s", in_error->message);
    }
  else
    {
      DEBUG ("Password forgot, proceed to reconnect");
    }

  /* reconnect is required */
  g_simple_async_result_set_op_res_gboolean (simple, in_error == NULL);
  g_simple_async_result_complete (simple);

  g_object_unref (simple);
}

static void
account_widget_skype_additional_apply_async (EmpathyAccountWidget *self,
    GAsyncReadyCallback callback,
    gpointer user_data)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  TpAccount *account = empathy_account_settings_get_account (priv->settings);
  GSimpleAsyncResult *simple = g_simple_async_result_new (G_OBJECT (self),
      callback, user_data, NULL);
  GtkWidget *password_entry, *remember_password;
  GObject *auth_observer;
  const char *password = NULL;
  gboolean remember;

  /* sync the password with the observer */
  auth_observer = g_object_get_data (G_OBJECT (self), "auth-observer");
  password_entry = g_object_get_data (G_OBJECT (self), "password-entry");
  remember_password = g_object_get_data (G_OBJECT (self), "remember-password");

  if (g_object_get_data (G_OBJECT (password_entry), "fake-password") == NULL)
    password = gtk_entry_get_text (GTK_ENTRY (password_entry));

  remember = gtk_toggle_button_get_active (
      GTK_TOGGLE_BUTTON (remember_password));

  DEBUG ("Setting password and remember-password (%u) on auth observer",
      remember);
  g_object_set_data_full (auth_observer, "password",
      g_strdup (password), g_free);
  g_object_set_data (auth_observer, "remember-password",
      GUINT_TO_POINTER (remember));

  if (priv->contains_pending_changes)
    {
      /* forget the password, else psyke won't query for the new one */
      DEBUG ("Forget the password so that Psyke queries for a new one");
      emp_cli_account_interface_external_password_storage_call_forget_password (
          TP_PROXY (account), -1,
          account_widget_skype_additional_apply_forget_passwd_cb,
          NULL, NULL, G_OBJECT (simple));
    }
}

static gboolean
account_widget_build_skype_password_entry_focus (GtkWidget *password_entry,
    GdkEventFocus *event,
    EmpathyAccountWidget *self)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  TpAccount *account = empathy_account_settings_get_account (priv->settings);

  if (g_object_get_data (G_OBJECT (password_entry), "fake-password") != NULL)
    {
      DEBUG ("Clearing fake password for editing");

      gtk_entry_set_text (GTK_ENTRY (password_entry), "");
      g_object_set_data (G_OBJECT (password_entry), "fake-password",
          GUINT_TO_POINTER (FALSE));
    }

  if (account != NULL && tp_account_is_enabled (account))
    {
      DEBUG ("Highlighting Apply/Cancel button");

      empathy_account_widget_changed (self);
    }

  return FALSE;
}

static void
account_widget_skype_remember_password_toggled (GtkToggleButton *button,
    EmpathyAccountWidget *self)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  TpAccount *account = empathy_account_settings_get_account (priv->settings);

  /* we only forget the password if the toggle goes inactive */
  if (gtk_toggle_button_get_active (button))
    return;

  DEBUG ("forgetting password");

  emp_cli_account_interface_external_password_storage_call_forget_password (
      TP_PROXY (account), -1, NULL, NULL, NULL, NULL);
}

static void
account_widget_build_skype_get_privacy_settings_cb (TpProxy *cm,
    GHashTable *props,
    const GError *in_error,
    gpointer user_data,
    GObject *weak_obj)
{
  EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (weak_obj);
  GtkBuilder *gui = user_data;
  guint i;

  static const char *widgets[] = {
      "allow-text-chats-from",
      "allow-skype-calls-from",
      "show-my-avatar-to",
      "show-my-web-status",
      "show-i-have-video-to"
  };

  if (in_error != NULL)
    {
      GtkWidget *table, *infobar, *label;

      DEBUG ("Failed to get properties: %s", in_error->message);

      table = GTK_WIDGET (gtk_builder_get_object (gui,
            "privacy-settings-table"));
      gtk_widget_set_sensitive (table, FALSE);

      infobar = gtk_info_bar_new ();
      gtk_box_pack_start (
          GTK_BOX (gtk_builder_get_object (gui, "privacy-settings-vbox")),
          infobar, FALSE, TRUE, 0);
      gtk_info_bar_set_message_type (GTK_INFO_BAR (infobar), GTK_MESSAGE_ERROR);
      label = gtk_label_new (_("Failed to retrieve privacy settings."));
      gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
      gtk_container_add (GTK_CONTAINER (
          gtk_info_bar_get_content_area (GTK_INFO_BAR (infobar))),
        label);
      gtk_widget_show (label);

      return;
    }

  for (i = 0; i < G_N_ELEMENTS (widgets); i++)
    {
      GtkWidget *widget = GTK_WIDGET (gtk_builder_get_object (gui, widgets[i]));
      const char *prop_name = g_object_get_data (G_OBJECT (widget),
          "dbus-property");
      GValue *value;

      DEBUG ("Widget '%s' (%s), prop = %s",
          widgets[i], G_OBJECT_TYPE_NAME (widget), prop_name);

      value = g_hash_table_lookup (props, prop_name);
      if (value == NULL)
        {
          g_warning ("Couldn't get a value for %s", prop_name);
          continue;
        }

      account_widget_skype_set_value (self, widget, value);
    }
}

static void
account_widget_skype_properties_changed_cb (TpProxy *conn,
    const char *interface,
    GHashTable *props,
    const char **invalidated,
    gpointer user_data,
    GObject *weak_obj)
{
  EmpathyAccountWidget *self = EMPATHY_ACCOUNT_WIDGET (weak_obj);
  GtkWidget *table = user_data;
  GHashTableIter iter;
  const char *prop;
  GValue *value;

  g_hash_table_iter_init (&iter, props);
  while (g_hash_table_iter_next (&iter, (gpointer) &prop, (gpointer) &value))
    {
      GList *children, *ptr;

      DEBUG ("Property changed: %s", prop);

      /* find this value in the widget tree */
      children = gtk_container_get_children (GTK_CONTAINER (table));

      for (ptr = children; ptr != NULL; ptr = ptr->next)
        {
          GtkWidget *widget = ptr->data;
          const char *prop_name = g_object_get_data (G_OBJECT (widget),
              "dbus-property");

          if (!tp_strdiff (prop_name, prop))
            {
              DEBUG ("Got child %p (%s)", widget, G_OBJECT_TYPE_NAME (widget));

              account_widget_skype_set_value (self, widget, value);
              break;
            }
        }

      g_list_free (children);
    }
}

/**
 * account_widget_build_skype_setup_combo:
 * @gui:
 * @widget:
 * @first_option: a list of options from the enum, terminated by -1
 */
static void
account_widget_build_skype_setup_combo (EmpathyAccountWidget *self,
    GtkBuilder *gui,
    const char *widget,
    const char *prop_name,
    int first_option,
    ...)
{
  GtkWidget *combo;
  GtkListStore *store;
  va_list var_args;
  int option;

  static const char *options[NUM_EMP_PRIVACY_SETTINGS] = {
        N_("No one"),
        N_("My contacts"),
        N_("Anyone"),
        N_("Known Numbers")
  };

  combo = GTK_WIDGET (gtk_builder_get_object (gui, widget));

  g_return_if_fail (combo != NULL);

  store = gtk_list_store_new (NUM_PS_COLS,
      G_TYPE_UINT, /* PS_COL_ENUM_VALUE */
      G_TYPE_STRING /* PS_COL_DISPLAY_NAME */
      );
  gtk_combo_box_set_model (GTK_COMBO_BOX (combo), GTK_TREE_MODEL (store));

  va_start (var_args, first_option);

  for (option = first_option; option != -1; option = va_arg (var_args, int))
    {
      gtk_list_store_insert_with_values (store, NULL, -1,
          PS_COL_ENUM_VALUE, option,
          PS_COL_DISPLAY_NAME, gettext (options[option]),
          -1);
    }

  va_end (var_args);

  g_object_set_data (G_OBJECT (combo), "dbus-property", (gpointer) prop_name);
  g_signal_connect (combo, "changed",
      G_CALLBACK (account_widget_skype_combo_changed_cb), self);
}

static void
account_widget_skype_privacy_settings (GtkWidget *button,
    EmpathyAccountWidget *self)
{
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  TpConnection *conn;
  char *filename;
  GtkBuilder *gui;
  GtkWidget *dialog, *show_my_web_status, *table, *vbox;
  GtkWidget *toplevel, *infobar, *label;

  DEBUG ("Loading privacy settings");

  filename = empathy_file_lookup ("empathy-account-widget-skype.ui",
      "libempathy-gtk");
  gui = empathy_builder_get_file (filename,
      "privacy-settings-dialog", &dialog,
      "privacy-settings-table", &table,
      "privacy-settings-vbox", &vbox,
      "show-my-web-status", &show_my_web_status,
      NULL);

  toplevel = gtk_widget_get_toplevel (self->ui_details->widget);
  if (gtk_widget_is_toplevel (toplevel) && GTK_IS_WINDOW (toplevel))
    gtk_window_set_transient_for (GTK_WINDOW (dialog), GTK_WINDOW (toplevel));

  /* make the settings insensitive when the account is disconnected */
  tp_account_bind_connection_status_to_property (
      empathy_account_settings_get_account (priv->settings),
      table, "sensitive", FALSE);

  /* set up an informative info bar */
  infobar = gtk_info_bar_new ();
  gtk_box_pack_start (GTK_BOX (vbox), infobar, FALSE, TRUE, 0);
  gtk_info_bar_set_message_type (GTK_INFO_BAR (infobar), GTK_MESSAGE_INFO);
  label = gtk_label_new (_("Privacy settings can only be changed while the "
                           "account is connected."));
  gtk_label_set_line_wrap (GTK_LABEL (label), TRUE);
  gtk_container_add (
      GTK_CONTAINER (gtk_info_bar_get_content_area (GTK_INFO_BAR (infobar))),
      label);
  gtk_widget_show (label);
  g_object_bind_property (table, "sensitive", infobar, "visible",
      G_BINDING_SYNC_CREATE | G_BINDING_INVERT_BOOLEAN);

  account_widget_build_skype_setup_combo (self, gui,
      "allow-text-chats-from", "AllowTextChannelsFrom",
      EMP_PRIVACY_SETTING_ANYONE,
      EMP_PRIVACY_SETTING_CONTACTS,
      -1);
  account_widget_build_skype_setup_combo (self, gui,
      "allow-skype-calls-from", "AllowCallChannelsFrom",
      EMP_PRIVACY_SETTING_ANYONE,
      EMP_PRIVACY_SETTING_CONTACTS,
      -1);
  account_widget_build_skype_setup_combo (self, gui,
      "show-i-have-video-to", "ShowIHaveVideoTo",
      EMP_PRIVACY_SETTING_ANYONE,
      EMP_PRIVACY_SETTING_CONTACTS,
      EMP_PRIVACY_SETTING_NOBODY,
      -1);
  account_widget_build_skype_setup_combo (self, gui,
      "show-my-avatar-to", "ShowMyAvatarTo",
      EMP_PRIVACY_SETTING_ANYONE,
      EMP_PRIVACY_SETTING_CONTACTS,
      -1);

  g_object_set_data (G_OBJECT (show_my_web_status), "dbus-property",
      "ShowMyWebStatus");
  g_signal_connect (show_my_web_status, "toggled",
      G_CALLBACK (account_widget_skype_toggle_changed_cb), self);

  /* get the current parameter values from psyke */
  conn = tp_account_get_connection (
      empathy_account_settings_get_account (priv->settings));

  tp_cli_dbus_properties_call_get_all (conn, -1,
      EMP_IFACE_CONNECTION_INTERFACE_PRIVACY_SETTINGS,
      account_widget_build_skype_get_privacy_settings_cb,
      g_object_ref (gui), g_object_unref,
      G_OBJECT (self));
  tp_cli_dbus_properties_connect_to_properties_changed (conn,
      account_widget_skype_properties_changed_cb,
      table, NULL, G_OBJECT (self), NULL);

  g_object_unref (gui);

  gtk_dialog_run (GTK_DIALOG (dialog));
  gtk_widget_destroy (dialog);
}

static void
account_widget_build_skype_set_pixmap (GtkBuilder *gui,
    const char *widget,
    const char *file)
{
  GtkImage *image = GTK_IMAGE (gtk_builder_get_object (gui, widget));
  char *filename = empathy_file_lookup (file, "data");

  gtk_image_set_from_file (image, filename);

  g_free (filename);
}

void
empathy_account_widget_build_skype (EmpathyAccountWidget *self,
    const char *filename)
{
  static TpBaseClient *auth_observer = NULL;
  EmpathyAccountWidgetPriv *priv = GET_PRIV (self);
  TpAccount *account = empathy_account_settings_get_account (priv->settings);
  GtkWidget *password_entry, *password_label, *remember_password;

  /* additional apply function */
  self->ui_details->additional_apply_async =
    account_widget_skype_additional_apply_async;

  if (priv->simple || priv->creating_account)
    {
      GtkWidget *skype_info;

      /* if we don't have an account it means we're doing the initial setup */
      self->ui_details->gui = empathy_builder_get_file (filename,
          "table_common_skype_settings_setup", &priv->table_common_settings,
          "vbox_skype_settings_setup", &self->ui_details->widget,
          "skype-info-vbox", &skype_info,
          "label_password_setup", &password_label,
          "entry_password_setup", &password_entry,
          "remember-password-setup", &remember_password,
          NULL);

      if (priv->simple)
        {
          /* The assistant doesn't work well with Skype passwords as
           * the observer goes away before the account goes offline,
           * so just hide the password and let Empathy ask for it later. */
          gtk_widget_hide (password_label);
          gtk_widget_hide (password_entry);
          gtk_widget_hide (remember_password);
        }

      account_widget_build_skype_set_pixmap (self->ui_details->gui,
          "plugged-into-skype-logo", "plugged-into-skype.png");
      account_widget_build_skype_set_pixmap (self->ui_details->gui,
          "canonical-logo", "canonical-logo.png");

      gtk_box_pack_end (GTK_BOX (self->ui_details->widget), skype_info,
          TRUE, TRUE, 0);

      empathy_account_widget_handle_params (self,
          "entry_id_setup", "account",
          NULL);

      self->ui_details->default_focus = g_strdup ("entry_id_setup");
    }
  else
    {
      GtkWidget *edit_privacy_settings_button, *skype_info;

      self->ui_details->gui = empathy_builder_get_file (filename,
          "table_common_skype_settings", &priv->table_common_settings,
          "vbox_skype_settings", &self->ui_details->widget,
          "skype-info-vbox", &skype_info,
          "edit-privacy-settings-button", &edit_privacy_settings_button,
          "entry_password", &password_entry,
          "remember-password", &remember_password,
          NULL);

      empathy_builder_connect (self->ui_details->gui, self,
          "edit-privacy-settings-button", "clicked",
              account_widget_skype_privacy_settings,
          "remember-password", "toggled",
              account_widget_skype_remember_password_toggled,
          NULL);

      if (account != NULL)
        tp_account_bind_connection_status_to_property (account,
            edit_privacy_settings_button, "sensitive", FALSE);
      else
        gtk_widget_set_sensitive (edit_privacy_settings_button, FALSE);

      account_widget_build_skype_set_pixmap (self->ui_details->gui,
          "plugged-into-skype-logo", "plugged-into-skype.png");
      account_widget_build_skype_set_pixmap (self->ui_details->gui,
          "canonical-logo", "canonical-logo.png");

      gtk_box_pack_end (GTK_BOX (self->ui_details->widget), skype_info,
          TRUE, TRUE, 0);

      empathy_account_widget_handle_params (self,
          "entry_id", "account",
          NULL);

      self->ui_details->default_focus = g_strdup ("entry_id");
    }

  /* create the Psyke pre-authentication observer */
  if (auth_observer == NULL)
    {
      /* the auth observer lives for the lifetime of the process */
      DEBUG ("Creating Psyke authentication observer");
      auth_observer = auth_observer_new ();
    }

  g_object_set_data (G_OBJECT (self), "auth-observer", auth_observer);
  g_object_set_data (G_OBJECT (self), "password-entry", password_entry);
  g_object_set_data (G_OBJECT (self), "remember-password", remember_password);

  g_object_bind_property (remember_password, "active",
      password_entry, "sensitive", G_BINDING_SYNC_CREATE);

  /* find out if we know the password */
  if (account != NULL && tp_proxy_has_interface_by_id (account,
        EMP_IFACE_QUARK_ACCOUNT_INTERFACE_EXTERNAL_PASSWORD_STORAGE))
    {
      tp_cli_dbus_properties_call_get (account, -1,
          EMP_IFACE_ACCOUNT_INTERFACE_EXTERNAL_PASSWORD_STORAGE,
          "PasswordSaved",
          account_widget_build_skype_get_password_saved_cb,
          remember_password, NULL, G_OBJECT (password_entry));
      tp_cli_dbus_properties_connect_to_properties_changed (account,
          account_widget_build_skype_account_properties_changed_cb,
          remember_password, NULL, G_OBJECT (password_entry), NULL);
    }

  /* if the user changes the password, it's probably no longer a fake
   * password */
  g_signal_connect (password_entry, "focus-in-event",
      G_CALLBACK (account_widget_build_skype_password_entry_focus), self);
}

gboolean
empathy_account_widget_skype_show_eula (GtkWindow *parent)
{
  GtkWidget *dialog, *textview, *vbox, *scrolledwindow;
  GtkTextBuffer *buffer;
  gchar *filename, *l10n_filename;
  const gchar * const * langs;
  GError *error = NULL;
  gchar *eula;
  gint result;
  gsize len;
  gint i;

  filename = empathy_file_lookup ("skype-eula.txt", "data");

  langs = g_get_language_names ();

  for (i = 0; langs[i] != NULL; i++)
    {
      l10n_filename = g_strconcat (filename, ".", langs[i], NULL);
      g_file_get_contents (l10n_filename, &eula, &len, NULL);
      g_free (l10n_filename);

      if (eula != NULL)
        break;
    }

  if (eula == NULL)
    {
      DEBUG ("Could not open translated Skype EULA");
      g_file_get_contents (filename, &eula, &len, &error);
    }

  g_free (filename);

  if (error != NULL)
    {
      g_warning ("Could not open Skype EULA: %s", error->message);
      g_error_free (error);
      return FALSE;
    }

  dialog = gtk_dialog_new_with_buttons (_("End User License Agreement"),
      parent, GTK_DIALOG_MODAL,
      _("Decline"), GTK_RESPONSE_CANCEL,
      _("Accept"), GTK_RESPONSE_ACCEPT,
      NULL);

  buffer = gtk_text_buffer_new (NULL);
  gtk_text_buffer_set_text (buffer, eula, len);
  g_free (eula);
  textview = gtk_text_view_new_with_buffer (buffer);
  gtk_text_view_set_editable (GTK_TEXT_VIEW (textview), FALSE);
  gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (textview), GTK_WRAP_WORD_CHAR);

  vbox = gtk_dialog_get_content_area (GTK_DIALOG (dialog));

  scrolledwindow = gtk_scrolled_window_new (NULL, NULL);
  gtk_scrolled_window_add_with_viewport (
      GTK_SCROLLED_WINDOW (scrolledwindow), textview);
  gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow),
      GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC);

  gtk_box_pack_start (GTK_BOX (vbox), scrolledwindow, TRUE, TRUE, 0);
  gtk_window_set_default_size (GTK_WINDOW (dialog), 400, 250);
  gtk_widget_show_all (dialog);

  result = gtk_dialog_run (GTK_DIALOG (dialog));

  gtk_widget_destroy (dialog);

  return (result == GTK_RESPONSE_ACCEPT);
}

static gboolean
is_other_psyke_account (TpAccount *ours,
    TpAccount *other)
{
  gboolean paths_diff;

  if (ours == NULL)
    paths_diff = TRUE;
  else
    paths_diff = tp_strdiff (
        tp_proxy_get_object_path (ours),
        tp_proxy_get_object_path (other));

  return (paths_diff &&
      tp_account_is_enabled (other) &&
      !tp_strdiff (tp_account_get_connection_manager (other), "psyke"));
}

gboolean
empathy_accounts_dialog_skype_disable_other_accounts (TpAccount *account,
    GtkWindow *parent)
{
  TpAccountManager *am = tp_account_manager_dup ();
  GList *accounts, *ptr;
  gboolean other_psyke_accounts = FALSE;

  /* check if any other psyke accounts are enabled */
  accounts = tp_account_manager_get_valid_accounts (am);

  for (ptr = accounts; ptr != NULL; ptr = ptr->next)
    {
      TpAccount *a = ptr->data;

      if (is_other_psyke_account (account, a))
        {
          other_psyke_accounts = TRUE;
          break;
        }
    }

  g_list_free (accounts);

  if (other_psyke_accounts)
    {
#if 0 /* this dialog is a good idea, but currently lacking from the UX spec,
       * so it's being left out for now */
      GtkWidget *msg = gtk_message_dialog_new (parent, 0,
          GTK_MESSAGE_QUESTION, GTK_BUTTONS_YES_NO,
          "%s\n\n%s",
          _("Multiple Skype accounts can not be enabled simultaneously. "
            "Enabling this Skype account will disable all others."),
          _("Continue to enable this account?"));
      int response;

      response = gtk_dialog_run (GTK_DIALOG (msg));
      gtk_widget_destroy (msg);

      if (response != GTK_RESPONSE_YES)
        {
          /* the user chose not to proceed */
          g_object_unref (am);

          return FALSE;
        }
#endif

      /* the user chose to proceed, disable the other accounts */
      accounts = tp_account_manager_get_valid_accounts (am);

      for (ptr = accounts; ptr != NULL; ptr = ptr->next)
        {
          TpAccount *a = ptr->data;

          if (is_other_psyke_account (account, a))
            {
              tp_account_set_enabled_async (a, FALSE, NULL, NULL);
            }
        }

      g_list_free (accounts);
    }

  g_object_unref (am);

  return TRUE;
}