1 /* modest-tny-header-tree-view.c
3 #include <glib/gi18n.h>
4 #include "modest-tny-header-tree-view.h"
5 #include <tny-list-iface.h>
8 /* 'private'/'protected' functions */
9 static void modest_tny_header_tree_view_class_init (ModestTnyHeaderTreeViewClass *klass);
10 static void modest_tny_header_tree_view_init (ModestTnyHeaderTreeView *obj);
11 static void modest_tny_header_tree_view_finalize (GObject *obj);
13 static void selection_changed (GtkTreeSelection *sel, gpointer user_data);
14 static void column_clicked (GtkTreeViewColumn *treeviewcolumn, gpointer user_data);
16 #define PIXMAP_PREFIX PREFIX "/share/modest/pixmaps/"
19 MESSAGE_SELECTED_SIGNAL,
31 typedef struct _ModestTnyHeaderTreeViewPrivate ModestTnyHeaderTreeViewPrivate;
32 struct _ModestTnyHeaderTreeViewPrivate {
34 TnyMsgFolderIface *tny_msg_folder;
35 TnyListIface *headers;
37 GdkPixbuf *icons[HEADER_ICON_NUM];
38 guint sort_columns[MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM];
41 ModestTnyHeaderTreeViewStyle style;
43 #define MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE((o), \
44 MODEST_TYPE_TNY_HEADER_TREE_VIEW, \
45 ModestTnyHeaderTreeViewPrivate))
47 static GObjectClass *parent_class = NULL;
49 /* uncomment the following if you have defined any signals */
50 static guint signals[LAST_SIGNAL] = {0};
53 modest_tny_header_tree_view_get_type (void)
55 static GType my_type = 0;
57 static const GTypeInfo my_info = {
58 sizeof(ModestTnyHeaderTreeViewClass),
60 NULL, /* base finalize */
61 (GClassInitFunc) modest_tny_header_tree_view_class_init,
62 NULL, /* class finalize */
63 NULL, /* class data */
64 sizeof(ModestTnyHeaderTreeView),
66 (GInstanceInitFunc) modest_tny_header_tree_view_init,
68 my_type = g_type_register_static (GTK_TYPE_TREE_VIEW,
69 "ModestTnyHeaderTreeView",
76 modest_tny_header_tree_view_class_init (ModestTnyHeaderTreeViewClass *klass)
78 GObjectClass *gobject_class;
79 gobject_class = (GObjectClass*) klass;
81 parent_class = g_type_class_peek_parent (klass);
82 gobject_class->finalize = modest_tny_header_tree_view_finalize;
84 g_type_class_add_private (gobject_class, sizeof(ModestTnyHeaderTreeViewPrivate));
86 signals[MESSAGE_SELECTED_SIGNAL] =
87 g_signal_new ("message_selected",
88 G_TYPE_FROM_CLASS (gobject_class),
90 G_STRUCT_OFFSET (ModestTnyHeaderTreeViewClass,message_selected),
92 g_cclosure_marshal_VOID__POINTER,
93 G_TYPE_NONE, 1, G_TYPE_POINTER);
101 msgtype_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
102 GtkTreeModel *tree_model, GtkTreeIter *iter,
103 GdkPixbuf *icons[HEADER_ICON_NUM])
105 TnyMsgHeaderFlags flags;
108 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
111 if (flags & TNY_MSG_HEADER_FLAG_SEEN)
112 pixbuf = icons[HEADER_ICON_READ];
114 pixbuf = icons[HEADER_ICON_UNREAD];
116 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf,
121 attach_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
122 GtkTreeModel *tree_model, GtkTreeIter *iter, GdkPixbuf *icons[HEADER_ICON_NUM])
124 TnyMsgHeaderFlags flags;
127 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
129 if (flags & TNY_MSG_HEADER_FLAG_ATTACHMENTS)
130 pixbuf = icons[HEADER_ICON_ATTACH];
134 g_object_set (G_OBJECT (renderer), "pixbuf", pixbuf, NULL);
140 header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
141 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
144 TnyMsgHeaderFlags flags;
146 gtk_tree_model_get (tree_model, iter, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
148 rendobj = G_OBJECT(renderer);
150 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
151 g_object_set (rendobj, "weight", 800, NULL);
153 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
159 sender_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
160 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
163 TnyMsgHeaderFlags flags;
167 gtk_tree_model_get (tree_model, iter,
168 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
169 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
170 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &flags
172 rendobj = G_OBJECT(renderer);
174 /* simplistic --> remove <email@address> from display */
175 address = g_strstr_len (from, strlen(from), "<");
178 g_object_set (rendobj, "text", from, NULL);
182 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
183 g_object_set (rendobj, "weight", 800, NULL);
185 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
190 compact_header_cell_data (GtkTreeViewColumn *column, GtkCellRenderer *renderer,
191 GtkTreeModel *tree_model, GtkTreeIter *iter, gpointer user_data)
194 TnyMsgHeaderFlags flags;
195 gchar *from, *subject, *date;
199 gtk_tree_model_get (tree_model, iter,
200 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN, &flags,
201 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN, &from,
202 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN, &subject,
203 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN, &date,
205 rendobj = G_OBJECT(renderer);
207 /* simplistic --> remove <email@address> from display */
208 address = g_strstr_len (from, strlen(from), "<");
210 address[0]='\0'; /* set a new endpoint */
212 header = g_strdup_printf ("%s %s\n%s", from, date, subject);
213 g_object_set (rendobj, "text", header, NULL);
220 if (!(flags & TNY_MSG_HEADER_FLAG_SEEN))
221 g_object_set (rendobj, "weight", 800, NULL);
223 g_object_set (rendobj, "weight", 400, NULL); /* default, non-bold */
231 init_icons (GdkPixbuf *icons[HEADER_ICON_NUM])
233 icons[HEADER_ICON_UNREAD] =
234 gdk_pixbuf_new_from_file (PIXMAP_PREFIX "qgn_list_messagin_mail_unread.png",NULL);
235 icons[HEADER_ICON_READ] =
236 gdk_pixbuf_new_from_file (PIXMAP_PREFIX "qgn_list_messagin_mail.png",NULL);
237 icons[HEADER_ICON_ATTACH] =
238 gdk_pixbuf_new_from_file (PIXMAP_PREFIX "clip.xpm",NULL);
243 static GtkTreeViewColumn*
244 get_new_column (const gchar *name, GtkCellRenderer *renderer,
245 gboolean resizable, gint sort_col_id, gboolean show_as_text,
246 GtkTreeCellDataFunc cell_data_func, gpointer user_data)
248 GtkTreeViewColumn *column;
250 column = gtk_tree_view_column_new_with_attributes(name, renderer, NULL);
251 gtk_tree_view_column_set_resizable (column, resizable);
254 gtk_tree_view_column_add_attribute (column, renderer, "text",
256 if (sort_col_id >= 0)
257 gtk_tree_view_column_set_sort_column_id (column, sort_col_id);
259 gtk_tree_view_column_set_sort_indicator (column, FALSE);
260 gtk_tree_view_column_set_reorderable (column, TRUE);
263 gtk_tree_view_column_set_cell_data_func(column, renderer, cell_data_func,
266 /* g_signal_connect (G_OBJECT (column), "clicked", */
267 /* G_CALLBACK (column_clicked), obj); */
276 remove_all_columns (ModestTnyHeaderTreeView *obj)
278 GList *columns, *cursor;
280 columns = gtk_tree_view_get_columns (GTK_TREE_VIEW(obj));
282 for (cursor = columns; cursor; cursor = cursor->next)
283 gtk_tree_view_remove_column (GTK_TREE_VIEW(obj),
284 GTK_TREE_VIEW_COLUMN(cursor->data));
285 g_list_free (columns);
292 init_columns (ModestTnyHeaderTreeView *obj)
294 GtkTreeViewColumn *column;
295 GtkCellRenderer *renderer_msgtype,
299 ModestTnyHeaderTreeViewPrivate *priv;
302 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
304 renderer_msgtype = gtk_cell_renderer_pixbuf_new ();
305 renderer_attach = gtk_cell_renderer_pixbuf_new ();
306 renderer_header = gtk_cell_renderer_text_new ();
308 remove_all_columns (obj);
310 for (cursor = priv->columns; cursor; cursor = cursor->next) {
311 ModestTnyHeaderTreeViewColumn col =
312 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
316 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
318 column = get_new_column (_("M"), renderer_msgtype, FALSE,
319 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
320 FALSE, (GtkTreeCellDataFunc)msgtype_cell_data, priv->icons);
323 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
325 column = get_new_column (_("A"), renderer_attach, FALSE,
326 TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
327 FALSE, (GtkTreeCellDataFunc)attach_cell_data, priv->icons);
330 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
331 column = get_new_column (_("Received"), renderer_header, TRUE,
332 TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
333 TRUE, (GtkTreeCellDataFunc)header_cell_data, NULL);
336 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_FROM:
337 column = get_new_column (_("From"), renderer_header, TRUE,
338 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
339 TRUE, (GtkTreeCellDataFunc)sender_cell_data, NULL);
342 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
343 column = get_new_column (_("Header"), renderer_header, TRUE,
344 TNY_MSG_HEADER_LIST_MODEL_FROM_COLUMN,
345 TRUE, (GtkTreeCellDataFunc)compact_header_cell_data, NULL);
348 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT:
349 column = get_new_column (_("Subject"), renderer_header, TRUE,
350 TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
351 TRUE, (GtkTreeCellDataFunc)header_cell_data, NULL);
355 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
356 column = get_new_column (_("Sent"), renderer_header, TRUE,
357 TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
358 TRUE, (GtkTreeCellDataFunc)header_cell_data, NULL);
361 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_TO:
362 column = get_new_column (_("To"), renderer_header, TRUE,
363 TNY_MSG_HEADER_LIST_MODEL_TO_COLUMN,
364 TRUE, (GtkTreeCellDataFunc)header_cell_data, NULL);
367 gtk_tree_view_append_column (GTK_TREE_VIEW(obj), column);
376 modest_tny_header_tree_view_init (ModestTnyHeaderTreeView *obj)
378 ModestTnyHeaderTreeViewPrivate *priv;
381 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(obj);
383 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i)
384 priv->sort_columns[i] = -1;
387 init_icons (priv->icons);
391 modest_tny_header_tree_view_finalize (GObject *obj)
393 ModestTnyHeaderTreeView *self;
394 ModestTnyHeaderTreeViewPrivate *priv;
397 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
398 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
401 g_object_unref (G_OBJECT(priv->headers));
403 priv->headers = NULL;
404 priv->tny_msg_folder = NULL;
406 /* cleanup our icons */
407 for (i = 0; i != HEADER_ICON_NUM; ++i) {
408 if (priv->icons[i]) {
409 g_object_unref (G_OBJECT(priv->icons[i]));
410 priv->icons[i] = NULL;
416 modest_tny_header_tree_view_new (TnyMsgFolderIface *folder,
418 ModestTnyHeaderTreeViewStyle style)
421 GtkTreeSelection *sel;
422 ModestTnyHeaderTreeView *self;
424 obj = G_OBJECT(g_object_new(MODEST_TYPE_TNY_HEADER_TREE_VIEW, NULL));
425 self = MODEST_TNY_HEADER_TREE_VIEW(obj);
427 if (!modest_tny_header_tree_view_set_folder (self, NULL)) {
428 g_warning ("could not set the folder");
429 g_object_unref (obj);
433 modest_tny_header_tree_view_set_style (self, style);
434 modest_tny_header_tree_view_set_columns (self, columns);
437 gtk_tree_view_set_headers_visible (GTK_TREE_VIEW(obj), TRUE);
438 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(obj), TRUE);
440 gtk_tree_view_set_rules_hint (GTK_TREE_VIEW(obj),
441 TRUE); /* alternating row colors */
443 sel = gtk_tree_view_get_selection(GTK_TREE_VIEW(self));
444 g_signal_connect (sel, "changed",
445 G_CALLBACK(selection_changed), self);
447 return GTK_WIDGET(self);
451 modest_tny_header_tree_view_set_columns (ModestTnyHeaderTreeView *self, GSList *columns)
453 ModestTnyHeaderTreeViewPrivate *priv;
456 g_return_val_if_fail (self, FALSE);
458 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
460 g_slist_free (priv->columns);
462 for (cursor = columns; cursor; cursor = cursor->next) {
463 ModestTnyHeaderTreeViewColumn col =
464 (ModestTnyHeaderTreeViewColumn) GPOINTER_TO_INT(cursor->data);
465 if (0 > col || col >= MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM)
466 g_warning ("invalid column in column list");
468 priv->columns = g_slist_append (priv->columns, cursor->data);
471 init_columns (self); /* redraw them */
478 modest_tny_header_tree_view_get_columns (ModestTnyHeaderTreeView *self)
480 ModestTnyHeaderTreeViewPrivate *priv;
483 g_return_val_if_fail (self, FALSE);
485 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
486 return priv->columns;
493 modest_tny_header_tree_view_set_style (ModestTnyHeaderTreeView *self,
494 ModestTnyHeaderTreeViewStyle style)
496 g_return_val_if_fail (self, FALSE);
497 g_return_val_if_fail (style >= 0 && style < MODEST_TNY_HEADER_TREE_VIEW_STYLE_NUM,
500 MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style = style;
505 ModestTnyHeaderTreeViewStyle
506 modest_tny_header_tree_view_get_style (ModestTnyHeaderTreeView *self)
508 g_return_val_if_fail (self, FALSE);
510 return MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self)->style;
515 /* get the length of any prefix that should be ignored for sorting */
517 get_prefix_len (const gchar *sub)
520 const static gchar* prefix[] = {"Re:", "RE:", "Fwd:", "FWD:", NULL};
522 if (sub[0] != 'R' && sub[0] != 'F') /* optimization */
526 if (g_str_has_prefix(sub, prefix[i])) {
527 int prefix_len = strlen(prefix[i]);
528 if (sub[prefix_len + 1] == ' ')
529 ++prefix_len; /* ignore space after prefix as well */
538 cmp_normalized_subject (const gchar* s1, const gchar *s2)
541 return strcmp (s1 + get_prefix_len(s1),
542 s2 + get_prefix_len(s2));
547 cmp_rows (GtkTreeModel *tree_model, GtkTreeIter *iter1, GtkTreeIter *iter2,
550 gint col_id = GPOINTER_TO_INT (user_data);
553 g_return_val_if_fail (GTK_IS_TREE_MODEL(tree_model), -1);
557 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_COMPACT_HEADER:
558 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_RECEIVED_DATE:
559 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
561 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_RECEIVED_COLUMN,
566 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SENT_DATE:
567 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
569 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_DATE_SENT_COLUMN,
573 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_SUBJECT: {
577 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
579 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_SUBJECT_COLUMN,
582 retval = cmp_normalized_subject(sub1, sub2);
590 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_ATTACH:
591 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
593 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
596 return (val1 & TNY_MSG_HEADER_FLAG_ATTACHMENTS) - (val2 & TNY_MSG_HEADER_FLAG_ATTACHMENTS);
599 case MODEST_TNY_HEADER_TREE_VIEW_COLUMN_MSGTYPE:
600 gtk_tree_model_get (tree_model, iter1, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
602 gtk_tree_model_get (tree_model, iter2, TNY_MSG_HEADER_LIST_MODEL_FLAGS_COLUMN,
605 g_message ("%p %p", iter1, iter2);
606 return (val1 & TNY_MSG_HEADER_FLAG_SEEN) - (val2 & TNY_MSG_HEADER_FLAG_SEEN);
609 g_message ("%p %p", iter1, iter2);
610 return &iter1 - &iter2;
616 modest_tny_header_tree_view_set_folder (ModestTnyHeaderTreeView *self,
617 TnyMsgFolderIface *folder)
620 GtkTreeModel *oldsortable, *sortable, *oldmodel;
621 ModestTnyHeaderTreeViewPrivate *priv;
623 g_return_val_if_fail (self, FALSE);
625 priv = MODEST_TNY_HEADER_TREE_VIEW_GET_PRIVATE(self);
628 priv->headers = TNY_LIST_IFACE(tny_msg_header_list_model_new ());
629 tny_msg_folder_iface_get_headers (folder, priv->headers,
631 tny_msg_header_list_model_set_folder (TNY_MSG_HEADER_LIST_MODEL(priv->headers),
634 oldsortable = gtk_tree_view_get_model(GTK_TREE_VIEW (self));
635 if (oldsortable && GTK_IS_TREE_MODEL_SORT(oldsortable)) {
636 GtkTreeModel *oldmodel = gtk_tree_model_sort_get_model
637 (GTK_TREE_MODEL_SORT(oldsortable));
639 g_object_unref (G_OBJECT(oldmodel));
640 g_object_unref (oldsortable);
643 sortable = gtk_tree_model_sort_new_with_model (GTK_TREE_MODEL(priv->headers));
645 for (i = 0; i != MODEST_TNY_HEADER_TREE_VIEW_COLUMN_NUM; ++i) {
646 int col_id = priv->sort_columns[i];
648 g_message ("%d: %p: %p: %d", i, GTK_TREE_SORTABLE(sortable), sortable, col_id);
649 gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE(sortable), col_id,
650 (GtkTreeIterCompareFunc)cmp_rows,
651 GINT_TO_POINTER(col_id), NULL);
654 gtk_tree_view_set_model (GTK_TREE_VIEW (self), sortable);
655 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), TRUE);
657 /* no need to unref sortable */
659 } else /* when there is no folder */
660 gtk_tree_view_set_headers_clickable (GTK_TREE_VIEW(self), FALSE);
667 selection_changed (GtkTreeSelection *sel, gpointer user_data)
670 TnyMsgHeaderIface *header;
672 ModestTnyHeaderTreeView *tree_view;
674 g_return_if_fail (sel);
675 g_return_if_fail (user_data);
677 if (!gtk_tree_selection_get_selected (sel, &model, &iter))
678 return; /* msg was _un_selected */
680 tree_view = MODEST_TNY_HEADER_TREE_VIEW (user_data);
682 gtk_tree_model_get (model, &iter,
683 TNY_MSG_HEADER_LIST_MODEL_INSTANCE_COLUMN,
687 TnyMsgHeaderFlags flags;
688 const TnyMsgIface *msg;
689 const TnyMsgFolderIface *folder;
691 folder = tny_msg_header_iface_get_folder (TNY_MSG_HEADER_IFACE(header));
693 g_message ("cannot find folder");
695 msg = tny_msg_folder_iface_get_message (TNY_MSG_FOLDER_IFACE(folder),
698 g_message ("cannot find msg");
699 /* FIXME: update display */
703 g_signal_emit (G_OBJECT(tree_view), signals[MESSAGE_SELECTED_SIGNAL], 0,
706 /* mark message as seen; _set_flags crashes, bug in tinymail? */
707 flags = tny_msg_header_iface_get_flags (TNY_MSG_HEADER_IFACE(header));
708 //tny_msg_header_iface_set_flags (header, flags | TNY_MSG_HEADER_FLAG_SEEN);
713 column_clicked (GtkTreeViewColumn *col, gpointer user_data)
715 GtkTreeView *treeview;
718 treeview = GTK_TREE_VIEW (user_data);
719 id = gtk_tree_view_column_get_sort_column_id (col);
721 gtk_tree_view_set_search_column (treeview, id);