aboutsummaryrefslogblamecommitdiffstats
path: root/libempathy/empathy-contact-manager.c
blob: 8db9a5a0910ce405668b11a304323d53239d074a (plain) (tree)
1
2
3
4
5
6
7
8
9

                                                                           
                                         
  



                                                                     
  
                                                                  

                                                                    
                                                  
  


                                                                             
  
                                                 





                   
                                           
                                 


                                          
                                    
                                 
                          

                                        
                          
 

                                                                   

                                                                        
                              
                                          
                            
 
                                                                                   



                                                                                       
 

                                                       
           






                                                                     
 


                                                                           
 











                                                                            
 








                                                                    


           




                                                               

                                                             
                                   
 


                                                                 
 


                                                             
                                                              
         


           























                                                                                  
                                                      




                                                          
                                                               
 
                                                          
                                        
                                              
 



                                                                                   


                                                                                  


                       

                                                                  
 



                                                                           
                                



                                                                          
                                

                                                                           
                                

                                                                         
                                


           





                                                                       
                                                                      
                                                                           
                                        



           







                                                                 
 
                                               

 













                                                                                           
                                                                                  




                      
















                                                                             





                                                                      
                                                                

                                                                                    


           






                                                                               
                             
 
                                                                         

                                                                                









                                                                           
                                                                            


                                                                                
                                                                      
                                                                   
                                

                               
 
                                                                                
                                                                              
                                         


           

                                                             

                                                                               
 
                             

                                                                 

                                                                              
 
                                                          
 
                                                            
                                                  

 
                       
                                            
 
                                                                 

 

                                                                 
                                                                    
 
                                                             

                                                                          
                                                                   
 
                                                             

 

                                                 
                                                 
                                                 
 
                                                             
                                        
                                              
 
                                                                
 

                                                              
 


                                                                  

 

                                                    
                                                    
                                                    
 
                                                             
                                        
                                              
 
                                                                
 

                                                              
 


                                                                     

 
           
                                                                       








                                                                           
              
                                                         
 
                                                             
                                                   

                                                                          
 
                                          
                                                                           
                                         
 

                        
 
           
                                                                        

                                                                      
 
                 
 

                                                                            

 

                                                          
 

                                                             
 
                                                                          
 


                                                                            
 
                        

 
           
                                                                          

                                                                          
 
                          
 








                                                                                   
         
 
                             

 

                                                            
 
                                                             



                                                                          
                                          
                                                                              




                                       


                                                        
 

                                                             
                                              
 

                                                                          

                                                              
 

                                                                       

         
                    


           


                                                          
 

                                                             
                                              
 
                                                                
 

                                                              
 

                                                                         



           


                                                               
 

                                                             
                                              
 
                                                                
 

                                                              
 


                                                                              

 



                               

           
                                                                       

                                                                 
 


                                                                       


           


                                                            
 

                                                             
 
                                                                
 




                                                                            

 
                                                                                   











                                                                             
 






                                                                            
           

                                                         

                                                         










                                                                

                                                                    























                                                                           
                                                           
 








                                                                     
                                                                

                                                               

 



                                                                  

                                                             

                                         
 
                                                                           
                                                         
 
                                                             
                           
                             

                                                      
 
                     
 
 
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
 * Copyright (C) 2007-2008 Collabora Ltd.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Authors: Xavier Claessens <xclaesse@gmail.com>
 */

#include <config.h>

#include <string.h>

#include <telepathy-glib/account-manager.h>
#include <telepathy-glib/enums.h>
#include <telepathy-glib/proxy-subclass.h>
#include <telepathy-glib/util.h>

#include "empathy-contact-manager.h"
#include "empathy-contact-list.h"
#include "empathy-utils.h"

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

#define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyContactManager)
typedef struct {
    /* Owned (TpConnection *) => Owned (TpContactList *)
       The contact list associated with each connected connection */
    GHashTable     *lists;
    TpAccountManager *account_manager;
} EmpathyContactManagerPriv;

static void contact_manager_iface_init         (EmpathyContactListIface    *iface);

G_DEFINE_TYPE_WITH_CODE (EmpathyContactManager, empathy_contact_manager, G_TYPE_OBJECT,
             G_IMPLEMENT_INTERFACE (EMPATHY_TYPE_CONTACT_LIST,
                        contact_manager_iface_init));

static EmpathyContactManager *manager_singleton = NULL;

static void
contact_manager_members_changed_cb (EmpathyTpContactList  *list,
                    EmpathyContact        *contact,
                    EmpathyContact        *actor,
                    guint                  reason,
                    gchar                 *message,
                    gboolean               is_member,
                    EmpathyContactManager *manager)
{
    g_signal_emit_by_name (manager, "members-changed",
                   contact, actor, reason, message, is_member);
}

static void
contact_manager_pendings_changed_cb (EmpathyTpContactList  *list,
                     EmpathyContact        *contact,
                     EmpathyContact        *actor,
                     guint                  reason,
                     gchar                 *message,
                     gboolean               is_pending,
                     EmpathyContactManager *manager)
{
    g_signal_emit_by_name (manager, "pendings-changed",
                   contact, actor, reason, message, is_pending);
}

static void
contact_manager_groups_changed_cb (EmpathyTpContactList  *list,
                   EmpathyContact        *contact,
                   gchar                 *group,
                   gboolean               is_member,
                   EmpathyContactManager *manager)
{
    g_signal_emit_by_name (manager, "groups-changed",
                   contact, group, is_member);
}

static void
contact_manager_invalidated_cb (TpProxy *connection,
                guint    domain,
                gint     code,
                gchar   *message,
                EmpathyContactManager *manager)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyTpContactList *list;

    DEBUG ("Removing connection: %s (%s)",
        tp_proxy_get_object_path (TP_PROXY (connection)),
        message);

    list = g_hash_table_lookup (priv->lists, connection);
    if (list) {
        empathy_tp_contact_list_remove_all (list);
        g_hash_table_remove (priv->lists, connection);
    }
}

static void
contact_manager_disconnect_foreach (gpointer key,
                    gpointer value,
                    gpointer user_data)
{
    TpConnection *connection = key;
    EmpathyTpContactList  *list = value;
    EmpathyContactManager *manager = user_data;

    /* Disconnect signals from the list */
    g_signal_handlers_disconnect_by_func (list,
                          contact_manager_members_changed_cb,
                          manager);
    g_signal_handlers_disconnect_by_func (list,
                          contact_manager_pendings_changed_cb,
                          manager);
    g_signal_handlers_disconnect_by_func (list,
                          contact_manager_groups_changed_cb,
                          manager);
    g_signal_handlers_disconnect_by_func (connection,
                          contact_manager_invalidated_cb,
                          manager);
}

static void
contact_manager_status_changed_cb (TpAccount *account,
                   guint old_status,
                   guint new_status,
                   guint reason,
                   gchar *dbus_error_name,
                   GHashTable *details,
                   EmpathyContactManager *self)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (self);
    EmpathyTpContactList      *list;
    TpConnection              *connection;

    if (new_status == TP_CONNECTION_STATUS_DISCONNECTED)
        /* No point to start tracking a connection which is about to die */
        return;

    connection = tp_account_get_connection (account);

    if (connection == NULL || g_hash_table_lookup (priv->lists, connection)) {
        return;
    }

    DEBUG ("Adding new connection: %s",
        tp_proxy_get_object_path (TP_PROXY (connection)));

    list = empathy_tp_contact_list_new (connection);
    g_hash_table_insert (priv->lists, g_object_ref (connection), list);
    g_signal_connect (connection, "invalidated",
              G_CALLBACK (contact_manager_invalidated_cb),
              self);

    /* Connect signals */
    g_signal_connect (list, "members-changed",
              G_CALLBACK (contact_manager_members_changed_cb),
              self);
    g_signal_connect (list, "pendings-changed",
              G_CALLBACK (contact_manager_pendings_changed_cb),
              self);
    g_signal_connect (list, "groups-changed",
              G_CALLBACK (contact_manager_groups_changed_cb),
              self);
}

static void
contact_manager_validity_changed_cb (TpAccountManager *account_manager,
                     TpAccount *account,
                     gboolean valid,
                     EmpathyContactManager *manager)
{
    if (valid) {
        tp_g_signal_connect_object (account, "status-changed",
                G_CALLBACK (contact_manager_status_changed_cb),
                manager, 0);
    }
}

static void
contact_manager_finalize (GObject *object)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (object);

    g_hash_table_foreach (priv->lists,
                  contact_manager_disconnect_foreach,
                  object);
    g_hash_table_destroy (priv->lists);

    g_object_unref (priv->account_manager);
}

static GObject *
contact_manager_constructor (GType type,
                 guint n_props,
                 GObjectConstructParam *props)
{
    GObject *retval;

    if (manager_singleton) {
        retval = g_object_ref (manager_singleton);
    } else {
        retval = G_OBJECT_CLASS (empathy_contact_manager_parent_class)->constructor
            (type, n_props, props);

        manager_singleton = EMPATHY_CONTACT_MANAGER (retval);
        g_object_add_weak_pointer (retval, (gpointer) &manager_singleton);
    }

    return retval;
}

/**
 * empathy_contact_manager_initialized:
 *
 * Reports whether or not the singleton has already been created.
 *
 * There can be instances where you want to access the #EmpathyContactManager
 * only if it has been set up for this process.
 *
 * Returns: %TRUE if the #EmpathyContactManager singleton has previously
 * been initialized.
 */
gboolean
empathy_contact_manager_initialized (void)
{
    return (manager_singleton != NULL);
}

static void
empathy_contact_manager_class_init (EmpathyContactManagerClass *klass)
{
    GObjectClass *object_class = G_OBJECT_CLASS (klass);

    object_class->finalize = contact_manager_finalize;
    object_class->constructor = contact_manager_constructor;

    g_type_class_add_private (object_class, sizeof (EmpathyContactManagerPriv));
}

static void
account_manager_prepared_cb (GObject *source_object,
                 GAsyncResult *result,
                 gpointer user_data)
{
    GList *accounts, *l;
    EmpathyContactManager *manager = user_data;
    TpAccountManager *account_manager = TP_ACCOUNT_MANAGER (source_object);
    GError *error = NULL;

    if (!tp_proxy_prepare_finish (account_manager, result, &error)) {
        DEBUG ("Failed to prepare account manager: %s", error->message);
        g_error_free (error);
        return;
    }

    accounts = tp_account_manager_get_valid_accounts (account_manager);

    for (l = accounts; l != NULL; l = l->next) {
        TpAccount *account = l->data;
        TpConnection *conn = tp_account_get_connection (account);

        if (conn != NULL) {
            contact_manager_status_changed_cb (account, 0, 0, 0,
                               NULL, NULL, manager);
        }

        tp_g_signal_connect_object (account, "status-changed",
            G_CALLBACK (contact_manager_status_changed_cb),
            manager, 0);
    }
    g_list_free (accounts);

    tp_g_signal_connect_object (account_manager, "account-validity-changed",
                 G_CALLBACK (contact_manager_validity_changed_cb),
                 manager, 0);
}

static void
empathy_contact_manager_init (EmpathyContactManager *manager)
{
    EmpathyContactManagerPriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (manager,
        EMPATHY_TYPE_CONTACT_MANAGER, EmpathyContactManagerPriv);

    manager->priv = priv;
    priv->lists = g_hash_table_new_full (empathy_proxy_hash,
                         empathy_proxy_equal,
                         (GDestroyNotify) g_object_unref,
                         (GDestroyNotify) g_object_unref);

    priv->account_manager = tp_account_manager_dup ();

    tp_proxy_prepare_async (priv->account_manager, NULL,
        account_manager_prepared_cb, manager);
}

EmpathyContactManager *
empathy_contact_manager_dup_singleton (void)
{
    return g_object_new (EMPATHY_TYPE_CONTACT_MANAGER, NULL);
}

EmpathyTpContactList *
empathy_contact_manager_get_list (EmpathyContactManager *manager,
                  TpConnection          *connection)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), NULL);
    g_return_val_if_fail (TP_IS_CONNECTION (connection), NULL);

    return g_hash_table_lookup (priv->lists, connection);
}

static void
contact_manager_add (EmpathyContactList *manager,
             EmpathyContact     *contact,
             const gchar        *message)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list) {
        empathy_contact_list_add (list, contact, message);
    }
}

static void
contact_manager_remove (EmpathyContactList *manager,
            EmpathyContact     *contact,
            const gchar        *message)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list) {
        empathy_contact_list_remove (list, contact, message);
    }
}

static void
contact_manager_get_members_foreach (TpConnection          *connection,
                     EmpathyTpContactList  *list,
                     GList                **contacts)
{
    GList *l;

    l = empathy_contact_list_get_members (EMPATHY_CONTACT_LIST (list));
    *contacts = g_list_concat (*contacts, l);
}

static GList *
contact_manager_get_members (EmpathyContactList *manager)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    GList                     *contacts = NULL;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), NULL);

    g_hash_table_foreach (priv->lists,
                  (GHFunc) contact_manager_get_members_foreach,
                  &contacts);

    return contacts;
}

static void
contact_manager_get_pendings_foreach (TpConnection          *connection,
                      EmpathyTpContactList  *list,
                      GList                **contacts)
{
    GList *l;

    l = empathy_contact_list_get_pendings (EMPATHY_CONTACT_LIST (list));
    *contacts = g_list_concat (*contacts, l);
}

static GList *
contact_manager_get_pendings (EmpathyContactList *manager)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    GList                     *contacts = NULL;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), NULL);

    g_hash_table_foreach (priv->lists,
                  (GHFunc) contact_manager_get_pendings_foreach,
                  &contacts);

    return contacts;
}

static void
contact_manager_get_all_groups_foreach (TpConnection          *connection,
                    EmpathyTpContactList  *list,
                    GList                **all_groups)
{
    GList *groups, *l;

    groups = empathy_contact_list_get_all_groups (EMPATHY_CONTACT_LIST (list));
    for (l = groups; l; l = l->next) {
        if (!g_list_find_custom (*all_groups,
                     l->data,
                     (GCompareFunc) strcmp)) {
            *all_groups = g_list_prepend (*all_groups, l->data);
        } else {
            g_free (l->data);
        }
    }

    g_list_free (groups);
}

static GList *
contact_manager_get_all_groups (EmpathyContactList *manager)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    GList                     *groups = NULL;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), NULL);

    g_hash_table_foreach (priv->lists,
                  (GHFunc) contact_manager_get_all_groups_foreach,
                  &groups);

    return groups;
}

static GList *
contact_manager_get_groups (EmpathyContactList *manager,
                EmpathyContact     *contact)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), NULL);

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list) {
        return empathy_contact_list_get_groups (list, contact);
    }

    return NULL;
}

static void
contact_manager_add_to_group (EmpathyContactList *manager,
                  EmpathyContact     *contact,
                  const gchar        *group)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list) {
        empathy_contact_list_add_to_group (list, contact, group);
    }
}

static void
contact_manager_remove_from_group (EmpathyContactList *manager,
                   EmpathyContact     *contact,
                   const gchar        *group)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list) {
        empathy_contact_list_remove_from_group (list, contact, group);
    }
}

typedef struct {
    const gchar *old_group;
    const gchar *new_group;
} RenameGroupData;

static void
contact_manager_rename_group_foreach (TpConnection         *connection,
                      EmpathyTpContactList *list,
                      RenameGroupData      *data)
{
    empathy_contact_list_rename_group (EMPATHY_CONTACT_LIST (list),
                       data->old_group,
                       data->new_group);
}

static void
contact_manager_rename_group (EmpathyContactList *manager,
                  const gchar        *old_group,
                  const gchar        *new_group)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    RenameGroupData            data;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    data.old_group = old_group;
    data.new_group = new_group;
    g_hash_table_foreach (priv->lists,
                  (GHFunc) contact_manager_rename_group_foreach,
                  &data);
}

static void contact_manager_remove_group_foreach (TpConnection         *connection,
                          EmpathyTpContactList *list,
                          const gchar *group)
{
    empathy_contact_list_remove_group (EMPATHY_CONTACT_LIST (list),
                       group);
}

static void
contact_manager_remove_group (EmpathyContactList *manager,
                  const gchar *group)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    g_hash_table_foreach (priv->lists,
                  (GHFunc) contact_manager_remove_group_foreach,
                  (gpointer) group);
}

static void
contact_manager_set_blocked (EmpathyContactList *manager,
                 EmpathyContact     *contact,
                 gboolean            blocked,
                 gboolean            abusive)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager));

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list != NULL) {
        empathy_contact_list_set_blocked (list, contact,
                          blocked, abusive);
    }
}

static gboolean
contact_manager_get_blocked (EmpathyContactList *manager,
                 EmpathyContact     *contact)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    TpConnection              *connection;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), FALSE);

    connection = empathy_contact_get_connection (contact);
    list = g_hash_table_lookup (priv->lists, connection);

    if (list != NULL) {
        return empathy_contact_list_get_blocked (list, contact);
    } else {
        return FALSE;
    }
}

static void
contact_manager_iface_init (EmpathyContactListIface *iface)
{
    iface->add               = contact_manager_add;
    iface->remove            = contact_manager_remove;
    iface->get_members       = contact_manager_get_members;
    iface->get_pendings      = contact_manager_get_pendings;
    iface->get_all_groups    = contact_manager_get_all_groups;
    iface->get_groups        = contact_manager_get_groups;
    iface->add_to_group      = contact_manager_add_to_group;
    iface->remove_from_group = contact_manager_remove_from_group;
    iface->rename_group      = contact_manager_rename_group;
    iface->remove_group  = contact_manager_remove_group;
    iface->set_blocked   = contact_manager_set_blocked;
    iface->get_blocked   = contact_manager_get_blocked;
}

EmpathyContactListFlags
empathy_contact_manager_get_flags_for_connection (
                EmpathyContactManager *manager,
                TpConnection          *connection)
{
    EmpathyContactManagerPriv *priv = GET_PRIV (manager);
    EmpathyContactList        *list;
    EmpathyContactListFlags    flags;

    g_return_val_if_fail (EMPATHY_IS_CONTACT_MANAGER (manager), FALSE);
    g_return_val_if_fail (connection != NULL, FALSE);

    list = g_hash_table_lookup (priv->lists, connection);
    if (list == NULL) {
        return FALSE;
    }
    flags = empathy_contact_list_get_flags (list);

    return flags;
}