2 * empathy-roster-model-manager.c
4 * Implementation of EmpathyRosterModel using EmpathyIndividualManager as
7 * Copyright (C) 2012 Collabora Ltd. <http://www.collabora.co.uk/>
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Lesser General Public
11 * License as published by the Free Software Foundation; either
12 * version 2.1 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Lesser General Public License for more details.
19 * You should have received a copy of the GNU Lesser General Public
20 * License along with this library; if not, write to the Free Software
21 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
25 #include "empathy-roster-model-manager.h"
27 #include <glib/gi18n-lib.h>
29 #include "empathy-roster-model.h"
30 #include "empathy-utils.h"
32 static void roster_model_iface_init (EmpathyRosterModelInterface
*iface
);
34 G_DEFINE_TYPE_WITH_CODE (EmpathyRosterModelManager
,
35 empathy_roster_model_manager
,
37 G_IMPLEMENT_INTERFACE (EMPATHY_TYPE_ROSTER_MODEL
, roster_model_iface_init
))
51 static guint signals[LAST_SIGNAL];
54 struct _EmpathyRosterModelManagerPriv
56 EmpathyIndividualManager
*manager
;
57 /* FolksIndividual (borrowed) */
58 GList
*top_group_members
;
62 is_xmpp_local_contact (FolksIndividual
*individual
)
64 EmpathyContact
*contact
;
65 TpConnection
*connection
;
66 const gchar
*protocol_name
= NULL
;
69 contact
= empathy_contact_dup_from_folks_individual (individual
);
74 connection
= empathy_contact_get_connection (contact
);
75 protocol_name
= tp_connection_get_protocol_name (connection
);
76 result
= !tp_strdiff (protocol_name
, "local-xmpp");
77 g_object_unref (contact
);
83 individual_in_top_group_members (EmpathyRosterModelManager
*self
,
84 FolksIndividual
*individual
)
86 return (g_list_find (self
->priv
->top_group_members
, individual
) != NULL
);
90 individual_should_be_in_top_group_members (EmpathyRosterModelManager
*self
,
91 FolksIndividual
*individual
)
95 tops
= empathy_individual_manager_get_top_individuals (self
->priv
->manager
);
97 return (folks_favourite_details_get_is_favourite (
98 FOLKS_FAVOURITE_DETAILS (individual
)) ||
99 g_list_find (tops
, individual
) != NULL
);
103 add_to_top_group_members (EmpathyRosterModelManager
*self
,
104 FolksIndividual
*individual
)
106 self
->priv
->top_group_members
= g_list_prepend (self
->priv
->top_group_members
,
111 remove_from_top_group_members (EmpathyRosterModelManager
*self
,
112 FolksIndividual
*individual
)
114 self
->priv
->top_group_members
= g_list_remove (self
->priv
->top_group_members
,
119 populate_model (EmpathyRosterModelManager
*self
)
121 GList
*individuals
, *l
;
123 individuals
= empathy_individual_manager_get_members (self
->priv
->manager
);
125 for (l
= individuals
; l
!= NULL
; l
= g_list_next (l
))
127 if (individual_should_be_in_top_group_members (self
, l
->data
))
128 add_to_top_group_members (self
, l
->data
);
130 empathy_roster_model_fire_individual_added (EMPATHY_ROSTER_MODEL (self
),
136 members_changed_cb (EmpathyIndividualManager
*manager
,
137 const gchar
*message
,
140 TpChannelGroupChangeReason reason
,
141 EmpathyRosterModelManager
*self
)
145 for (l
= added
; l
!= NULL
; l
= g_list_next (l
))
147 if (individual_should_be_in_top_group_members (self
, l
->data
) &&
148 !individual_in_top_group_members (self
, l
->data
))
149 add_to_top_group_members (self
, l
->data
);
151 empathy_roster_model_fire_individual_added (EMPATHY_ROSTER_MODEL (self
),
155 for (l
= removed
; l
!= NULL
; l
= g_list_next (l
))
157 if (individual_in_top_group_members (self
, l
->data
))
158 remove_from_top_group_members (self
, l
->data
);
160 empathy_roster_model_fire_individual_removed (EMPATHY_ROSTER_MODEL (self
),
166 groups_changed_cb (EmpathyIndividualManager
*manager
,
167 FolksIndividual
*individual
,
170 EmpathyRosterModelManager
*self
)
172 empathy_roster_model_fire_groups_changed (EMPATHY_ROSTER_MODEL (self
),
173 individual
, group
, is_member
);
177 top_individuals_changed_cb (EmpathyIndividualManager
*manager
,
179 EmpathyRosterModelManager
*self
)
183 tops
= empathy_individual_manager_get_top_individuals (self
->priv
->manager
);
185 for (l
= tops
; l
!= NULL
; l
= g_list_next (l
))
187 if (!individual_in_top_group_members (self
, l
->data
))
189 add_to_top_group_members (self
, l
->data
);
191 empathy_roster_model_fire_groups_changed (
192 EMPATHY_ROSTER_MODEL (self
), l
->data
,
193 EMPATHY_ROSTER_MODEL_GROUP_TOP_GROUP
, TRUE
);
197 l
= self
->priv
->top_group_members
;
200 FolksIndividual
*individual
= l
->data
;
202 /* remove_from_top_group_members will modify the list so we already take
203 * the next pointer. */
206 if (!individual_should_be_in_top_group_members (self
, individual
))
208 remove_from_top_group_members (self
, individual
);
210 empathy_roster_model_fire_groups_changed (EMPATHY_ROSTER_MODEL (self
),
211 individual
, EMPATHY_ROSTER_MODEL_GROUP_TOP_GROUP
, FALSE
);
217 favourites_changed_cb (EmpathyIndividualManager
*manager
,
218 FolksIndividual
*individual
,
220 EmpathyRosterModelManager
*self
)
222 if (favourite
&& !individual_in_top_group_members (self
, individual
))
224 add_to_top_group_members (self
, individual
);
226 empathy_roster_model_fire_groups_changed (
227 EMPATHY_ROSTER_MODEL (self
), individual
,
228 EMPATHY_ROSTER_MODEL_GROUP_TOP_GROUP
, favourite
);
230 else if (!favourite
&&
231 !individual_should_be_in_top_group_members (self
, individual
))
233 remove_from_top_group_members (self
, individual
);
235 empathy_roster_model_fire_groups_changed (
236 EMPATHY_ROSTER_MODEL (self
), individual
,
237 EMPATHY_ROSTER_MODEL_GROUP_TOP_GROUP
, favourite
);
242 empathy_roster_model_manager_get_property (GObject
*object
,
247 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (object
);
252 g_value_set_object (value
, self
->priv
->manager
);
255 G_OBJECT_WARN_INVALID_PROPERTY_ID (object
, property_id
, pspec
);
261 empathy_roster_model_manager_set_property (GObject
*object
,
266 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (object
);
271 g_assert (self
->priv
->manager
== NULL
); /* construct only */
272 self
->priv
->manager
= g_value_dup_object (value
);
275 G_OBJECT_WARN_INVALID_PROPERTY_ID (object
, property_id
, pspec
);
281 empathy_roster_model_manager_constructed (GObject
*object
)
283 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (object
);
284 void (*chain_up
) (GObject
*) =
285 ((GObjectClass
*) empathy_roster_model_manager_parent_class
)->constructed
;
287 if (chain_up
!= NULL
)
290 g_assert (EMPATHY_IS_INDIVIDUAL_MANAGER (self
->priv
->manager
));
292 populate_model (self
);
294 tp_g_signal_connect_object (self
->priv
->manager
, "members-changed",
295 G_CALLBACK (members_changed_cb
), self
, 0);
296 tp_g_signal_connect_object (self
->priv
->manager
, "groups-changed",
297 G_CALLBACK (groups_changed_cb
), self
, 0);
298 tp_g_signal_connect_object (self
->priv
->manager
, "notify::top-individuals",
299 G_CALLBACK (top_individuals_changed_cb
), self
, 0);
300 tp_g_signal_connect_object (self
->priv
->manager
, "favourites-changed",
301 G_CALLBACK (favourites_changed_cb
), self
, 0);
305 empathy_roster_model_manager_dispose (GObject
*object
)
307 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (object
);
308 void (*chain_up
) (GObject
*) =
309 ((GObjectClass
*) empathy_roster_model_manager_parent_class
)->dispose
;
311 g_clear_object (&self
->priv
->manager
);
313 if (chain_up
!= NULL
)
318 empathy_roster_model_manager_finalize (GObject
*object
)
320 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (object
);
321 void (*chain_up
) (GObject
*) =
322 ((GObjectClass
*) empathy_roster_model_manager_parent_class
)->finalize
;
324 g_list_free (self
->priv
->top_group_members
);
326 if (chain_up
!= NULL
)
331 empathy_roster_model_manager_class_init (
332 EmpathyRosterModelManagerClass
*klass
)
334 GObjectClass
*oclass
= G_OBJECT_CLASS (klass
);
337 oclass
->get_property
= empathy_roster_model_manager_get_property
;
338 oclass
->set_property
= empathy_roster_model_manager_set_property
;
339 oclass
->constructed
= empathy_roster_model_manager_constructed
;
340 oclass
->dispose
= empathy_roster_model_manager_dispose
;
341 oclass
->finalize
= empathy_roster_model_manager_finalize
;
343 spec
= g_param_spec_object ("manager", "Manager",
344 "EmpathyIndividualManager",
345 EMPATHY_TYPE_INDIVIDUAL_MANAGER
,
346 G_PARAM_READWRITE
| G_PARAM_CONSTRUCT_ONLY
| G_PARAM_STATIC_STRINGS
);
347 g_object_class_install_property (oclass
, PROP_MANAGER
, spec
);
349 g_type_class_add_private (klass
, sizeof (EmpathyRosterModelManagerPriv
));
353 empathy_roster_model_manager_init (EmpathyRosterModelManager
*self
)
355 self
->priv
= G_TYPE_INSTANCE_GET_PRIVATE (self
,
356 EMPATHY_TYPE_ROSTER_MODEL_MANAGER
, EmpathyRosterModelManagerPriv
);
358 self
->priv
->top_group_members
= NULL
;
361 EmpathyRosterModelManager
*
362 empathy_roster_model_manager_new (EmpathyIndividualManager
*manager
)
364 g_return_val_if_fail (EMPATHY_IS_INDIVIDUAL_MANAGER (manager
), NULL
);
366 return g_object_new (EMPATHY_TYPE_ROSTER_MODEL_MANAGER
,
372 empathy_roster_model_manager_get_individuals (EmpathyRosterModel
*model
)
374 EmpathyRosterModelManager
*self
= EMPATHY_ROSTER_MODEL_MANAGER (model
);
376 return empathy_individual_manager_get_members (self
->priv
->manager
);
380 empathy_roster_model_manager_dup_groups_for_individual (
381 EmpathyRosterModel
*model
,
382 FolksIndividual
*individual
)
384 GList
*groups_list
= NULL
;
387 if (is_xmpp_local_contact (individual
))
389 groups_list
= g_list_prepend (groups_list
,
390 g_strdup (EMPATHY_ROSTER_MODEL_GROUP_PEOPLE_NEARBY
));
394 if (individual_in_top_group_members (EMPATHY_ROSTER_MODEL_MANAGER (model
),
396 groups_list
= g_list_prepend (groups_list
,
397 g_strdup (EMPATHY_ROSTER_MODEL_GROUP_TOP_GROUP
));
399 groups_set
= folks_group_details_get_groups (
400 FOLKS_GROUP_DETAILS (individual
));
401 if (gee_collection_get_size (GEE_COLLECTION (groups_set
)) > 0)
403 GeeIterator
*iter
= gee_iterable_iterator (GEE_ITERABLE (groups_set
));
405 while (iter
!= NULL
&& gee_iterator_next (iter
))
407 groups_list
= g_list_prepend (groups_list
, gee_iterator_get (iter
));
409 g_clear_object (&iter
);
416 roster_model_iface_init (EmpathyRosterModelInterface
*iface
)
418 iface
->get_individuals
= empathy_roster_model_manager_get_individuals
;
419 iface
->dup_groups_for_individual
=
420 empathy_roster_model_manager_dup_groups_for_individual
;