2 /* modest-tny-header-tree-view.c
4 #include <glib/gi18n.h>
5 #include "modest-tny-header-tree-view.h"
6 #include <tny-list-iface.h>
9 #include <modest-icon-names.h>
10 #include "modest-icon-factory.h"
12 /* 'private'/'protected' functions */
13 static void modest_tny_header_tree_view_class_init (ModestTnyHeaderTreeViewClass *klass);
14 static void modest_tny_header_tree_view_init (ModestTnyHeaderTreeView *obj);
15 static void modest_tny_header_tree_view_finalize (GObject *obj);
17 static void selection_changed (GtkTreeSelection *sel, gpointer user_data);
18 static void column_clicked (GtkTreeViewColumn *treeviewcolumn, gpointer user_data);
21 MESSAGE_SELECTED_SIGNAL,
25 typedef struct _ModestTnyHeaderTreeViewPrivate ModestTnyHeaderTreeViewPrivate;
26 struct _ModestTnyHeaderTreeViewPrivate {
28 TnyMsgFolderIface *tny_msg_folder;
29 TnyListIface *headers;
31 guint sort_columns[MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM];
34 ModestTnyHeaderTreeViewStyle style;
36 #define MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE((o), \
37 MODEST_TYPE_TNY_HEADER_TREE_VIEW, \
38 ModestTnyHeaderTreeViewPrivate))
40 static GObjectClass *parent_class = NULL;
42 /* uncomment the following if you have defined any signals */
43 static guint signals[LAST_SIGNAL] = {0};
46 modest_tny_header_tree_view_get_type (void)
48 static GType my_type = 0;
50 static const GTypeInfo my_info = {
51 sizeof(ModestTnyHeaderTreeViewClass),
53 NULL, /* base finalize */
54 (GClassInitFunc) modest_tny_header_tree_view_class_init,
55 NULL, /* class finalize */
56 NULL, /* class data */
57 sizeof(ModestTnyHeaderTreeView),
59 (GInstanceInitFunc) modest_tny_header_tree_view_init,
61 my_type = g_type_register_static (GTK_TYPE_TREE_VIEW,
62 "ModestTnyHeaderTreeView",
69 modest_tny_header_tree_view_class_init (ModestTnyHeaderTreeViewClass *klass)
71 GObjectClass *gobject_class;
72 gobject_class = (GObjectClass*) klass;
74 parent_class = g_type_class_peek_parent (klass);
75 gobject_class->finalize = modest_tny_header_tree_view_finalize;
77 g_type_class_add_private (gobject_class, sizeof(ModestTnyHeaderTreeViewPrivate));
79 signals[MESSAGE_SELECTED_SIGNAL] =
80 g_signal_new ("message_selected",
81 G_TYPE_FROM_CLASS (gobject_class),
83 G_STRUCT_OFFSET (ModestTnyHeaderTreeViewClass,message_selected),
85 g_cclosure_marshal_VOID__POINTER,
86 G_TYPE_NONE, 1, G_TYPE_POINTER);
94 msgtype_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
95 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
97 TnyMsgHeaderFlags flags;
100 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
103 if (flags & TNY_MSG_HEADER_FLAG_SEEN)
104 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_READ);
106 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_UNREAD);
108 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf, NULL);
112 attach_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
113 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
115 TnyMsgHeaderFlags flags;
116 GdkPixbuf *pixbuf = NULL;
118 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
121 if (flags & TNY_MSG_HEADER_FLAG_ATTACHMENTS) {
122 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_ATTACH);
124 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf, NULL);
131 header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
132 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
135 TnyMsgHeaderFlags flags;
137 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
139 rendobj = G_OBJECT(renderer);
141 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
142 g_object_set (rendobj, "weight", 800, NULL);
144 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
150 sender_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
151 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
154 TnyMsgHeaderFlags flags;
158 gtk_tree_model_get (tree_model, iter,
159 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
160 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
161 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &flags
163 rendobj = G_OBJECT(renderer);
165 /* simplistic --> remove <email@address> from display */
166 address = g_strstr_len (from, strlen(from), "<");
169 g_object_set (rendobj, "text", from, NULL);
173 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
174 g_object_set (rendobj, "weight", 800, NULL);
176 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
180 /* not reentrant/thread-safe */
182 display_date (time_t date)
184 struct tm date_tm, now_tm;
187 const gint buf_size = 64;
188 static gchar date_buf[64]; /* buf_size is not ... */
189 static gchar now_buf[64]; /* ...const enough... */
193 localtime_r(&now, &now_tm);
194 localtime_r(&date, &date_tm);
196 /* get today's date */
197 strftime (date_buf, buf_size, "%x", &date_tm);
198 strftime (now_buf, buf_size, "%x", &now_tm); /* today */
200 /* if this is today, get the time instead of the date */
201 if (strcmp (date_buf, now_buf) == 0)
202 strftime (date_buf, buf_size, _("%X"), &date_tm);
209 compact_header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
210 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
213 TnyMsgHeaderFlags flags;
214 gchar *from, *subject;
219 gtk_tree_model_get (tree_model, iter,
220 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
221 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
222 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &subject,
223 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_TIME_T_COLUMN, &date,
225 rendobj = G_OBJECT(renderer);
227 /* simplistic --> remove <email@address> from display */
228 address = g_strstr_len (from, strlen(from), "<");
230 address[0]='\0'; /* set a new endpoint */
232 header = g_strdup_printf ("%s %s\n%s", from,
233 display_date(date), subject);
234 g_object_set (rendobj, "text", header, NULL);
240 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
241 g_object_set (rendobj, "weight", 800, NULL);
243 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
247 static GtkTreeViewColumn*
248 get_new_column (const gchar *name, GtkCellRenderer *renderer,
249 gboolean resizable, gint sort_col_id, gboolean show_as_text,
250 GtkTreeCellDataFunc cell_data_func, gpointer user_data)
252 GtkTreeViewColumn *column;
254 column = gtk_tree_view_column_new_with_attributes(name, renderer, NULL);
255 gtk_tree_view_column_set_resizable (column, resizable);
258 gtk_tree_view_column_add_attribute (column, renderer, "text",
260 if (sort_col_id >= 0)
261 gtk_tree_view_column_set_sort_column_id (column, sort_col_id);
263 gtk_tree_view_column_set_sort_indicator (column, FALSE);
264 gtk_tree_view_column_set_reorderable (column, TRUE);
267 gtk_tree_view_column_set_cell_data_func(column, renderer, cell_data_func,
270 /* g_signal_connect (G_OBJECT (column), "clicked", */
271 /* G_CALLBACK (column_clicked), obj); */
280 remove_all_columns (ModestTnyHeaderTreeView *obj)
282 GList *columns, *cursor;
284 columns = gtk_tree_view_get_columns (GTK_TREE_VIEW(obj));
286 for (cursor = columns; cursor; cursor = cursor->next)
287 gtk_tree_view_remove_column (GTK_TREE_VIEW(obj),
288 GTK_TREE_VIEW_COLUMN(cursor->data));
289 g_list_free (columns);
296 init_columns (ModestTnyHeaderTreeView *obj)
298 GtkTreeViewColumn *column;
299 GtkCellRenderer *renderer_msgtype,
303 ModestTnyHeaderTreeViewPrivate *priv;
306 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
308 renderer_msgtype = gtk_cell_renderer_pixbuf_new ();
309 renderer_attach = gtk_cell_renderer_pixbuf_new ();
310 renderer_header = gtk_cell_renderer_text_new ();
312 remove_all_columns (obj);
314 for (cursor = priv->columns; cursor; cursor = cursor->next) {
315 ModestTnyHeaderTreeViewColumn col =
316 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
320 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
322 column = get_new_column (_("M"), renderer_msgtype, FALSE,
323 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
324 FALSE, (GtkTreeCellDataFunc)msgtype_cell_data,
328 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
330 column = get_new_column (_("A"), renderer_attach, FALSE,
331 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
332 FALSE, (GtkTreeCellDataFunc)attach_cell_data,
336 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
337 column = get_new_column (_("Received"), renderer_header, TRUE,
338 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
339 TRUE, (GtkTreeCellDataFunc)header_cell_data,
343 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_FROM:
344 column = get_new_column (_("From"), renderer_header, TRUE,
345 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
346 TRUE, (GtkTreeCellDataFunc)sender_cell_data,
350 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
351 column = get_new_column (_("Header"), renderer_header, TRUE,
352 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
353 TRUE, (GtkTreeCellDataFunc)compact_header_cell_data,
357 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT:
358 column = get_new_column (_("Subject"), renderer_header, TRUE,
359 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
360 TRUE, (GtkTreeCellDataFunc)header_cell_data,
365 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
366 column = get_new_column (_("Sent"), renderer_header, TRUE,
367 TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
368 TRUE, (GtkTreeCellDataFunc)header_cell_data,
372 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_TO:
373 column = get_new_column (_("To"), renderer_header, TRUE,
374 TNY_MSG_HEADER_LIST_MODEL_TO_COLUMN,
375 TRUE, (GtkTreeCellDataFunc)header_cell_data,
379 gtk_tree_view_append_column (GTK_TREE_VIEW(obj), column);
388 modest_tny_header_tree_view_init (ModestTnyHeaderTreeView *obj)
390 ModestTnyHeaderTreeViewPrivate *priv;
393 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
395 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i)
396 priv->sort_columns[i] = -1;
401 modest_tny_header_tree_view_finalize (GObject *obj)
403 ModestTnyHeaderTreeView *self;
404 ModestTnyHeaderTreeViewPrivate *priv;
407 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
408 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
411 g_object_unref (G_OBJECT(priv->headers));
413 priv->headers = NULL;
414 priv->tny_msg_folder = NULL;
418 modest_tny_header_tree_view_new (TnyMsgFolderIface *folder,
420 ModestTnyHeaderTreeViewStyle style)
423 GtkTreeSelection *sel;
424 ModestTnyHeaderTreeView *self;
426 obj = G_OBJECT(g_object_new(MODEST_TYPE_TNY_HEADER_TREE_VIEW, NULL));
427 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
429 if (!modest_tny_header_tree_view_set_folder (self, NULL)) {
430 g_warning ("could not set the folder");
431 g_object_unref (obj);
435 modest_tny_header_tree_view_set_style (self, style);
436 modest_tny_header_tree_view_set_columns (self, columns);
439 gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(obj), TRUE);
440 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(obj), TRUE);
442 gtk_tree_view_set_rules_hint (GTK_TREE_VIEW(obj),
443 TRUE); /* alternating row colors */
445 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(self));
446 g_signal_connect (sel, "changed",
447 G_CALLBACK(selection_changed), self);
449 return GTK_WIDGET(self);
453 modest_tny_header_tree_view_set_columns (ModestTnyHeaderTreeView *self, GSList *columns)
455 ModestTnyHeaderTreeViewPrivate *priv;
458 g_return_val_if_fail (self, FALSE);
460 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
462 g_slist_free (priv->columns);
464 for (cursor = columns; cursor; cursor = cursor->next) {
465 ModestTnyHeaderTreeViewColumn col =
466 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
467 if (0 > col || col >= MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM)
468 g_warning ("invalid column in column list");
470 priv->columns = g_slist_append (priv->columns, cursor->data);
473 init_columns (self); /* redraw them */
480 modest_tny_header_tree_view_get_columns (ModestTnyHeaderTreeView *self)
482 ModestTnyHeaderTreeViewPrivate *priv;
485 g_return_val_if_fail (self, FALSE);
487 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
488 return priv->columns;
495 modest_tny_header_tree_view_set_style (ModestTnyHeaderTreeView *self,
496 ModestTnyHeaderTreeViewStyle style)
498 g_return_val_if_fail (self, FALSE);
499 g_return_val_if_fail (style >= 0 && style < MODEST_TNY_HEADER_TREE_VIEW_STYLE_NUM,
502 MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style = style;
507 ModestTnyHeaderTreeViewStyle
508 modest_tny_header_tree_view_get_style (ModestTnyHeaderTreeView *self)
510 g_return_val_if_fail (self, FALSE);
512 return MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style;
517 /* get the length of any prefix that should be ignored for sorting */
519 get_prefix_len (const gchar *sub)
522 const static gchar* prefix[] = {"Re:", "RE:", "Fwd:", "FWD:", NULL};
524 if (sub[0] != 'R' && sub[0] != 'F') /* optimization */
528 if (g_str_has_prefix(sub, prefix[i])) {
529 int prefix_len = strlen(prefix[i]);
530 if (sub[prefix_len + 1] == ' ')
531 ++prefix_len; /* ignore space after prefix as well */
540 cmp_normalized_subject (const gchar* s1, const gchar *s2)
543 return strcmp (s1 + get_prefix_len(s1),
544 s2 + get_prefix_len(s2));
549 cmp_rows (GtkTreeModel *tree_model, GtkTreeIter *iter1, GtkTreeIter *iter2,
552 gint col_id = GPOINTER_TO_INT (user_data);
555 g_return_val_if_fail (GTK_IS_TREE_MODEL(tree_model), -1);
559 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
560 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
561 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
563 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
568 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
569 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
571 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
575 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT: {
579 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
581 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
584 retval = cmp_normalized_subject(sub1, sub2);
592 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
593 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
595 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
598 return (val1 & TNY_MSG_HEADER_FLAG_ATTACHMENTS) - (val2 & TNY_MSG_HEADER_FLAG_ATTACHMENTS);
601 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
602 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
604 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
607 g_message ("%p %p", iter1, iter2);
608 return (val1 & TNY_MSG_HEADER_FLAG_SEEN) - (val2 & TNY_MSG_HEADER_FLAG_SEEN);
611 g_message ("%p %p", iter1, iter2);
612 return &iter1 - &iter2;
618 modest_tny_header_tree_view_set_folder (ModestTnyHeaderTreeView *self,
619 TnyMsgFolderIface *folder)
622 GtkTreeModel *oldsortable, *sortable, *oldmodel;
623 ModestTnyHeaderTreeViewPrivate *priv;
625 g_return_val_if_fail (self, FALSE);
627 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
630 priv->headers = TNY_LIST_IFACE(tny_msg_header_list_model_new ());
631 tny_msg_folder_iface_get_headers (folder, priv->headers,
633 tny_msg_header_list_model_set_folder (TNY_MSG_HEADER_LIST_MODEL(priv->headers),
636 oldsortable = gtk_tree_view_get_model(GTK_TREE_VIEW (self));
637 if (oldsortable && GTK_IS_TREE_MODEL_SORT(oldsortable)) {
638 GtkTreeModel *oldmodel = gtk_tree_model_sort_get_model
639 (GTK_TREE_MODEL_SORT(oldsortable));
641 g_object_unref (G_OBJECT(oldmodel));
642 g_object_unref (oldsortable);
645 sortable = gtk_tree_model_sort_new_with_model (GTK_TREE_MODEL(priv->headers));
647 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i) {
648 int col_id = priv->sort_columns[i];
650 g_message ("%d: %p: %p: %d", i, GTK_TREE_SORTABLE(sortable), sortable, col_id);
651 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE(sortable), col_id,
652 (GtkTreeIterCompareFunc)cmp_rows,
653 GINT_TO_POINTER(col_id), NULL);
656 gtk_tree_view_set_model (GTK_TREE_VIEW (self), sortable);
657 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), TRUE);
659 /* no need to unref sortable */
661 } else /* when there is no folder */
662 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), FALSE);
669 selection_changed (GtkTreeSelection *sel, gpointer user_data)
672 TnyMsgHeaderIface *header;
674 ModestTnyHeaderTreeView *tree_view;
676 g_return_if_fail (sel);
677 g_return_if_fail (user_data);
679 if (!gtk_tree_selection_get_selected (sel, &model, &iter))
680 return; /* msg was _un_selected */
682 tree_view = MODEST_TNY_HEADER_TREE_VIEW (user_data);
684 gtk_tree_model_get (model, &iter,
685 TNY_MSG_HEADER_LIST_MODEL_INSTANCE_COLUMN,
689 TnyMsgHeaderFlags flags;
690 const TnyMsgIface *msg;
691 const TnyMsgFolderIface *folder;
693 folder = tny_msg_header_iface_get_folder (TNY_MSG_HEADER_IFACE(header));
695 g_message ("cannot find folder");
697 msg = tny_msg_folder_iface_get_message (TNY_MSG_FOLDER_IFACE(folder),
700 g_message ("cannot find msg");
701 /* FIXME: update display */
705 g_signal_emit (G_OBJECT(tree_view), signals[MESSAGE_SELECTED_SIGNAL], 0,
708 /* mark message as seen; _set_flags crashes, bug in tinymail? */
709 flags = tny_msg_header_iface_get_flags (TNY_MSG_HEADER_IFACE(header));
710 //tny_msg_header_iface_set_flags (header, flags | TNY_MSG_HEADER_FLAG_SEEN);
715 column_clicked (GtkTreeViewColumn *col, gpointer user_data)
717 GtkTreeView *treeview;
720 treeview = GTK_TREE_VIEW (user_data);
721 id = gtk_tree_view_column_get_sort_column_id (col);
723 gtk_tree_view_set_search_column (treeview, id);