diff options
Diffstat (limited to 'libempathy-gtk')
-rw-r--r-- | libempathy-gtk/empathy-roster-view.c | 226 | ||||
-rw-r--r-- | libempathy-gtk/empathy-roster-view.h | 8 |
2 files changed, 234 insertions, 0 deletions
diff --git a/libempathy-gtk/empathy-roster-view.c b/libempathy-gtk/empathy-roster-view.c index ce8f19f64..e68315cd1 100644 --- a/libempathy-gtk/empathy-roster-view.c +++ b/libempathy-gtk/empathy-roster-view.c @@ -10,6 +10,9 @@ G_DEFINE_TYPE (EmpathyRosterView, empathy_roster_view, EGG_TYPE_LIST_BOX) +/* Flashing delay for icons (milliseconds). */ +#define FLASH_TIMEOUT 500 + enum { PROP_MANAGER = 1, @@ -23,6 +26,7 @@ enum { SIG_INDIVIDUAL_ACTIVATED, SIG_POPUP_INDIVIDUAL_MENU, + SIG_EVENT_ACTIVATED, LAST_SIGNAL }; @@ -49,6 +53,13 @@ struct _EmpathyRosterViewPriv /* Hash of the EmpathyRosterContact currently displayed */ GHashTable *displayed_contacts; + guint last_event_id; + /* queue of (Event *). The most recent events are in the head of the queue + * so we always display the icon of the oldest one. */ + GQueue *events; + guint flash_id; + gboolean display_flash_event; + gboolean show_offline; gboolean show_groups; gboolean empty; @@ -59,6 +70,39 @@ struct _EmpathyRosterViewPriv gpointer individual_tooltip_data; }; +typedef struct +{ + guint id; + FolksIndividual *individual; + gchar *icon; + gpointer user_data; +} Event; + +static Event * +event_new (guint id, + FolksIndividual *individual, + const gchar *icon, + gpointer user_data) +{ + Event *event = g_slice_new (Event); + + event->id = id; + event->individual = g_object_ref (individual); + event->icon = g_strdup (icon); + event->user_data = user_data; + return event; +} + +static void +event_free (gpointer data) +{ + Event *event = data; + g_object_unref (event->individual); + g_free (event->icon); + + g_slice_free (Event, event); +} + static void empathy_roster_view_get_property (GObject *object, guint property_id, @@ -281,6 +325,115 @@ update_group_widgets_count (EmpathyRosterView *self, } static void +set_event_icon_on_individual (EmpathyRosterView *self, + FolksIndividual *individual, + const gchar *icon) +{ + GHashTable *contacts; + GHashTableIter iter; + gpointer v; + + contacts = g_hash_table_lookup (self->priv->roster_contacts, individual); + if (contacts == NULL) + return; + + g_hash_table_iter_init (&iter, contacts); + while (g_hash_table_iter_next (&iter, NULL, &v)) + { + EmpathyRosterContact *contact =v; + + empathy_roster_contact_set_event_icon (contact, icon); + } +} + +static void +flash_event (Event *event, + EmpathyRosterView *self) +{ + set_event_icon_on_individual (self, event->individual, event->icon); +} + +static void +unflash_event (Event *event, + EmpathyRosterView *self) +{ + set_event_icon_on_individual (self, event->individual, NULL); +} + +static gboolean +flash_cb (gpointer data) +{ + EmpathyRosterView *self = data; + + if (self->priv->display_flash_event) + { + g_queue_foreach (self->priv->events, (GFunc) flash_event, self); + self->priv->display_flash_event = FALSE; + } + else + { + g_queue_foreach (self->priv->events, (GFunc) unflash_event, self); + self->priv->display_flash_event = TRUE; + } + + return TRUE; +} + +static void +start_flashing (EmpathyRosterView *self) +{ + if (self->priv->flash_id != 0) + return; + + self->priv->display_flash_event = TRUE; + + self->priv->flash_id = g_timeout_add (FLASH_TIMEOUT, + flash_cb, self); +} + +static void +stop_flashing (EmpathyRosterView *self) +{ + if (self->priv->flash_id == 0) + return; + + g_source_remove (self->priv->flash_id); + self->priv->flash_id = 0; +} + +static void +remove_event (EmpathyRosterView *self, + Event *event) +{ + unflash_event (event, self); + g_queue_remove (self->priv->events, event); + + if (g_queue_get_length (self->priv->events) == 0) + { + stop_flashing (self); + } +} + +static void +remove_all_individual_event (EmpathyRosterView *self, + FolksIndividual *individual) +{ + GList *l; + + for (l = g_queue_peek_head_link (self->priv->events); l != NULL; + l = g_list_next (l)) + { + Event *event = l->data; + + if (event->individual == individual) + { + remove_event (self, event); + return; + } + } +} + +static void individual_removed (EmpathyRosterView *self, FolksIndividual *individual) { @@ -292,6 +445,8 @@ individual_removed (EmpathyRosterView *self, if (contacts == NULL) return; + remove_all_individual_event (self, individual); + g_hash_table_iter_init (&iter, contacts); while (g_hash_table_iter_next (&iter, &key, &value)) { @@ -826,6 +981,8 @@ empathy_roster_view_dispose (GObject *object) void (*chain_up) (GObject *) = ((GObjectClass *) empathy_roster_view_parent_class)->dispose; + stop_flashing (self); + empathy_roster_view_set_live_search (self, NULL); g_clear_object (&self->priv->manager); @@ -843,6 +1000,7 @@ empathy_roster_view_finalize (GObject *object) g_hash_table_unref (self->priv->roster_contacts); g_hash_table_unref (self->priv->roster_groups); g_hash_table_unref (self->priv->displayed_contacts); + g_queue_free_full (self->priv->events, event_free); if (chain_up != NULL) chain_up (object); @@ -852,8 +1010,10 @@ static void empathy_roster_view_child_activated (EggListBox *box, GtkWidget *child) { + EmpathyRosterView *self = EMPATHY_ROSTER_VIEW (box); EmpathyRosterContact *contact; FolksIndividual *individual; + GList *l; if (!EMPATHY_IS_ROSTER_CONTACT (child)) return; @@ -861,6 +1021,20 @@ empathy_roster_view_child_activated (EggListBox *box, contact = EMPATHY_ROSTER_CONTACT (child); individual = empathy_roster_contact_get_individual (contact); + /* Activate the oldest event associated with this contact, if any */ + for (l = g_queue_peek_tail_link (self->priv->events); l != NULL; + l = g_list_previous (l)) + { + Event *event = l->data; + + if (event->individual == individual) + { + g_signal_emit (box, signals[SIG_EVENT_ACTIVATED], 0, individual, + event->user_data); + return; + } + } + g_signal_emit (box, signals[SIG_INDIVIDUAL_ACTIVATED], 0, individual); } @@ -1038,6 +1212,13 @@ empathy_roster_view_class_init ( G_TYPE_NONE, 3, FOLKS_TYPE_INDIVIDUAL, G_TYPE_UINT, G_TYPE_UINT); + signals[SIG_EVENT_ACTIVATED] = g_signal_new ("event-activated", + G_OBJECT_CLASS_TYPE (klass), + G_SIGNAL_RUN_LAST, + 0, NULL, NULL, NULL, + G_TYPE_NONE, + 2, FOLKS_TYPE_INDIVIDUAL, G_TYPE_POINTER); + g_type_class_add_private (klass, sizeof (EmpathyRosterViewPriv)); } @@ -1053,6 +1234,8 @@ empathy_roster_view_init (EmpathyRosterView *self) g_free, NULL); self->priv->displayed_contacts = g_hash_table_new (NULL, NULL); + self->priv->events = g_queue_new (); + self->priv->empty = TRUE; } @@ -1194,3 +1377,46 @@ empathy_roster_view_is_searching (EmpathyRosterView *self) return (self->priv->search != NULL && gtk_widget_get_visible (GTK_WIDGET (self->priv->search))); } + +/* Don't use EmpathyEvent as I prefer to keep this object not too specific to + * Empathy's internals. */ +guint +empathy_roster_view_add_event (EmpathyRosterView *self, + FolksIndividual *individual, + const gchar *icon, + gpointer user_data) +{ + GHashTable *contacts; + + contacts = g_hash_table_lookup (self->priv->roster_contacts, individual); + if (contacts == NULL) + return 0; + + self->priv->last_event_id++; + + g_queue_push_head (self->priv->events, + event_new (self->priv->last_event_id, individual, icon, user_data)); + + start_flashing (self); + + return self->priv->last_event_id; +} + +void +empathy_roster_view_remove_event (EmpathyRosterView *self, + guint event_id) +{ + GList *l; + + for (l = g_queue_peek_head_link (self->priv->events); l != NULL; + l = g_list_next (l)) + { + Event *event = l->data; + + if (event->id == event_id) + { + remove_event (self, event); + return; + } + } +} diff --git a/libempathy-gtk/empathy-roster-view.h b/libempathy-gtk/empathy-roster-view.h index 7a87bea7a..c3d475b0b 100644 --- a/libempathy-gtk/empathy-roster-view.h +++ b/libempathy-gtk/empathy-roster-view.h @@ -79,6 +79,14 @@ gboolean empathy_roster_view_is_empty (EmpathyRosterView *self); gboolean empathy_roster_view_is_searching (EmpathyRosterView *self); +guint empathy_roster_view_add_event (EmpathyRosterView *self, + FolksIndividual *individual, + const gchar *icon, + gpointer user_data); + +void empathy_roster_view_remove_event (EmpathyRosterView *self, + guint event_id); + G_END_DECLS #endif /* #ifndef __EMPATHY_ROSTER_VIEW_H__*/ |