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_DELETED)
104 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_DELETED);
105 else if (flags & TNY_MSG_HEADER_FLAG_SEEN)
106 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_READ);
108 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_UNREAD);
110 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf, NULL);
114 attach_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
115 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
117 TnyMsgHeaderFlags flags;
118 GdkPixbuf *pixbuf = NULL;
120 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
123 if (flags & TNY_MSG_HEADER_FLAG_ATTACHMENTS) {
124 pixbuf = modest_icon_factory_get_icon (MODEST_HEADER_ICON_ATTACH);
126 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf, NULL);
133 header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
134 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
137 TnyMsgHeaderFlags flags;
139 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
141 rendobj = G_OBJECT(renderer);
143 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
144 g_object_set (rendobj, "weight", 800, NULL);
146 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
152 sender_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
153 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
156 TnyMsgHeaderFlags flags;
160 gtk_tree_model_get (tree_model, iter,
161 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
162 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
163 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &flags
165 rendobj = G_OBJECT(renderer);
167 /* simplistic --> remove <email@address> from display */
168 address = g_strstr_len (from, strlen(from), "<");
171 g_object_set (rendobj, "text", from, NULL);
175 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
176 g_object_set (rendobj, "weight", 800, NULL);
178 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
182 /* not reentrant/thread-safe */
184 display_date (time_t date)
186 struct tm date_tm, now_tm;
189 const gint buf_size = 64;
190 static gchar date_buf[64]; /* buf_size is not ... */
191 static gchar now_buf[64]; /* ...const enough... */
195 localtime_r(&now, &now_tm);
196 localtime_r(&date, &date_tm);
198 /* get today's date */
199 strftime (date_buf, buf_size, "%x", &date_tm);
200 strftime (now_buf, buf_size, "%x", &now_tm); /* today */
202 /* if this is today, get the time instead of the date */
203 if (strcmp (date_buf, now_buf) == 0)
204 strftime (date_buf, buf_size, _("%X"), &date_tm);
211 compact_header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
212 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
215 TnyMsgHeaderFlags flags;
216 gchar *from, *subject;
221 gtk_tree_model_get (tree_model, iter,
222 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
223 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
224 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &subject,
225 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_TIME_T_COLUMN, &date,
227 rendobj = G_OBJECT(renderer);
229 /* simplistic --> remove <email@address> from display */
230 address = g_strstr_len (from, strlen(from), "<");
232 address[0]='\0'; /* set a new endpoint */
234 header = g_strdup_printf ("%s %s\n%s", from,
235 display_date(date), subject);
236 g_object_set (rendobj, "text", header, NULL);
242 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
243 g_object_set (rendobj, "weight", 800, NULL);
245 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
249 static GtkTreeViewColumn*
250 get_new_column (const gchar *name, GtkCellRenderer *renderer,
251 gboolean resizable, gint sort_col_id, gboolean show_as_text,
252 GtkTreeCellDataFunc cell_data_func, gpointer user_data)
254 GtkTreeViewColumn *column;
256 column = gtk_tree_view_column_new_with_attributes(name, renderer, NULL);
257 gtk_tree_view_column_set_resizable (column, resizable);
260 gtk_tree_view_column_add_attribute (column, renderer, "text",
262 if (sort_col_id >= 0)
263 gtk_tree_view_column_set_sort_column_id (column, sort_col_id);
265 gtk_tree_view_column_set_sort_indicator (column, FALSE);
266 gtk_tree_view_column_set_reorderable (column, TRUE);
269 gtk_tree_view_column_set_cell_data_func(column, renderer, cell_data_func,
272 /* g_signal_connect (G_OBJECT (column), "clicked", */
273 /* G_CALLBACK (column_clicked), obj); */
282 remove_all_columns (ModestTnyHeaderTreeView *obj)
284 GList *columns, *cursor;
286 columns = gtk_tree_view_get_columns (GTK_TREE_VIEW(obj));
288 for (cursor = columns; cursor; cursor = cursor->next)
289 gtk_tree_view_remove_column (GTK_TREE_VIEW(obj),
290 GTK_TREE_VIEW_COLUMN(cursor->data));
291 g_list_free (columns);
298 init_columns (ModestTnyHeaderTreeView *obj)
300 GtkTreeViewColumn *column;
301 GtkCellRenderer *renderer_msgtype,
305 ModestTnyHeaderTreeViewPrivate *priv;
308 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
310 renderer_msgtype = gtk_cell_renderer_pixbuf_new ();
311 renderer_attach = gtk_cell_renderer_pixbuf_new ();
312 renderer_header = gtk_cell_renderer_text_new ();
314 remove_all_columns (obj);
316 for (cursor = priv->columns; cursor; cursor = cursor->next) {
317 ModestTnyHeaderTreeViewColumn col =
318 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
322 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
324 column = get_new_column (_("M"), renderer_msgtype, FALSE,
325 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
326 FALSE, (GtkTreeCellDataFunc)msgtype_cell_data,
330 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
332 column = get_new_column (_("A"), renderer_attach, FALSE,
333 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
334 FALSE, (GtkTreeCellDataFunc)attach_cell_data,
338 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
339 column = get_new_column (_("Received"), renderer_header, TRUE,
340 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
341 TRUE, (GtkTreeCellDataFunc)header_cell_data,
345 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_FROM:
346 column = get_new_column (_("From"), renderer_header, TRUE,
347 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
348 TRUE, (GtkTreeCellDataFunc)sender_cell_data,
352 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
353 column = get_new_column (_("Header"), renderer_header, TRUE,
354 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
355 TRUE, (GtkTreeCellDataFunc)compact_header_cell_data,
359 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT:
360 column = get_new_column (_("Subject"), renderer_header, TRUE,
361 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
362 TRUE, (GtkTreeCellDataFunc)header_cell_data,
367 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
368 column = get_new_column (_("Sent"), renderer_header, TRUE,
369 TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
370 TRUE, (GtkTreeCellDataFunc)header_cell_data,
374 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_TO:
375 column = get_new_column (_("To"), renderer_header, TRUE,
376 TNY_MSG_HEADER_LIST_MODEL_TO_COLUMN,
377 TRUE, (GtkTreeCellDataFunc)header_cell_data,
381 gtk_tree_view_append_column (GTK_TREE_VIEW(obj), column);
390 modest_tny_header_tree_view_init (ModestTnyHeaderTreeView *obj)
392 ModestTnyHeaderTreeViewPrivate *priv;
395 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
397 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i)
398 priv->sort_columns[i] = -1;
403 modest_tny_header_tree_view_finalize (GObject *obj)
405 ModestTnyHeaderTreeView *self;
406 ModestTnyHeaderTreeViewPrivate *priv;
409 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
410 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
413 g_object_unref (G_OBJECT(priv->headers));
415 priv->headers = NULL;
416 priv->tny_msg_folder = NULL;
420 modest_tny_header_tree_view_new (TnyMsgFolderIface *folder,
422 ModestTnyHeaderTreeViewStyle style)
425 GtkTreeSelection *sel;
426 ModestTnyHeaderTreeView *self;
428 obj = G_OBJECT(g_object_new(MODEST_TYPE_TNY_HEADER_TREE_VIEW, NULL));
429 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
431 if (!modest_tny_header_tree_view_set_folder (self, NULL)) {
432 g_warning ("could not set the folder");
433 g_object_unref (obj);
437 modest_tny_header_tree_view_set_style (self, style);
438 modest_tny_header_tree_view_set_columns (self, columns);
441 gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(obj), TRUE);
442 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(obj), TRUE);
444 gtk_tree_view_set_rules_hint (GTK_TREE_VIEW(obj),
445 TRUE); /* alternating row colors */
447 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(self));
448 g_signal_connect (sel, "changed",
449 G_CALLBACK(selection_changed), self);
451 return GTK_WIDGET(self);
455 modest_tny_header_tree_view_set_columns (ModestTnyHeaderTreeView *self, GSList *columns)
457 ModestTnyHeaderTreeViewPrivate *priv;
460 g_return_val_if_fail (self, FALSE);
462 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
464 g_slist_free (priv->columns);
466 for (cursor = columns; cursor; cursor = cursor->next) {
467 ModestTnyHeaderTreeViewColumn col =
468 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
469 if (0 > col || col >= MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM)
470 g_warning ("invalid column in column list");
472 priv->columns = g_slist_append (priv->columns, cursor->data);
475 init_columns (self); /* redraw them */
482 modest_tny_header_tree_view_get_columns (ModestTnyHeaderTreeView *self)
484 ModestTnyHeaderTreeViewPrivate *priv;
487 g_return_val_if_fail (self, FALSE);
489 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
490 return priv->columns;
497 modest_tny_header_tree_view_set_style (ModestTnyHeaderTreeView *self,
498 ModestTnyHeaderTreeViewStyle style)
500 g_return_val_if_fail (self, FALSE);
501 g_return_val_if_fail (style >= 0 && style < MODEST_TNY_HEADER_TREE_VIEW_STYLE_NUM,
504 MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style = style;
509 ModestTnyHeaderTreeViewStyle
510 modest_tny_header_tree_view_get_style (ModestTnyHeaderTreeView *self)
512 g_return_val_if_fail (self, FALSE);
514 return MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style;
519 /* get the length of any prefix that should be ignored for sorting */
521 get_prefix_len (const gchar *sub)
524 const static gchar* prefix[] = {"Re:", "RE:", "Fwd:", "FWD:", NULL};
526 if (sub[0] != 'R' && sub[0] != 'F') /* optimization */
530 if (g_str_has_prefix(sub, prefix[i])) {
531 int prefix_len = strlen(prefix[i]);
532 if (sub[prefix_len + 1] == ' ')
533 ++prefix_len; /* ignore space after prefix as well */
542 cmp_normalized_subject (const gchar* s1, const gchar *s2)
545 return strcmp (s1 + get_prefix_len(s1),
546 s2 + get_prefix_len(s2));
551 cmp_rows (GtkTreeModel *tree_model, GtkTreeIter *iter1, GtkTreeIter *iter2,
554 gint col_id = GPOINTER_TO_INT (user_data);
557 g_return_val_if_fail (GTK_IS_TREE_MODEL(tree_model), -1);
561 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
562 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
563 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
565 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
570 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
571 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
573 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
577 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT: {
581 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
583 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
586 retval = cmp_normalized_subject(sub1, sub2);
594 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
595 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
597 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
600 return (val1 & TNY_MSG_HEADER_FLAG_ATTACHMENTS) - (val2 & TNY_MSG_HEADER_FLAG_ATTACHMENTS);
603 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
604 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
606 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
609 g_message ("%p %p", iter1, iter2);
610 return (val1 & TNY_MSG_HEADER_FLAG_SEEN) - (val2 & TNY_MSG_HEADER_FLAG_SEEN);
613 g_message ("%p %p", iter1, iter2);
614 return &iter1 - &iter2;
620 modest_tny_header_tree_view_set_folder (ModestTnyHeaderTreeView *self,
621 TnyMsgFolderIface *folder)
624 GtkTreeModel *oldsortable, *sortable, *oldmodel;
625 ModestTnyHeaderTreeViewPrivate *priv;
627 g_return_val_if_fail (self, FALSE);
629 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
632 priv->headers = TNY_LIST_IFACE(tny_msg_header_list_model_new ());
633 tny_msg_folder_iface_get_headers (folder, priv->headers,
635 tny_msg_header_list_model_set_folder (TNY_MSG_HEADER_LIST_MODEL(priv->headers),
638 oldsortable = gtk_tree_view_get_model(GTK_TREE_VIEW (self));
639 if (oldsortable && GTK_IS_TREE_MODEL_SORT(oldsortable)) {
640 GtkTreeModel *oldmodel = gtk_tree_model_sort_get_model
641 (GTK_TREE_MODEL_SORT(oldsortable));
643 g_object_unref (G_OBJECT(oldmodel));
644 g_object_unref (oldsortable);
647 sortable = gtk_tree_model_sort_new_with_model (GTK_TREE_MODEL(priv->headers));
649 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i) {
650 int col_id = priv->sort_columns[i];
652 g_message ("%d: %p: %p: %d", i, GTK_TREE_SORTABLE(sortable), sortable, col_id);
653 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE(sortable), col_id,
654 (GtkTreeIterCompareFunc)cmp_rows,
655 GINT_TO_POINTER(col_id), NULL);
658 gtk_tree_view_set_model (GTK_TREE_VIEW (self), sortable);
659 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), TRUE);
661 /* no need to unref sortable */
663 } else /* when there is no folder */
664 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), FALSE);
671 selection_changed (GtkTreeSelection *sel, gpointer user_data)
674 TnyMsgHeaderIface *header;
676 ModestTnyHeaderTreeView *tree_view;
678 g_return_if_fail (sel);
679 g_return_if_fail (user_data);
681 if (!gtk_tree_selection_get_selected (sel, &model, &iter))
682 return; /* msg was _un_selected */
684 tree_view = MODEST_TNY_HEADER_TREE_VIEW (user_data);
686 gtk_tree_model_get (model, &iter,
687 TNY_MSG_HEADER_LIST_MODEL_INSTANCE_COLUMN,
691 TnyMsgHeaderFlags flags;
692 const TnyMsgIface *msg;
693 const TnyMsgFolderIface *folder;
695 folder = tny_msg_header_iface_get_folder (TNY_MSG_HEADER_IFACE(header));
697 g_message ("cannot find folder");
699 msg = tny_msg_folder_iface_get_message (TNY_MSG_FOLDER_IFACE(folder),
702 g_message ("cannot find msg");
703 /* FIXME: update display */
707 g_signal_emit (G_OBJECT(tree_view), signals[MESSAGE_SELECTED_SIGNAL], 0,
710 /* mark message as seen; _set_flags crashes, bug in tinymail? */
711 flags = tny_msg_header_iface_get_flags (TNY_MSG_HEADER_IFACE(header));
712 //tny_msg_header_iface_set_flags (header, flags | TNY_MSG_HEADER_FLAG_SEEN);
717 column_clicked (GtkTreeViewColumn *col, gpointer user_data)
719 GtkTreeView *treeview;
722 treeview = GTK_TREE_VIEW (user_data);
723 id = gtk_tree_view_column_get_sort_column_id (col);
725 gtk_tree_view_set_search_column (treeview, id);