+ if (current_folder_needs_filtering (priv))
+ modest_header_view_refilter (self);
+}
+
+static void
+on_notify_style (GObject *obj, GParamSpec *spec, gpointer userdata)
+{
+ if (strcmp ("style", spec->name) == 0) {
+ update_style (MODEST_HEADER_VIEW (obj));
+ gtk_widget_queue_draw (GTK_WIDGET (obj));
+ }
+}
+
+static void
+update_style (ModestHeaderView *self)
+{
+ ModestHeaderViewPrivate *priv;
+ GdkColor style_color;
+ GdkColor style_active_color;
+ PangoAttrList *attr_list;
+ GtkStyle *style;
+ PangoAttribute *attr;
+
+ g_return_if_fail (MODEST_IS_HEADER_VIEW (self));
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);
+
+ /* Set color */
+
+ attr_list = pango_attr_list_new ();
+ if (!gtk_style_lookup_color (gtk_widget_get_style (GTK_WIDGET (self)), "SecondaryTextColor", &style_color)) {
+ gdk_color_parse (MODEST_SECONDARY_COLOR, &style_color);
+ }
+ priv->secondary_color = style_color;
+ attr = pango_attr_foreground_new (style_color.red, style_color.green, style_color.blue);
+ pango_attr_list_insert (attr_list, attr);
+
+ /* set font */
+ style = gtk_rc_get_style_by_paths (gtk_widget_get_settings
+ (GTK_WIDGET(self)),
+ "SmallSystemFont", NULL,
+ G_TYPE_NONE);
+ if (style) {
+ attr = pango_attr_font_desc_new (pango_font_description_copy
+ (style->font_desc));
+ pango_attr_list_insert (attr_list, attr);
+
+ g_object_set (G_OBJECT (priv->renderer_address),
+ "foreground-gdk", &(priv->secondary_color),
+ "foreground-set", TRUE,
+ "attributes", attr_list,
+ NULL);
+ g_object_set (G_OBJECT (priv->renderer_date_status),
+ "foreground-gdk", &(priv->secondary_color),
+ "foreground-set", TRUE,
+ "attributes", attr_list,
+ NULL);
+ pango_attr_list_unref (attr_list);
+ } else {
+ g_object_set (G_OBJECT (priv->renderer_address),
+ "foreground-gdk", &(priv->secondary_color),
+ "foreground-set", TRUE,
+ "scale", PANGO_SCALE_SMALL,
+ "scale-set", TRUE,
+ NULL);
+ g_object_set (G_OBJECT (priv->renderer_date_status),
+ "foreground-gdk", &(priv->secondary_color),
+ "foreground-set", TRUE,
+ "scale", PANGO_SCALE_SMALL,
+ "scale-set", TRUE,
+ NULL);
+ }
+
+ if (gtk_style_lookup_color (gtk_widget_get_style (GTK_WIDGET (self)), "ActiveTextColor", &style_active_color)) {
+ priv->active_color = style_active_color;
+#ifdef MODEST_TOOLKIT_HILDON2
+ g_object_set_data (G_OBJECT (priv->renderer_subject), BOLD_IS_ACTIVE_COLOR, GINT_TO_POINTER (TRUE));
+ g_object_set_data (G_OBJECT (priv->renderer_subject), ACTIVE_COLOR, &(priv->active_color));
+#endif
+ } else {
+#ifdef MODEST_TOOLKIT_HILDON2
+ g_object_set_data (G_OBJECT (priv->renderer_subject), BOLD_IS_ACTIVE_COLOR, GINT_TO_POINTER (FALSE));
+#endif
+ }
+}
+
+TnyHeader *
+modest_header_view_get_header_at_pos (ModestHeaderView *header_view,
+ gint initial_x,
+ gint initial_y)
+{
+ GtkTreePath *path;
+ GtkTreeModel *tree_model;
+ GtkTreeIter iter;
+ TnyHeader *header;
+
+ /* Get tree path */
+ if (!gtk_tree_view_get_dest_row_at_pos ((GtkTreeView *) header_view,
+ initial_x,
+ initial_y,
+ &path,
+ NULL))
+ return NULL;
+
+ /* Get model */
+ tree_model = gtk_tree_view_get_model ((GtkTreeView *) header_view);
+ if (!gtk_tree_model_get_iter (tree_model, &iter, path))
+ return NULL;
+
+ /* Get header */
+ gtk_tree_model_get (tree_model, &iter,
+ TNY_GTK_HEADER_LIST_MODEL_INSTANCE_COLUMN,
+ &header, -1);
+
+ return header;
+}
+
+static gboolean
+parse_date_side (const gchar *string, time_t *date_side)
+{
+ gchar *today;
+ gchar *yesterday;
+ gchar *casefold;
+ GDate *date;
+ gboolean result = FALSE;
+
+ if (string && string[0] == '\0') {
+ *date_side = 0;
+ return TRUE;
+ }
+
+ casefold = g_utf8_casefold (string, -1);
+ today = g_utf8_casefold (dgettext ("gtk20", "Today"), -1);
+ yesterday = g_utf8_casefold (dgettext ("gtk20", "Yesterday"), -1);
+ date = g_date_new ();
+
+ if (g_utf8_collate (casefold, today) == 0) {
+ *date_side = time (NULL);
+ result = TRUE;
+ goto frees;
+ }
+
+ if (g_utf8_collate (casefold, yesterday) == 0) {
+ *date_side = time (NULL) - 24*60*60;
+ result = TRUE;
+ goto frees;
+ }
+
+ g_date_set_parse (date, string);
+ if (g_date_valid (date)) {
+ struct tm tm = {0};
+ g_date_to_struct_tm (date, &tm);
+ *date_side = mktime (&tm);
+
+ result = TRUE;
+ goto frees;
+ }
+frees:
+ g_free (today);
+ g_free (yesterday);
+ g_free (casefold);
+ g_date_free (date);
+
+ return result;
+}
+
+static gboolean
+parse_date_range (const gchar *string, time_t *date_range_start, time_t *date_range_end)
+{
+ gchar ** parts;
+ gboolean valid;
+
+ parts = g_strsplit (string, "..", 2);
+ valid = TRUE;
+
+ if (g_strv_length (parts) != 2) {
+ valid = FALSE;
+ goto frees;
+ g_strfreev (parts);
+ return FALSE;
+ }
+
+ if (!parse_date_side (parts[0], date_range_start)) {
+ valid = FALSE;
+ goto frees;
+ }
+
+ if (parse_date_side (parts[1], date_range_end)) {
+ if (*date_range_end == 0) {
+ *date_range_end = (time_t) -1;
+ } else {
+ *date_range_end += (24*60*60 - 1);
+ }
+ } else {
+ valid = FALSE;
+ goto frees;
+ }
+
+frees:
+ g_strfreev (parts);
+ return valid;
+}
+
+void
+modest_header_view_set_filter_string (ModestHeaderView *self,
+ const gchar *filter_string)
+{
+ ModestHeaderViewPrivate *priv;
+
+ g_return_if_fail (MODEST_IS_HEADER_VIEW (self));
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);
+
+ if (priv->filter_string)
+ g_free (priv->filter_string);
+
+ priv->filter_string = g_strdup (filter_string);
+ priv->filter_date_range = FALSE;
+
+ if (priv->filter_string_splitted) {
+ g_strfreev (priv->filter_string_splitted);
+ priv->filter_string_splitted = NULL;
+ }
+
+ if (priv->filter_string) {
+ gchar **split, **current, **current_target;
+
+ split = g_strsplit (priv->filter_string, " ", 0);
+
+ priv->filter_string_splitted = g_malloc0 (sizeof (gchar *)*(g_strv_length (split) + 1));
+ current_target = priv->filter_string_splitted;
+ for (current = split; *current != 0; current ++) {
+ gboolean has_date_range = FALSE;;
+ if (g_strstr_len (*current, -1, "..") && strcmp(*current, "..")) {
+ time_t range_start, range_end;
+ /* It contains .. but it's not ".." so it may be a date range */
+ if (parse_date_range (*current, &range_start, &range_end)) {
+ priv->filter_date_range = TRUE;
+ has_date_range = TRUE;
+ priv->date_range_start = range_start;
+ priv->date_range_end = range_end;
+ }
+ }
+ if (!has_date_range) {
+ *current_target = g_utf8_casefold (*current, -1);
+ current_target++;
+ }
+ }
+ *current_target = '\0';
+ g_strfreev (split);
+ }
+ modest_header_view_refilter (MODEST_HEADER_VIEW (self));
+}
+
+#ifdef MODEST_TOOLKIT_HILDON2
+static gboolean
+on_live_search_timeout (ModestHeaderView *self)
+{
+ const gchar *needle;
+ ModestHeaderViewPrivate *priv;
+
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);
+
+ needle = hildon_live_search_get_text ((HildonLiveSearch *) priv->live_search);
+ if (needle && needle[0] != '\0') {
+ modest_header_view_set_filter_string (MODEST_HEADER_VIEW (self), needle);
+ } else {
+ modest_header_view_set_filter_string (MODEST_HEADER_VIEW (self), NULL);
+ }
+
+ priv->live_search_timeout = 0;
+
+ return FALSE;
+}
+
+static gboolean
+on_live_search_refilter (HildonLiveSearch *livesearch,
+ ModestHeaderView *self)
+{
+ ModestHeaderViewPrivate *priv;
+ GtkTreeModel *model, *sortable, *filter;
+
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);
+
+ if (priv->live_search_timeout > 0) {
+ g_source_remove (priv->live_search_timeout);
+ priv->live_search_timeout = 0;
+ }
+
+ model = NULL;
+ filter = gtk_tree_view_get_model (GTK_TREE_VIEW (self));
+ if (GTK_IS_TREE_MODEL_FILTER (filter)) {
+ sortable = gtk_tree_model_filter_get_model (GTK_TREE_MODEL_FILTER (filter));
+ if (GTK_IS_TREE_MODEL_SORT (sortable)) {
+ model = gtk_tree_model_sort_get_model (GTK_TREE_MODEL_SORT (sortable));
+ }
+ }
+
+ if (model && tny_list_get_length (TNY_LIST (model)) > 250) {
+ priv->live_search_timeout = g_timeout_add (1000, (GSourceFunc) on_live_search_timeout, self);
+ } else {
+ on_live_search_timeout (self);
+ }
+
+ return TRUE;
+}
+
+GtkWidget *
+modest_header_view_setup_live_search (ModestHeaderView *self)
+{
+ ModestHeaderViewPrivate *priv;
+
+ g_return_val_if_fail (MODEST_IS_HEADER_VIEW (self), NULL);
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);
+ priv->live_search = hildon_live_search_new ();
+
+ g_signal_connect (G_OBJECT (priv->live_search), "refilter", G_CALLBACK (on_live_search_refilter), self);
+
+ return priv->live_search;
+}
+#endif
+
+static gboolean
+refilter_idle_handler (gpointer userdata)
+{
+ ModestHeaderView *self = MODEST_HEADER_VIEW (userdata);
+ ModestHeaderViewPrivate *priv;
+ GtkTreeModel *filter_model;
+ GtkTreeModel *filtered_model;
+ gint i;
+ gboolean has_more;
+
+ priv = MODEST_HEADER_VIEW_GET_PRIVATE (self);