1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * Copyright (C) 2005-2007 Imendio AB
4 * Copyright (C) 2007-2008 Collabora Ltd.
6 * This program is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU General Public License as
8 * published by the Free Software Foundation; either version 2 of the
9 * License, or (at your option) any later version.
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
16 * You should have received a copy of the GNU General Public
17 * License along with this program; if not, write to the
18 * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
19 * Boston, MA 02110-1301 USA
21 * Authors: Mikael Hallendal <micke@imendio.com>
22 * Martyn Russell <martyn@imendio.com>
23 * Xavier Claessens <xclaesse@gmail.com>
31 #include <glib/gi18n-lib.h>
34 #include <telepathy-glib/util.h>
36 #include <libempathy/empathy-utils.h>
37 #include <libempathy/empathy-tp-chat.h>
38 #include <libempathy/empathy-enum-types.h>
39 #include <libempathy/empathy-contact-manager.h>
41 #include "empathy-contact-list-store.h"
42 #include "empathy-ui-utils.h"
43 #include "empathy-gtk-enum-types.h"
44 #include "empathy-images.h"
46 #define DEBUG_FLAG EMPATHY_DEBUG_CONTACT
47 #include <libempathy/empathy-debug.h>
49 /* Active users are those which have recently changed state
50 * (e.g. online, offline or from normal to a busy state).
53 /* Time in seconds user is shown as active */
54 #define ACTIVE_USER_SHOW_TIME 7
56 /* Time in seconds after connecting which we wait before active users are enabled */
57 #define ACTIVE_USER_WAIT_TO_ENABLE_TIME 5
59 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyContactListStore)
61 EmpathyContactList *list;
62 gboolean show_offline;
63 gboolean show_avatars;
66 gboolean show_protocols;
68 EmpathyContactListStoreSort sort_criterium;
71 gboolean dispose_has_run;
72 GHashTable *status_icons;
73 } EmpathyContactListStorePriv;
82 EmpathyContact *contact;
88 EmpathyContactListStore *store;
89 EmpathyContact *contact;
93 static void contact_list_store_dispose (GObject *object);
94 static void contact_list_store_get_property (GObject *object,
98 static void contact_list_store_set_property (GObject *object,
102 static void contact_list_store_setup (EmpathyContactListStore *store);
103 static gboolean contact_list_store_inibit_active_cb (EmpathyContactListStore *store);
104 static void contact_list_store_members_changed_cb (EmpathyContactList *list_iface,
105 EmpathyContact *contact,
106 EmpathyContact *actor,
110 EmpathyContactListStore *store);
111 static void contact_list_store_favourites_changed_cb (EmpathyContactList *list_iface,
112 EmpathyContact *contact,
113 gboolean is_favourite,
114 EmpathyContactListStore *store);
115 static void contact_list_store_member_renamed_cb (EmpathyContactList *list_iface,
116 EmpathyContact *old_contact,
117 EmpathyContact *new_contact,
120 EmpathyContactListStore *store);
121 static void contact_list_store_groups_changed_cb (EmpathyContactList *list_iface,
122 EmpathyContact *contact,
125 EmpathyContactListStore *store);
126 static void contact_list_store_add_contact (EmpathyContactListStore *store,
127 EmpathyContact *contact);
128 static void contact_list_store_remove_contact (EmpathyContactListStore *store,
129 EmpathyContact *contact);
130 static void contact_list_store_contact_update (EmpathyContactListStore *store,
131 EmpathyContact *contact);
132 static void contact_list_store_contact_updated_cb (EmpathyContact *contact,
134 EmpathyContactListStore *store);
135 static void contact_list_store_contact_set_active (EmpathyContactListStore *store,
136 EmpathyContact *contact,
138 gboolean set_changed);
139 static ShowActiveData * contact_list_store_contact_active_new (EmpathyContactListStore *store,
140 EmpathyContact *contact,
142 static void contact_list_store_contact_active_free (ShowActiveData *data);
143 static gboolean contact_list_store_contact_active_cb (ShowActiveData *data);
144 static gboolean contact_list_store_get_group_foreach (GtkTreeModel *model,
148 static void contact_list_store_get_group (EmpathyContactListStore *store,
150 GtkTreeIter *iter_group_to_set,
151 GtkTreeIter *iter_separator_to_set,
153 gboolean is_fake_group);
154 static gint contact_list_store_state_sort_func (GtkTreeModel *model,
158 static gint contact_list_store_name_sort_func (GtkTreeModel *model,
162 static gboolean contact_list_store_find_contact_foreach (GtkTreeModel *model,
166 static GList * contact_list_store_find_contact (EmpathyContactListStore *store,
167 EmpathyContact *contact);
168 static gboolean contact_list_store_update_list_mode_foreach (GtkTreeModel *model,
171 EmpathyContactListStore *store);
184 G_DEFINE_TYPE (EmpathyContactListStore, empathy_contact_list_store, GTK_TYPE_TREE_STORE);
187 contact_list_store_chat_state_changed_cb (TpChannel *self,
188 guint contact_handle,
192 EmpathyContactListStorePriv *priv = GET_PRIV (store);
195 contacts = empathy_contact_list_get_members (priv->list);
197 /* Find the contact in the list. After that l is the list elem or NULL */
198 for (l = contacts; l != NULL; l = l->next) {
199 if (empathy_contact_get_handle (EMPATHY_CONTACT (l->data)) ==
206 contact_list_store_contact_update (store, l->data);
208 g_list_foreach (contacts, (GFunc) g_object_unref, NULL);
209 g_list_free (contacts);
213 contact_list_store_iface_setup (gpointer user_data)
215 EmpathyContactListStore *store = user_data;
216 EmpathyContactListStorePriv *priv = GET_PRIV (store);
219 /* Signal connection. */
220 g_signal_connect (priv->list,
222 G_CALLBACK (contact_list_store_member_renamed_cb),
224 g_signal_connect (priv->list,
226 G_CALLBACK (contact_list_store_members_changed_cb),
228 g_signal_connect (priv->list,
229 "favourites-changed",
230 G_CALLBACK (contact_list_store_favourites_changed_cb),
232 g_signal_connect (priv->list,
234 G_CALLBACK (contact_list_store_groups_changed_cb),
237 if (EMPATHY_IS_TP_CHAT (priv->list)) {
240 channel = empathy_tp_chat_get_channel (EMPATHY_TP_CHAT (priv->list));
241 if (!tp_proxy_is_prepared (channel, TP_CHANNEL_FEATURE_CHAT_STATES)) {
242 DEBUG ("Chat state feature not prepared");
244 g_signal_connect (channel,
245 "chat-state-changed",
246 G_CALLBACK (contact_list_store_chat_state_changed_cb),
251 /* Add contacts already created. */
252 contacts = empathy_contact_list_get_members (priv->list);
253 for (l = contacts; l; l = l->next) {
254 contact_list_store_members_changed_cb (priv->list, l->data,
259 g_object_unref (l->data);
261 g_list_free (contacts);
263 priv->setup_idle_id = 0;
269 contact_list_store_set_contact_list (EmpathyContactListStore *store,
270 EmpathyContactList *list_iface)
272 EmpathyContactListStorePriv *priv = GET_PRIV (store);
274 priv->list = g_object_ref (list_iface);
276 /* Let a chance to have all properties set before populating */
277 priv->setup_idle_id = g_idle_add (contact_list_store_iface_setup, store);
281 empathy_contact_list_store_class_init (EmpathyContactListStoreClass *klass)
283 GObjectClass *object_class = G_OBJECT_CLASS (klass);
285 object_class->dispose = contact_list_store_dispose;
286 object_class->get_property = contact_list_store_get_property;
287 object_class->set_property = contact_list_store_set_property;
289 g_object_class_install_property (object_class,
291 g_param_spec_object ("contact-list",
292 "The contact list iface",
293 "The contact list iface",
294 EMPATHY_TYPE_CONTACT_LIST,
295 G_PARAM_CONSTRUCT_ONLY |
297 g_object_class_install_property (object_class,
299 g_param_spec_boolean ("show-offline",
301 "Whether contact list should display "
305 g_object_class_install_property (object_class,
307 g_param_spec_boolean ("show-avatars",
309 "Whether contact list should display "
310 "avatars for contacts",
313 g_object_class_install_property (object_class,
315 g_param_spec_boolean ("show-protocols",
317 "Whether contact list should display "
318 "protocols for contacts",
321 g_object_class_install_property (object_class,
323 g_param_spec_boolean ("show-groups",
325 "Whether contact list should display "
329 g_object_class_install_property (object_class,
331 g_param_spec_boolean ("is-compact",
333 "Whether the contact list is in compact mode or not",
337 g_object_class_install_property (object_class,
339 g_param_spec_enum ("sort-criterium",
341 "The sort criterium to use for sorting the contact list",
342 EMPATHY_TYPE_CONTACT_LIST_STORE_SORT,
343 EMPATHY_CONTACT_LIST_STORE_SORT_NAME,
346 g_type_class_add_private (object_class, sizeof (EmpathyContactListStorePriv));
350 empathy_contact_list_store_init (EmpathyContactListStore *store)
352 EmpathyContactListStorePriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (store,
353 EMPATHY_TYPE_CONTACT_LIST_STORE, EmpathyContactListStorePriv);
356 priv->show_avatars = TRUE;
357 priv->show_groups = TRUE;
358 priv->show_protocols = FALSE;
359 priv->inhibit_active = g_timeout_add_seconds (ACTIVE_USER_WAIT_TO_ENABLE_TIME,
360 (GSourceFunc) contact_list_store_inibit_active_cb,
362 priv->status_icons = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_object_unref);
363 contact_list_store_setup (store);
367 contact_list_store_dispose (GObject *object)
369 EmpathyContactListStorePriv *priv = GET_PRIV (object);
372 if (priv->dispose_has_run)
374 priv->dispose_has_run = TRUE;
376 contacts = empathy_contact_list_get_members (priv->list);
377 for (l = contacts; l; l = l->next) {
378 g_signal_handlers_disconnect_by_func (l->data,
379 G_CALLBACK (contact_list_store_contact_updated_cb),
382 g_object_unref (l->data);
384 g_list_free (contacts);
386 g_signal_handlers_disconnect_by_func (priv->list,
387 G_CALLBACK (contact_list_store_member_renamed_cb),
389 g_signal_handlers_disconnect_by_func (priv->list,
390 G_CALLBACK (contact_list_store_members_changed_cb),
392 g_signal_handlers_disconnect_by_func (priv->list,
393 G_CALLBACK (contact_list_store_favourites_changed_cb),
395 g_signal_handlers_disconnect_by_func (priv->list,
396 G_CALLBACK (contact_list_store_groups_changed_cb),
398 g_object_unref (priv->list);
400 if (priv->inhibit_active) {
401 g_source_remove (priv->inhibit_active);
404 if (priv->setup_idle_id != 0) {
405 g_source_remove (priv->setup_idle_id);
408 g_hash_table_destroy (priv->status_icons);
409 G_OBJECT_CLASS (empathy_contact_list_store_parent_class)->dispose (object);
413 contact_list_store_get_property (GObject *object,
418 EmpathyContactListStorePriv *priv;
420 priv = GET_PRIV (object);
423 case PROP_CONTACT_LIST:
424 g_value_set_object (value, priv->list);
426 case PROP_SHOW_OFFLINE:
427 g_value_set_boolean (value, priv->show_offline);
429 case PROP_SHOW_AVATARS:
430 g_value_set_boolean (value, priv->show_avatars);
432 case PROP_SHOW_PROTOCOLS:
433 g_value_set_boolean (value, priv->show_protocols);
435 case PROP_SHOW_GROUPS:
436 g_value_set_boolean (value, priv->show_groups);
438 case PROP_IS_COMPACT:
439 g_value_set_boolean (value, priv->is_compact);
441 case PROP_SORT_CRITERIUM:
442 g_value_set_enum (value, priv->sort_criterium);
445 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
451 contact_list_store_set_property (GObject *object,
457 case PROP_CONTACT_LIST:
458 contact_list_store_set_contact_list (EMPATHY_CONTACT_LIST_STORE (object),
459 g_value_get_object (value));
461 case PROP_SHOW_OFFLINE:
462 empathy_contact_list_store_set_show_offline (EMPATHY_CONTACT_LIST_STORE (object),
463 g_value_get_boolean (value));
465 case PROP_SHOW_AVATARS:
466 empathy_contact_list_store_set_show_avatars (EMPATHY_CONTACT_LIST_STORE (object),
467 g_value_get_boolean (value));
469 case PROP_SHOW_PROTOCOLS:
470 empathy_contact_list_store_set_show_protocols (EMPATHY_CONTACT_LIST_STORE (object),
471 g_value_get_boolean (value));
473 case PROP_SHOW_GROUPS:
474 empathy_contact_list_store_set_show_groups (EMPATHY_CONTACT_LIST_STORE (object),
475 g_value_get_boolean (value));
477 case PROP_IS_COMPACT:
478 empathy_contact_list_store_set_is_compact (EMPATHY_CONTACT_LIST_STORE (object),
479 g_value_get_boolean (value));
481 case PROP_SORT_CRITERIUM:
482 empathy_contact_list_store_set_sort_criterium (EMPATHY_CONTACT_LIST_STORE (object),
483 g_value_get_enum (value));
486 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
491 EmpathyContactListStore *
492 empathy_contact_list_store_new (EmpathyContactList *list_iface)
494 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST (list_iface), NULL);
496 return g_object_new (EMPATHY_TYPE_CONTACT_LIST_STORE,
497 "contact-list", list_iface,
502 empathy_contact_list_store_get_list_iface (EmpathyContactListStore *store)
504 EmpathyContactListStorePriv *priv;
506 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), FALSE);
508 priv = GET_PRIV (store);
514 empathy_contact_list_store_get_show_offline (EmpathyContactListStore *store)
516 EmpathyContactListStorePriv *priv;
518 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), FALSE);
520 priv = GET_PRIV (store);
522 return priv->show_offline;
526 empathy_contact_list_store_set_show_offline (EmpathyContactListStore *store,
527 gboolean show_offline)
529 EmpathyContactListStorePriv *priv;
531 gboolean show_active;
533 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
535 priv = GET_PRIV (store);
537 priv->show_offline = show_offline;
538 show_active = priv->show_active;
540 /* Disable temporarily. */
541 priv->show_active = FALSE;
543 contacts = empathy_contact_list_get_members (priv->list);
544 for (l = contacts; l; l = l->next) {
545 contact_list_store_contact_update (store, l->data);
547 g_object_unref (l->data);
549 g_list_free (contacts);
551 /* Restore to original setting. */
552 priv->show_active = show_active;
554 g_object_notify (G_OBJECT (store), "show-offline");
558 empathy_contact_list_store_get_show_avatars (EmpathyContactListStore *store)
560 EmpathyContactListStorePriv *priv;
562 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), TRUE);
564 priv = GET_PRIV (store);
566 return priv->show_avatars;
570 empathy_contact_list_store_set_show_avatars (EmpathyContactListStore *store,
571 gboolean show_avatars)
573 EmpathyContactListStorePriv *priv;
576 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
578 priv = GET_PRIV (store);
580 priv->show_avatars = show_avatars;
582 model = GTK_TREE_MODEL (store);
584 gtk_tree_model_foreach (model,
585 (GtkTreeModelForeachFunc)
586 contact_list_store_update_list_mode_foreach,
589 g_object_notify (G_OBJECT (store), "show-avatars");
594 empathy_contact_list_store_get_show_protocols (EmpathyContactListStore *store)
596 EmpathyContactListStorePriv *priv;
598 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), TRUE);
600 priv = GET_PRIV (store);
602 return priv->show_protocols;
606 empathy_contact_list_store_set_show_protocols (EmpathyContactListStore *store,
607 gboolean show_protocols)
609 EmpathyContactListStorePriv *priv;
612 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
614 priv = GET_PRIV (store);
616 priv->show_protocols = show_protocols;
618 model = GTK_TREE_MODEL (store);
620 gtk_tree_model_foreach (model,
621 (GtkTreeModelForeachFunc)
622 contact_list_store_update_list_mode_foreach,
625 g_object_notify (G_OBJECT (store), "show-protocols");
629 empathy_contact_list_store_get_show_groups (EmpathyContactListStore *store)
631 EmpathyContactListStorePriv *priv;
633 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), TRUE);
635 priv = GET_PRIV (store);
637 return priv->show_groups;
641 empathy_contact_list_store_set_show_groups (EmpathyContactListStore *store,
642 gboolean show_groups)
644 EmpathyContactListStorePriv *priv;
646 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
648 priv = GET_PRIV (store);
650 if (priv->show_groups == show_groups) {
654 priv->show_groups = show_groups;
656 if (priv->setup_idle_id == 0) {
657 /* Remove all contacts and add them back, not optimized but
658 * that's the easy way :)
660 * This is only done if there's not a pending setup idle
661 * callback, otherwise it will race and the contacts will get
665 gtk_tree_store_clear (GTK_TREE_STORE (store));
666 contacts = empathy_contact_list_get_members (priv->list);
667 for (l = contacts; l; l = l->next) {
668 contact_list_store_members_changed_cb (priv->list,
674 g_object_unref (l->data);
676 g_list_free (contacts);
679 g_object_notify (G_OBJECT (store), "show-groups");
683 empathy_contact_list_store_get_is_compact (EmpathyContactListStore *store)
685 EmpathyContactListStorePriv *priv;
687 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), TRUE);
689 priv = GET_PRIV (store);
691 return priv->is_compact;
695 empathy_contact_list_store_set_is_compact (EmpathyContactListStore *store,
698 EmpathyContactListStorePriv *priv;
701 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
703 priv = GET_PRIV (store);
705 priv->is_compact = is_compact;
707 model = GTK_TREE_MODEL (store);
709 gtk_tree_model_foreach (model,
710 (GtkTreeModelForeachFunc)
711 contact_list_store_update_list_mode_foreach,
714 g_object_notify (G_OBJECT (store), "is-compact");
717 EmpathyContactListStoreSort
718 empathy_contact_list_store_get_sort_criterium (EmpathyContactListStore *store)
720 EmpathyContactListStorePriv *priv;
722 g_return_val_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store), 0);
724 priv = GET_PRIV (store);
726 return priv->sort_criterium;
730 empathy_contact_list_store_set_sort_criterium (EmpathyContactListStore *store,
731 EmpathyContactListStoreSort sort_criterium)
733 EmpathyContactListStorePriv *priv;
735 g_return_if_fail (EMPATHY_IS_CONTACT_LIST_STORE (store));
737 priv = GET_PRIV (store);
739 priv->sort_criterium = sort_criterium;
741 switch (sort_criterium) {
742 case EMPATHY_CONTACT_LIST_STORE_SORT_STATE:
743 gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
744 EMPATHY_CONTACT_LIST_STORE_COL_STATUS,
748 case EMPATHY_CONTACT_LIST_STORE_SORT_NAME:
749 gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (store),
750 EMPATHY_CONTACT_LIST_STORE_COL_NAME,
755 g_assert_not_reached ();
758 g_object_notify (G_OBJECT (store), "sort-criterium");
762 empathy_contact_list_store_row_separator_func (GtkTreeModel *model,
766 gboolean is_separator = FALSE;
768 g_return_val_if_fail (GTK_IS_TREE_MODEL (model), FALSE);
770 gtk_tree_model_get (model, iter,
771 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator,
778 empathy_contact_list_store_get_parent_group (GtkTreeModel *model,
780 gboolean *path_is_group,
781 gboolean *is_fake_group)
783 GtkTreeIter parent_iter, iter;
788 g_return_val_if_fail (GTK_IS_TREE_MODEL (model), NULL);
791 *path_is_group = FALSE;
794 if (!gtk_tree_model_get_iter (model, &iter, path)) {
798 gtk_tree_model_get (model, &iter,
799 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
800 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name,
807 if (!gtk_tree_model_iter_parent (model, &parent_iter, &iter)) {
813 gtk_tree_model_get (model, &iter,
814 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
815 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name,
816 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, &fake,
825 *path_is_group = TRUE;
828 if (is_fake_group != NULL)
829 *is_fake_group = fake;
835 contact_list_store_setup (EmpathyContactListStore *store)
837 EmpathyContactListStorePriv *priv;
839 GDK_TYPE_PIXBUF, /* Status pixbuf */
840 GDK_TYPE_PIXBUF, /* Avatar pixbuf */
841 G_TYPE_BOOLEAN, /* Avatar pixbuf visible */
842 G_TYPE_STRING, /* Name */
843 G_TYPE_UINT, /* Presence type */
844 G_TYPE_STRING, /* Status string */
845 G_TYPE_BOOLEAN, /* Compact view */
846 EMPATHY_TYPE_CONTACT, /* Contact type */
847 G_TYPE_BOOLEAN, /* Is group */
848 G_TYPE_BOOLEAN, /* Is active */
849 G_TYPE_BOOLEAN, /* Is online */
850 G_TYPE_BOOLEAN, /* Is separator */
851 G_TYPE_BOOLEAN, /* Can make audio calls */
852 G_TYPE_BOOLEAN, /* Can make video calls */
853 EMPATHY_TYPE_CONTACT_LIST_FLAGS, /* Flags */
854 G_TYPE_BOOLEAN, /* Is a fake group */
857 priv = GET_PRIV (store);
859 gtk_tree_store_set_column_types (GTK_TREE_STORE (store),
860 EMPATHY_CONTACT_LIST_STORE_COL_COUNT,
864 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (store),
865 EMPATHY_CONTACT_LIST_STORE_COL_NAME,
866 contact_list_store_name_sort_func,
868 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (store),
869 EMPATHY_CONTACT_LIST_STORE_COL_STATUS,
870 contact_list_store_state_sort_func,
873 priv->sort_criterium = EMPATHY_CONTACT_LIST_STORE_SORT_NAME;
874 empathy_contact_list_store_set_sort_criterium (store, priv->sort_criterium);
878 contact_list_store_inibit_active_cb (EmpathyContactListStore *store)
880 EmpathyContactListStorePriv *priv;
882 priv = GET_PRIV (store);
884 priv->show_active = TRUE;
885 priv->inhibit_active = 0;
891 contact_list_store_add_contact_and_connect (EmpathyContactListStore *store, EmpathyContact *contact)
893 g_signal_connect (contact, "notify::presence",
894 G_CALLBACK (contact_list_store_contact_updated_cb),
896 g_signal_connect (contact, "notify::presence-message",
897 G_CALLBACK (contact_list_store_contact_updated_cb),
899 g_signal_connect (contact, "notify::name",
900 G_CALLBACK (contact_list_store_contact_updated_cb),
902 g_signal_connect (contact, "notify::avatar",
903 G_CALLBACK (contact_list_store_contact_updated_cb),
905 g_signal_connect (contact, "notify::capabilities",
906 G_CALLBACK (contact_list_store_contact_updated_cb),
909 contact_list_store_add_contact (store, contact);
913 contact_list_store_remove_contact_and_disconnect (EmpathyContactListStore *store, EmpathyContact *contact)
915 g_signal_handlers_disconnect_by_func (contact,
916 G_CALLBACK (contact_list_store_contact_updated_cb),
919 contact_list_store_remove_contact (store, contact);
923 contact_list_store_members_changed_cb (EmpathyContactList *list_iface,
924 EmpathyContact *contact,
925 EmpathyContact *actor,
929 EmpathyContactListStore *store)
931 DEBUG ("Contact %s (%d) %s",
932 empathy_contact_get_id (contact),
933 empathy_contact_get_handle (contact),
934 is_member ? "added" : "removed");
937 contact_list_store_add_contact_and_connect (store, contact);
939 contact_list_store_remove_contact_and_disconnect (store, contact);
944 contact_list_store_favourites_changed_cb (EmpathyContactList *list_iface,
945 EmpathyContact *contact,
946 gboolean is_favourite,
947 EmpathyContactListStore *store)
949 DEBUG ("Contact %s (%d) is %s a favourite",
950 empathy_contact_get_id (contact),
951 empathy_contact_get_handle (contact),
952 is_favourite ? "now" : "no longer");
954 contact_list_store_remove_contact (store, contact);
955 contact_list_store_add_contact (store, contact);
959 contact_list_store_member_renamed_cb (EmpathyContactList *list_iface,
960 EmpathyContact *old_contact,
961 EmpathyContact *new_contact,
964 EmpathyContactListStore *store)
966 DEBUG ("Contact %s (%d) renamed to %s (%d)",
967 empathy_contact_get_id (old_contact),
968 empathy_contact_get_handle (old_contact),
969 empathy_contact_get_id (new_contact),
970 empathy_contact_get_handle (new_contact));
972 /* add the new contact */
973 contact_list_store_add_contact_and_connect (store, new_contact);
975 /* remove old contact */
976 contact_list_store_remove_contact_and_disconnect (store, old_contact);
980 contact_list_store_groups_changed_cb (EmpathyContactList *list_iface,
981 EmpathyContact *contact,
984 EmpathyContactListStore *store)
986 EmpathyContactListStorePriv *priv;
987 gboolean show_active;
989 priv = GET_PRIV (store);
991 DEBUG ("Updating groups for contact %s (%d)",
992 empathy_contact_get_id (contact),
993 empathy_contact_get_handle (contact));
995 /* We do this to make sure the groups are correct, if not, we
996 * would have to check the groups already set up for each
997 * contact and then see what has been updated.
999 show_active = priv->show_active;
1000 priv->show_active = FALSE;
1001 contact_list_store_remove_contact (store, contact);
1002 contact_list_store_add_contact (store, contact);
1003 priv->show_active = show_active;
1007 add_contact_to_store (GtkTreeStore *store,
1009 GtkTreeIter *parent,
1010 EmpathyContact *contact,
1011 EmpathyContactListFlags flags)
1013 gtk_tree_store_insert_with_values (store, iter, parent, 0,
1014 EMPATHY_CONTACT_LIST_STORE_COL_NAME, empathy_contact_get_alias (contact),
1015 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, contact,
1016 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, FALSE,
1017 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, FALSE,
1018 EMPATHY_CONTACT_LIST_STORE_COL_CAN_AUDIO_CALL,
1019 empathy_contact_get_capabilities (contact) &
1020 EMPATHY_CAPABILITIES_AUDIO,
1021 EMPATHY_CONTACT_LIST_STORE_COL_CAN_VIDEO_CALL,
1022 empathy_contact_get_capabilities (contact) &
1023 EMPATHY_CAPABILITIES_VIDEO,
1024 EMPATHY_CONTACT_LIST_STORE_COL_FLAGS, flags,
1029 contact_list_store_add_contact (EmpathyContactListStore *store,
1030 EmpathyContact *contact)
1032 EmpathyContactListStorePriv *priv;
1034 GList *groups = NULL, *l;
1035 TpConnection *connection;
1036 EmpathyContactListFlags flags = 0;
1037 char *protocol_name;
1039 priv = GET_PRIV (store);
1041 if (EMP_STR_EMPTY (empathy_contact_get_alias (contact)) ||
1042 (!priv->show_offline && !empathy_contact_is_online (contact))) {
1046 if (priv->show_groups) {
1047 groups = empathy_contact_list_get_groups (priv->list, contact);
1050 connection = empathy_contact_get_connection (contact);
1051 if (EMPATHY_IS_CONTACT_MANAGER (priv->list)) {
1052 flags = empathy_contact_manager_get_flags_for_connection (
1053 EMPATHY_CONTACT_MANAGER (priv->list), connection);
1056 tp_connection_parse_object_path (connection, &protocol_name, NULL);
1059 GtkTreeIter iter_group, *parent;
1061 parent = &iter_group;
1063 if (!priv->show_groups) {
1065 } else if (!tp_strdiff (protocol_name, "local-xmpp")) {
1066 /* these are People Nearby */
1067 contact_list_store_get_group (store,
1068 EMPATHY_CONTACT_LIST_STORE_PEOPLE_NEARBY,
1069 &iter_group, NULL, NULL, TRUE);
1071 contact_list_store_get_group (store,
1072 EMPATHY_CONTACT_LIST_STORE_UNGROUPED,
1073 &iter_group, NULL, NULL, TRUE);
1076 add_contact_to_store (GTK_TREE_STORE (store), &iter, parent,
1080 g_free (protocol_name);
1082 /* Else add to each group. */
1083 for (l = groups; l; l = l->next) {
1084 GtkTreeIter iter_group;
1086 contact_list_store_get_group (store, l->data, &iter_group, NULL, NULL, FALSE);
1088 add_contact_to_store (GTK_TREE_STORE (store), &iter, &iter_group, contact, flags);
1091 g_list_free (groups);
1093 if (priv->show_groups &&
1094 empathy_contact_list_is_favourite (priv->list, contact)) {
1095 /* Add contact to the fake 'Favorites' group */
1096 GtkTreeIter iter_group;
1098 contact_list_store_get_group (store, EMPATHY_CONTACT_LIST_STORE_FAVORITE,
1099 &iter_group, NULL, NULL, TRUE);
1101 add_contact_to_store (GTK_TREE_STORE (store), &iter, &iter_group, contact, flags);
1104 contact_list_store_contact_update (store, contact);
1108 contact_list_store_remove_contact (EmpathyContactListStore *store,
1109 EmpathyContact *contact)
1111 GtkTreeModel *model;
1114 iters = contact_list_store_find_contact (store, contact);
1119 /* Clean up model */
1120 model = GTK_TREE_MODEL (store);
1122 for (l = iters; l; l = l->next) {
1125 /* NOTE: it is only <= 2 here because we have
1126 * separators after the group name, otherwise it
1129 if (gtk_tree_model_iter_parent (model, &parent, l->data) &&
1130 gtk_tree_model_iter_n_children (model, &parent) <= 2) {
1131 gtk_tree_store_remove (GTK_TREE_STORE (store), &parent);
1133 gtk_tree_store_remove (GTK_TREE_STORE (store), l->data);
1137 g_list_foreach (iters, (GFunc) gtk_tree_iter_free, NULL);
1138 g_list_free (iters);
1142 contact_list_store_contact_update (EmpathyContactListStore *store,
1143 EmpathyContact *contact)
1145 EmpathyContactListStorePriv *priv;
1146 ShowActiveData *data;
1147 GtkTreeModel *model;
1150 gboolean should_be_in_list;
1151 gboolean was_online = TRUE;
1152 gboolean now_online = FALSE;
1153 gboolean set_model = FALSE;
1154 gboolean do_remove = FALSE;
1155 gboolean do_set_active = FALSE;
1156 gboolean do_set_refresh = FALSE;
1157 gboolean show_avatar = FALSE;
1158 GdkPixbuf *pixbuf_avatar;
1159 GdkPixbuf *pixbuf_status;
1161 priv = GET_PRIV (store);
1163 model = GTK_TREE_MODEL (store);
1165 iters = contact_list_store_find_contact (store, contact);
1172 /* Get online state now. */
1173 now_online = empathy_contact_is_online (contact);
1175 if (priv->show_offline || now_online) {
1176 should_be_in_list = TRUE;
1178 should_be_in_list = FALSE;
1181 if (!in_list && !should_be_in_list) {
1182 /* Nothing to do. */
1183 DEBUG ("Contact:'%s' in list:NO, should be:NO",
1184 empathy_contact_get_alias (contact));
1186 g_list_foreach (iters, (GFunc) gtk_tree_iter_free, NULL);
1187 g_list_free (iters);
1190 else if (in_list && !should_be_in_list) {
1191 DEBUG ("Contact:'%s' in list:YES, should be:NO",
1192 empathy_contact_get_alias (contact));
1194 if (priv->show_active) {
1196 do_set_active = TRUE;
1197 do_set_refresh = TRUE;
1200 DEBUG ("Remove item (after timeout)");
1202 DEBUG ("Remove item (now)!");
1203 contact_list_store_remove_contact (store, contact);
1206 else if (!in_list && should_be_in_list) {
1207 DEBUG ("Contact:'%s' in list:NO, should be:YES",
1208 empathy_contact_get_alias (contact));
1210 contact_list_store_add_contact (store, contact);
1212 if (priv->show_active) {
1213 do_set_active = TRUE;
1215 DEBUG ("Set active (contact added)");
1218 DEBUG ("Contact:'%s' in list:YES, should be:YES",
1219 empathy_contact_get_alias (contact));
1221 /* Get online state before. */
1222 if (iters && g_list_length (iters) > 0) {
1223 gtk_tree_model_get (model, iters->data,
1224 EMPATHY_CONTACT_LIST_STORE_COL_IS_ONLINE, &was_online,
1228 /* Is this really an update or an online/offline. */
1229 if (priv->show_active) {
1230 if (was_online != now_online) {
1231 do_set_active = TRUE;
1232 do_set_refresh = TRUE;
1234 DEBUG ("Set active (contact updated %s)",
1235 was_online ? "online -> offline" :
1236 "offline -> online");
1238 /* Was TRUE for presence updates. */
1239 /* do_set_active = FALSE; */
1240 do_set_refresh = TRUE;
1242 DEBUG ("Set active (contact updated)");
1249 if (priv->show_avatars && !priv->is_compact) {
1252 pixbuf_avatar = empathy_pixbuf_avatar_from_contact_scaled (contact, 32, 32);
1253 pixbuf_status = contact_list_store_get_contact_status_icon (store, contact);
1254 for (l = iters; l && set_model; l = l->next) {
1255 gtk_tree_store_set (GTK_TREE_STORE (store), l->data,
1256 EMPATHY_CONTACT_LIST_STORE_COL_ICON_STATUS, pixbuf_status,
1257 EMPATHY_CONTACT_LIST_STORE_COL_PIXBUF_AVATAR, pixbuf_avatar,
1258 EMPATHY_CONTACT_LIST_STORE_COL_PIXBUF_AVATAR_VISIBLE, show_avatar,
1259 EMPATHY_CONTACT_LIST_STORE_COL_NAME, empathy_contact_get_alias (contact),
1260 EMPATHY_CONTACT_LIST_STORE_COL_PRESENCE_TYPE,
1261 empathy_contact_get_presence (contact),
1262 EMPATHY_CONTACT_LIST_STORE_COL_STATUS,
1263 empathy_contact_get_presence_message (contact),
1264 EMPATHY_CONTACT_LIST_STORE_COL_COMPACT, priv->is_compact,
1265 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, FALSE,
1266 EMPATHY_CONTACT_LIST_STORE_COL_IS_ONLINE, now_online,
1267 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, FALSE,
1268 EMPATHY_CONTACT_LIST_STORE_COL_CAN_AUDIO_CALL,
1269 empathy_contact_get_capabilities (contact) &
1270 EMPATHY_CAPABILITIES_AUDIO,
1271 EMPATHY_CONTACT_LIST_STORE_COL_CAN_VIDEO_CALL,
1272 empathy_contact_get_capabilities (contact) &
1273 EMPATHY_CAPABILITIES_VIDEO,
1277 if (pixbuf_avatar) {
1278 g_object_unref (pixbuf_avatar);
1281 if (priv->show_active && do_set_active) {
1282 contact_list_store_contact_set_active (store, contact, do_set_active, do_set_refresh);
1284 if (do_set_active) {
1285 data = contact_list_store_contact_active_new (store, contact, do_remove);
1286 g_timeout_add_seconds (ACTIVE_USER_SHOW_TIME,
1287 (GSourceFunc) contact_list_store_contact_active_cb,
1292 /* FIXME: when someone goes online then offline quickly, the
1293 * first timeout sets the user to be inactive and the second
1294 * timeout removes the user from the contact list, really we
1295 * should remove the first timeout.
1297 g_list_foreach (iters, (GFunc) gtk_tree_iter_free, NULL);
1298 g_list_free (iters);
1302 contact_list_store_contact_updated_cb (EmpathyContact *contact,
1304 EmpathyContactListStore *store)
1306 DEBUG ("Contact:'%s' updated, checking roster is in sync...",
1307 empathy_contact_get_alias (contact));
1309 contact_list_store_contact_update (store, contact);
1313 contact_list_store_contact_set_active (EmpathyContactListStore *store,
1314 EmpathyContact *contact,
1316 gboolean set_changed)
1318 GtkTreeModel *model;
1321 model = GTK_TREE_MODEL (store);
1323 iters = contact_list_store_find_contact (store, contact);
1324 for (l = iters; l; l = l->next) {
1327 gtk_tree_store_set (GTK_TREE_STORE (store), l->data,
1328 EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, active,
1331 DEBUG ("Set item %s", active ? "active" : "inactive");
1334 path = gtk_tree_model_get_path (model, l->data);
1335 gtk_tree_model_row_changed (model, path, l->data);
1336 gtk_tree_path_free (path);
1340 g_list_foreach (iters, (GFunc) gtk_tree_iter_free, NULL);
1341 g_list_free (iters);
1345 static ShowActiveData *
1346 contact_list_store_contact_active_new (EmpathyContactListStore *store,
1347 EmpathyContact *contact,
1350 ShowActiveData *data;
1352 DEBUG ("Contact:'%s' now active, and %s be removed",
1353 empathy_contact_get_alias (contact),
1354 remove_ ? "WILL" : "WILL NOT");
1356 data = g_slice_new0 (ShowActiveData);
1358 data->store = g_object_ref (store);
1359 data->contact = g_object_ref (contact);
1360 data->remove = remove_;
1366 contact_list_store_contact_active_free (ShowActiveData *data)
1368 g_object_unref (data->contact);
1369 g_object_unref (data->store);
1371 g_slice_free (ShowActiveData, data);
1375 contact_list_store_contact_active_cb (ShowActiveData *data)
1377 EmpathyContactListStorePriv *priv;
1379 priv = GET_PRIV (data->store);
1382 !priv->show_offline &&
1383 !empathy_contact_is_online (data->contact)) {
1384 DEBUG ("Contact:'%s' active timeout, removing item",
1385 empathy_contact_get_alias (data->contact));
1386 contact_list_store_remove_contact (data->store, data->contact);
1389 DEBUG ("Contact:'%s' no longer active",
1390 empathy_contact_get_alias (data->contact));
1392 contact_list_store_contact_set_active (data->store,
1397 contact_list_store_contact_active_free (data);
1403 contact_list_store_get_group_foreach (GtkTreeModel *model,
1411 /* Groups are only at the top level. */
1412 if (gtk_tree_path_get_depth (path) != 1) {
1416 gtk_tree_model_get (model, iter,
1417 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &str,
1418 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, &is_group,
1421 if (is_group && !tp_strdiff (str, fg->name)) {
1432 contact_list_store_get_group (EmpathyContactListStore *store,
1434 GtkTreeIter *iter_group_to_set,
1435 GtkTreeIter *iter_separator_to_set,
1437 gboolean is_fake_group)
1439 GtkTreeModel *model;
1440 GtkTreeIter iter_group;
1441 GtkTreeIter iter_separator;
1444 memset (&fg, 0, sizeof (fg));
1448 model = GTK_TREE_MODEL (store);
1449 gtk_tree_model_foreach (model,
1450 (GtkTreeModelForeachFunc) contact_list_store_get_group_foreach,
1458 gtk_tree_store_insert_with_values (GTK_TREE_STORE (store), &iter_group, NULL, 0,
1459 EMPATHY_CONTACT_LIST_STORE_COL_ICON_STATUS, NULL,
1460 EMPATHY_CONTACT_LIST_STORE_COL_NAME, name,
1461 EMPATHY_CONTACT_LIST_STORE_COL_IS_GROUP, TRUE,
1462 EMPATHY_CONTACT_LIST_STORE_COL_IS_ACTIVE, FALSE,
1463 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, FALSE,
1464 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, is_fake_group,
1467 if (iter_group_to_set) {
1468 *iter_group_to_set = iter_group;
1471 gtk_tree_store_insert_with_values (GTK_TREE_STORE (store), &iter_separator, &iter_group, 0,
1472 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, TRUE,
1475 if (iter_separator_to_set) {
1476 *iter_separator_to_set = iter_separator;
1483 if (iter_group_to_set) {
1484 *iter_group_to_set = fg.iter;
1487 iter_separator = fg.iter;
1489 if (gtk_tree_model_iter_next (model, &iter_separator)) {
1490 gboolean is_separator;
1492 gtk_tree_model_get (model, &iter_separator,
1493 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator,
1496 if (is_separator && iter_separator_to_set) {
1497 *iter_separator_to_set = iter_separator;
1504 get_position (const char **strv,
1509 for (i = 0; strv[i] != NULL; i++) {
1510 if (!tp_strdiff (strv[i], str))
1518 compare_separator_and_groups (gboolean is_separator_a,
1519 gboolean is_separator_b,
1520 const gchar *name_a,
1521 const gchar *name_b,
1522 EmpathyContact *contact_a,
1523 EmpathyContact *contact_b,
1524 gboolean fake_group_a,
1525 gboolean fake_group_b)
1527 /* these two lists are the sorted list of fake groups to include at the
1528 * top and bottom of the roster */
1529 const char *top_groups[] = {
1530 EMPATHY_CONTACT_LIST_STORE_FAVORITE,
1534 const char *bottom_groups[] = {
1535 EMPATHY_CONTACT_LIST_STORE_UNGROUPED,
1539 if (is_separator_a || is_separator_b) {
1540 /* We have at least one separator */
1541 if (is_separator_a) {
1543 } else if (is_separator_b) {
1548 /* One group and one contact */
1549 if (!contact_a && contact_b) {
1551 } else if (contact_a && !contact_b) {
1553 } else if (!contact_a && !contact_b) {
1554 gboolean a_in_top, b_in_top, a_in_bottom, b_in_bottom;
1556 a_in_top = fake_group_a &&
1557 tp_strv_contains (top_groups, name_a);
1558 b_in_top = fake_group_b &&
1559 tp_strv_contains (top_groups, name_b);
1560 a_in_bottom = fake_group_a &&
1561 tp_strv_contains (bottom_groups, name_a);
1562 b_in_bottom = fake_group_b &&
1563 tp_strv_contains (bottom_groups, name_b);
1565 if (a_in_top && b_in_top) {
1566 /* compare positions */
1567 return CLAMP (get_position (top_groups, name_a) -
1568 get_position (top_groups, name_b),
1570 } else if (a_in_bottom && b_in_bottom) {
1571 /* compare positions */
1572 return CLAMP (get_position (bottom_groups, name_a) -
1573 get_position (bottom_groups, name_b),
1575 } else if (a_in_top || b_in_bottom) {
1577 } else if (b_in_top || a_in_bottom) {
1580 return g_utf8_collate (name_a, name_b);
1584 /* Two contacts, ordering depends of the sorting policy */
1589 contact_list_store_contact_sort (EmpathyContact *contact_a,
1590 EmpathyContact *contact_b)
1592 TpAccount *account_a, *account_b;
1595 g_return_val_if_fail (contact_a != NULL || contact_b != NULL, 0);
1598 ret_val = g_utf8_collate (empathy_contact_get_alias (contact_a),
1599 empathy_contact_get_alias (contact_b));
1605 ret_val = g_utf8_collate (empathy_contact_get_id (contact_a),
1606 empathy_contact_get_id (contact_b));
1611 account_a = empathy_contact_get_account (contact_a);
1612 account_b = empathy_contact_get_account (contact_b);
1615 ret_val = strcmp (tp_account_get_protocol (account_a),
1616 tp_account_get_protocol (account_b));
1622 ret_val = strcmp (tp_proxy_get_object_path (account_a),
1623 tp_proxy_get_object_path (account_b));
1630 contact_list_store_state_sort_func (GtkTreeModel *model,
1631 GtkTreeIter *iter_a,
1632 GtkTreeIter *iter_b,
1636 gchar *name_a, *name_b;
1637 gboolean is_separator_a, is_separator_b;
1638 EmpathyContact *contact_a, *contact_b;
1639 gboolean fake_group_a, fake_group_b;
1641 gtk_tree_model_get (model, iter_a,
1642 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name_a,
1643 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact_a,
1644 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator_a,
1645 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, &fake_group_a,
1647 gtk_tree_model_get (model, iter_b,
1648 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name_b,
1649 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact_b,
1650 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator_b,
1651 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, &fake_group_b,
1654 if (contact_a == NULL || contact_b == NULL) {
1655 ret_val = compare_separator_and_groups (is_separator_a, is_separator_b,
1656 name_a, name_b, contact_a, contact_b, fake_group_a, fake_group_b);
1660 /* If we managed to get this far, we can start looking at
1663 ret_val = -tp_connection_presence_type_cmp_availability (
1664 empathy_contact_get_presence (EMPATHY_CONTACT (contact_a)),
1665 empathy_contact_get_presence (EMPATHY_CONTACT (contact_b)));
1668 /* Fallback: compare by name et al. */
1669 ret_val = contact_list_store_contact_sort (contact_a, contact_b);
1677 g_object_unref (contact_a);
1681 g_object_unref (contact_b);
1688 contact_list_store_name_sort_func (GtkTreeModel *model,
1689 GtkTreeIter *iter_a,
1690 GtkTreeIter *iter_b,
1693 gchar *name_a, *name_b;
1694 EmpathyContact *contact_a, *contact_b;
1695 gboolean is_separator_a = FALSE, is_separator_b = FALSE;
1697 gboolean fake_group_a, fake_group_b;
1699 gtk_tree_model_get (model, iter_a,
1700 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name_a,
1701 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact_a,
1702 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator_a,
1703 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, &fake_group_a,
1705 gtk_tree_model_get (model, iter_b,
1706 EMPATHY_CONTACT_LIST_STORE_COL_NAME, &name_b,
1707 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact_b,
1708 EMPATHY_CONTACT_LIST_STORE_COL_IS_SEPARATOR, &is_separator_b,
1709 EMPATHY_CONTACT_LIST_STORE_COL_IS_FAKE_GROUP, &fake_group_b,
1712 if (contact_a == NULL || contact_b == NULL)
1713 ret_val = compare_separator_and_groups (is_separator_a, is_separator_b,
1714 name_a, name_b, contact_a, contact_b, fake_group_a, fake_group_b);
1716 ret_val = contact_list_store_contact_sort (contact_a, contact_b);
1719 g_object_unref (contact_a);
1723 g_object_unref (contact_b);
1730 contact_list_store_find_contact_foreach (GtkTreeModel *model,
1735 EmpathyContact *contact;
1737 gtk_tree_model_get (model, iter,
1738 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
1741 if (contact == fc->contact) {
1743 fc->iters = g_list_append (fc->iters, gtk_tree_iter_copy (iter));
1747 g_object_unref (contact);
1754 contact_list_store_find_contact (EmpathyContactListStore *store,
1755 EmpathyContact *contact)
1757 GtkTreeModel *model;
1761 memset (&fc, 0, sizeof (fc));
1763 fc.contact = contact;
1765 model = GTK_TREE_MODEL (store);
1766 gtk_tree_model_foreach (model,
1767 (GtkTreeModelForeachFunc) contact_list_store_find_contact_foreach,
1778 contact_list_store_update_list_mode_foreach (GtkTreeModel *model,
1781 EmpathyContactListStore *store)
1783 EmpathyContactListStorePriv *priv;
1784 gboolean show_avatar = FALSE;
1785 EmpathyContact *contact;
1786 GdkPixbuf *pixbuf_status;
1788 priv = GET_PRIV (store);
1790 if (priv->show_avatars && !priv->is_compact) {
1794 gtk_tree_model_get (model, iter,
1795 EMPATHY_CONTACT_LIST_STORE_COL_CONTACT, &contact,
1798 if (contact == NULL){
1801 /* get icon from hash_table */
1802 pixbuf_status = contact_list_store_get_contact_status_icon (store, contact);
1804 gtk_tree_store_set (GTK_TREE_STORE (store), iter,
1805 EMPATHY_CONTACT_LIST_STORE_COL_ICON_STATUS, pixbuf_status,
1806 EMPATHY_CONTACT_LIST_STORE_COL_PIXBUF_AVATAR_VISIBLE, show_avatar,
1807 EMPATHY_CONTACT_LIST_STORE_COL_COMPACT, priv->is_compact,
1814 contact_list_store_get_contact_status_icon_with_icon_name (
1815 EmpathyContactListStore *store,
1816 EmpathyContact *contact,
1817 const gchar *status_icon_name)
1819 GdkPixbuf *pixbuf_status = NULL;
1820 EmpathyContactListStorePriv *priv;
1821 const gchar *protocol_name = NULL;
1822 gchar *icon_name = NULL;
1824 priv = GET_PRIV (store);
1826 if (priv->show_protocols) {
1827 protocol_name = empathy_protocol_name_for_contact (contact);
1828 icon_name = g_strdup_printf ("%s-%s", status_icon_name, protocol_name);
1830 icon_name = g_strdup_printf ("%s", status_icon_name);
1832 pixbuf_status = g_hash_table_lookup (priv->status_icons, icon_name);
1833 if (pixbuf_status == NULL) {
1834 pixbuf_status = empathy_pixbuf_contact_status_icon_with_icon_name (contact,
1836 priv->show_protocols);
1837 if (pixbuf_status != NULL) {
1838 g_hash_table_insert (priv->status_icons,
1839 g_strdup (icon_name),
1845 return pixbuf_status;
1849 contact_list_store_get_contact_status_icon (EmpathyContactListStore *store,
1850 EmpathyContact *contact)
1852 GdkPixbuf *pixbuf_status = NULL;
1853 EmpathyContactListStorePriv *priv;
1854 const gchar *status_icon_name = NULL;
1855 gboolean composing = FALSE;
1857 priv = GET_PRIV (store);
1859 if (EMPATHY_IS_TP_CHAT (priv->list)) {
1860 if (empathy_tp_chat_get_chat_state (EMPATHY_TP_CHAT (priv->list),
1862 TP_CHANNEL_CHAT_STATE_COMPOSING)
1867 status_icon_name = EMPATHY_IMAGE_TYPING;
1869 status_icon_name = empathy_icon_name_for_contact (contact);
1872 if (status_icon_name == NULL)
1875 pixbuf_status = contact_list_store_get_contact_status_icon_with_icon_name (
1880 return pixbuf_status;