2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2008 Hiroyuki Yamamoto and the Claws Mail team
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
23 #include <glib/gi18n.h>
24 #include <gdk/gdkkeysyms.h>
25 #include <gtk/gtkscrolledwindow.h>
26 #include <gtk/gtkwidget.h>
27 #include <gtk/gtkpixmap.h>
28 #include <gtk/gtkctree.h>
29 #include <gtk/gtkcontainer.h>
30 #include <gtk/gtksignal.h>
31 #include <gtk/gtktext.h>
32 #include <gtk/gtkmenu.h>
33 #include <gtk/gtkmenuitem.h>
34 #include <gtk/gtkitemfactory.h>
35 #include <gtk/gtkvbox.h>
36 #include <gtk/gtkhbox.h>
37 #include <gtk/gtkwindow.h>
38 #include <gtk/gtkstyle.h>
39 #include <gtk/gtkarrow.h>
40 #include <gtk/gtkeventbox.h>
41 #include <gtk/gtkstatusbar.h>
42 #include <gtk/gtkmenuitem.h>
52 #include "mainwindow.h"
53 #include "folderview.h"
54 #include "summaryview.h"
55 #include "messageview.h"
56 #include "foldersel.h"
58 #include "procheader.h"
59 #include "sourcewindow.h"
60 #include "prefs_common.h"
61 #include "prefs_summary_column.h"
62 #include "prefs_summary_open.h"
63 #include "prefs_filtering.h"
68 #include "stock_pixmap.h"
70 #include "alertpanel.h"
71 #include "inputdialog.h"
72 #include "statusbar.h"
74 #include "colorlabel.h"
77 #include "addressbook.h"
78 #include "addr_compl.h"
79 #include "folder_item_prefs.h"
80 #include "filtering.h"
81 #include "string_match.h"
85 #include "description_window.h"
86 #include "folderutils.h"
87 #include "quicksearch.h"
88 #include "partial_download.h"
91 #include "gedit-print.h"
96 #define SUMMARY_COL_MARK_WIDTH 10
97 #define SUMMARY_COL_STATUS_WIDTH 13
98 #define SUMMARY_COL_LOCKED_WIDTH 13
99 #define SUMMARY_COL_MIME_WIDTH 11
101 static int normal_row_height = -1;
102 static GtkStyle *bold_style;
103 static GtkStyle *bold_marked_style;
104 static GtkStyle *bold_deleted_style;
105 static GtkStyle *small_style;
106 static GtkStyle *small_marked_style;
107 static GtkStyle *small_deleted_style;
109 static GdkPixmap *markxpm;
110 static GdkBitmap *markxpmmask;
111 static GdkPixmap *deletedxpm;
112 static GdkBitmap *deletedxpmmask;
113 static GdkPixmap *movedxpm;
114 static GdkBitmap *movedxpmmask;
115 static GdkPixmap *copiedxpm;
116 static GdkBitmap *copiedxpmmask;
118 static GdkPixmap *newxpm;
119 static GdkBitmap *newxpmmask;
120 static GdkPixmap *unreadxpm;
121 static GdkBitmap *unreadxpmmask;
122 static GdkPixmap *repliedxpm;
123 static GdkBitmap *repliedxpmmask;
124 static GdkPixmap *forwardedxpm;
125 static GdkBitmap *forwardedxpmmask;
126 static GdkPixmap *ignorethreadxpm;
127 static GdkBitmap *ignorethreadxpmmask;
128 static GdkPixmap *watchthreadxpm;
129 static GdkBitmap *watchthreadxpmmask;
130 static GdkPixmap *lockedxpm;
131 static GdkBitmap *lockedxpmmask;
132 static GdkPixmap *spamxpm;
133 static GdkBitmap *spamxpmmask;
135 static GdkPixmap *clipxpm;
136 static GdkBitmap *clipxpmmask;
137 static GdkPixmap *keyxpm;
138 static GdkBitmap *keyxpmmask;
139 static GdkPixmap *clipkeyxpm;
140 static GdkBitmap *clipkeyxpmmask;
141 static GdkPixmap *gpgsignedxpm;
142 static GdkBitmap *gpgsignedxpmmask;
143 static GdkPixmap *clipgpgsignedxpm;
144 static GdkBitmap *clipgpgsignedxpmmask;
146 static void summary_free_msginfo_func (GtkCTree *ctree,
149 static void summary_set_marks_func (GtkCTree *ctree,
153 void summary_set_menu_sensitive (SummaryView *summaryview);
154 guint summary_get_msgnum (SummaryView *summaryview,
158 static void summary_set_hide_read_msgs_menu (SummaryView *summaryview,
161 static GtkCTreeNode *summary_find_prev_msg
162 (SummaryView *summaryview,
163 GtkCTreeNode *current_node);
164 static GtkCTreeNode *summary_find_next_msg
165 (SummaryView *summaryview,
166 GtkCTreeNode *current_node);
168 static GtkCTreeNode *summary_find_prev_flagged_msg
169 (SummaryView *summaryview,
170 GtkCTreeNode *current_node,
172 gboolean start_from_prev);
173 static GtkCTreeNode *summary_find_next_flagged_msg
174 (SummaryView *summaryview,
175 GtkCTreeNode *current_node,
177 gboolean start_from_next);
179 static GtkCTreeNode *summary_find_msg_by_msgnum
180 (SummaryView *summaryview,
183 static void summary_update_status (SummaryView *summaryview);
185 /* display functions */
186 static void summary_status_show (SummaryView *summaryview);
187 static void summary_set_column_titles (SummaryView *summaryview);
188 static void summary_set_ctree_from_list (SummaryView *summaryview,
190 static inline void summary_set_header (SummaryView *summaryview,
193 static void summary_display_msg (SummaryView *summaryview,
195 static void summary_display_msg_full (SummaryView *summaryview,
198 gboolean all_headers);
199 static void summary_set_row_marks (SummaryView *summaryview,
202 static gboolean summary_set_row_tag (SummaryView *summaryview,
207 /* message handling */
208 static void summary_mark_row (SummaryView *summaryview,
210 static void summary_lock_row (SummaryView *summaryview,
212 static void summary_unlock_row (SummaryView *summaryview,
214 static void summary_mark_row_as_read (SummaryView *summaryview,
216 static void summary_mark_row_as_unread (SummaryView *summaryview,
218 static void summary_delete_row (SummaryView *summaryview,
220 static void summary_unmark_row (SummaryView *summaryview,
222 static void summary_move_row_to (SummaryView *summaryview,
224 FolderItem *to_folder);
225 static void summary_copy_row_to (SummaryView *summaryview,
227 FolderItem *to_folder);
229 static gint summary_execute_move (SummaryView *summaryview);
230 static void summary_execute_move_func (GtkCTree *ctree,
233 static void summary_execute_copy (SummaryView *summaryview);
234 static void summary_execute_copy_func (GtkCTree *ctree,
237 static void summary_execute_delete (SummaryView *summaryview);
238 static void summary_execute_delete_func (GtkCTree *ctree,
242 static void summary_thread_init (SummaryView *summaryview);
244 static void summary_unthread_for_exec (SummaryView *summaryview);
245 static void summary_unthread_for_exec_func (GtkCTree *ctree,
249 void summary_simplify_subject(SummaryView *summaryview, gchar * rexp,
252 static void summary_filter_func (MsgInfo *msginfo);
254 static void summary_colorlabel_menu_item_activate_cb
257 static void summary_colorlabel_menu_item_activate_item_cb
258 (GtkMenuItem *label_menu_item,
260 static void summary_colorlabel_menu_create(SummaryView *summaryview,
262 static void summary_tags_menu_item_activate_cb
265 static void summary_tags_menu_item_activate_item_cb
266 (GtkMenuItem *label_menu_item,
268 static void summary_tags_menu_create(SummaryView *summaryview,
271 static GtkWidget *summary_ctree_create (SummaryView *summaryview);
273 /* callback functions */
274 static gint summary_toggle_pressed (GtkWidget *eventbox,
275 GdkEventButton *event,
276 SummaryView *summaryview);
278 static void summary_toggle_multiple_pressed
280 SummaryView *summaryview);
282 static gint summary_folder_eventbox_pressed
283 (GtkWidget *eventbox,
284 GdkEventButton *event,
285 SummaryView *summaryview);
286 static gboolean summary_button_pressed (GtkWidget *ctree,
287 GdkEventButton *event,
288 SummaryView *summaryview);
289 static gboolean summary_button_released (GtkWidget *ctree,
290 GdkEventButton *event,
291 SummaryView *summaryview);
292 static gboolean summary_key_pressed (GtkWidget *ctree,
294 SummaryView *summaryview);
295 static void summary_tree_expanded (GtkCTree *ctree,
297 SummaryView *summaryview);
298 static void summary_tree_collapsed (GtkCTree *ctree,
300 SummaryView *summaryview);
301 static void summary_selected (GtkCTree *ctree,
304 SummaryView *summaryview);
305 static void summary_unselected (GtkCTree *ctree,
308 SummaryView *summaryview);
309 static void summary_col_resized (GtkCList *clist,
312 SummaryView *summaryview);
313 static void summary_reply_cb (SummaryView *summaryview,
317 static void summary_show_all_header_cb (SummaryView *summaryview,
321 static void summary_add_address_cb (SummaryView *summaryview,
325 static void summary_create_filter_cb (SummaryView *summaryview,
329 static void summary_create_processing_cb(SummaryView *summaryview,
333 static void summary_mark_clicked (GtkWidget *button,
334 SummaryView *summaryview);
335 static void summary_status_clicked (GtkWidget *button,
336 SummaryView *summaryview);
337 static void summary_mime_clicked (GtkWidget *button,
338 SummaryView *summaryview);
339 static void summary_num_clicked (GtkWidget *button,
340 SummaryView *summaryview);
341 static void summary_score_clicked (GtkWidget *button,
342 SummaryView *summaryview);
343 static void summary_size_clicked (GtkWidget *button,
344 SummaryView *summaryview);
345 static void summary_date_clicked (GtkWidget *button,
346 SummaryView *summaryview);
347 static void summary_from_clicked (GtkWidget *button,
348 SummaryView *summaryview);
349 static void summary_to_clicked (GtkWidget *button,
350 SummaryView *summaryview);
351 static void summary_subject_clicked (GtkWidget *button,
352 SummaryView *summaryview);
353 static void summary_score_clicked (GtkWidget *button,
354 SummaryView *summaryview);
355 static void summary_locked_clicked (GtkWidget *button,
356 SummaryView *summaryview);
357 static void summary_tags_clicked (GtkWidget *button,
358 SummaryView *summaryview);
360 static void summary_start_drag (GtkWidget *widget,
363 SummaryView *summaryview);
364 static void summary_drag_data_get (GtkWidget *widget,
365 GdkDragContext *drag_context,
366 GtkSelectionData *selection_data,
369 SummaryView *summaryview);
370 static void summary_drag_data_received(GtkWidget *widget,
371 GdkDragContext *drag_context,
374 GtkSelectionData *data,
377 SummaryView *summaryview);
378 static gboolean summary_drag_motion_cb(GtkWidget *widget,
379 GdkDragContext *context,
383 SummaryView *summaryview);
384 static void summary_drag_end(GtkWidget *widget,
385 GdkDragContext *drag_context,
386 SummaryView *summaryview);
387 /* custom compare functions for sorting */
389 static gint summary_cmp_by_mark (GtkCList *clist,
392 static gint summary_cmp_by_status (GtkCList *clist,
395 static gint summary_cmp_by_mime (GtkCList *clist,
398 static gint summary_cmp_by_num (GtkCList *clist,
401 static gint summary_cmp_by_size (GtkCList *clist,
404 static gint summary_cmp_by_date (GtkCList *clist,
407 static gint summary_cmp_by_thread_date (GtkCList *clist,
410 static gint summary_cmp_by_from (GtkCList *clist,
413 static gint summary_cmp_by_simplified_subject
417 static gint summary_cmp_by_score (GtkCList *clist,
420 static gint summary_cmp_by_label (GtkCList *clist,
423 static gint summary_cmp_by_to (GtkCList *clist,
426 static gint summary_cmp_by_subject (GtkCList *clist,
429 static gint summary_cmp_by_locked (GtkCList *clist,
432 static gint summary_cmp_by_tags (GtkCList *clist,
436 static void quicksearch_execute_cb (QuickSearch *quicksearch,
438 static void tog_searchbar_cb (GtkWidget *w,
441 static void summary_find_answers (SummaryView *summaryview,
444 static gboolean summary_update_msg (gpointer source, gpointer data);
445 static gboolean summary_update_folder_item_hook(gpointer source, gpointer data);
446 static gboolean summary_update_folder_hook(gpointer source, gpointer data);
447 static void summary_set_colorlabel_color (GtkCTree *ctree,
450 static void summary_thread_build(SummaryView *summaryview);
452 GtkTargetEntry summary_drag_types[2] =
454 {"text/uri-list", 0, TARGET_MAIL_URI_LIST},
455 {"claws-mail/internal", GTK_TARGET_SAME_APP, TARGET_DUMMY}
458 static GtkItemFactoryEntry summary_popup_entries[] =
460 {N_("/_Reply"), "<control>R", summary_reply_cb, COMPOSE_REPLY, NULL},
462 {N_("/Repl_y to"), NULL, NULL, 0, "<Branch>"},
463 {N_("/Repl_y to/_all"), "<shift><control>R", summary_reply_cb, COMPOSE_REPLY_TO_ALL, NULL},
464 {N_("/Repl_y to/_sender"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_SENDER, NULL},
465 {N_("/Repl_y to/mailing _list"),
466 "<control>L", summary_reply_cb, COMPOSE_REPLY_TO_LIST, NULL},
467 {"/---", NULL, NULL, 0, "<Separator>"},
469 {N_("/_Forward"), "<control><alt>F", summary_reply_cb, COMPOSE_FORWARD_INLINE, NULL},
471 {N_("/For_ward as attachment"), NULL, summary_reply_cb, COMPOSE_FORWARD_AS_ATTACH, NULL},
472 {N_("/Redirect"), NULL, summary_reply_cb, COMPOSE_REDIRECT, NULL},
474 {"/---", NULL, NULL, 0, "<Separator>"},
475 {N_("/M_ove..."), "<control>O", summary_move_to, 0, NULL},
476 {N_("/_Copy..."), "<shift><control>O", summary_copy_to, 0, NULL},
477 {N_("/Move to _trash"), "<control>D", summary_delete_trash, 0, NULL},
479 {N_("/_Delete..."), NULL, summary_delete, 0, NULL},
481 {"/---", NULL, NULL, 0, "<Separator>"},
482 {N_("/_Mark"), NULL, NULL, 0, "<Branch>"},
483 {N_("/_Mark/_Mark"), NULL, summary_mark, 0, NULL},
484 {N_("/_Mark/_Unmark"), NULL, summary_unmark, 0, NULL},
485 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
486 {N_("/_Mark/Mark as unr_ead"), NULL, summary_mark_as_unread, 0, NULL},
487 {N_("/_Mark/Mark as rea_d"), NULL, summary_mark_as_read, 0, NULL},
488 {N_("/_Mark/Mark all read"), NULL, summary_mark_all_read, 0, NULL},
489 {N_("/_Mark/Ignore thread"), NULL, summary_ignore_thread, 0, NULL},
490 {N_("/_Mark/Unignore thread"), NULL, summary_unignore_thread, 0, NULL},
491 {N_("/_Mark/Watch thread"), NULL, summary_watch_thread, 0, NULL},
492 {N_("/_Mark/Unwatch thread"), NULL, summary_unwatch_thread, 0, NULL},
493 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
494 {N_("/_Mark/Mark as _spam"), NULL, summary_mark_as_spam, 1, NULL},
495 {N_("/_Mark/Mark as _ham"), NULL, summary_mark_as_spam, 0, NULL},
496 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
497 {N_("/_Mark/Lock"), NULL, summary_msgs_lock, 0, NULL},
498 {N_("/_Mark/Unlock"), NULL, summary_msgs_unlock, 0, NULL},
499 {N_("/Color la_bel"), NULL, NULL, 0, NULL},
500 {N_("/Ta_gs"), NULL, NULL, 0, NULL},
502 {"/---", NULL, NULL, 0, "<Separator>"},
504 {N_("/Add sender to address boo_k"),
505 NULL, summary_add_address_cb, 0, NULL},
507 {N_("/Create f_ilter rule"), NULL, NULL, 0, "<Branch>"},
508 {N_("/Create f_ilter rule/_Automatically"),
509 NULL, summary_create_filter_cb, FILTER_BY_AUTO, NULL},
510 {N_("/Create f_ilter rule/by _From"),
511 NULL, summary_create_filter_cb, FILTER_BY_FROM, NULL},
512 {N_("/Create f_ilter rule/by _To"),
513 NULL, summary_create_filter_cb, FILTER_BY_TO, NULL},
514 {N_("/Create f_ilter rule/by _Subject"),
515 NULL, summary_create_filter_cb, FILTER_BY_SUBJECT, NULL},
517 {N_("/Create processing rule"), NULL, NULL, 0, "<Branch>"},
518 {N_("/Create processing rule/_Automatically"),
519 NULL, summary_create_processing_cb, FILTER_BY_AUTO, NULL},
520 {N_("/Create processing rule/by _From"),
521 NULL, summary_create_processing_cb, FILTER_BY_FROM, NULL},
522 {N_("/Create processing rule/by _To"),
523 NULL, summary_create_processing_cb, FILTER_BY_TO, NULL},
524 {N_("/Create processing rule/by _Subject"),
525 NULL, summary_create_processing_cb, FILTER_BY_SUBJECT, NULL},
527 {"/---", NULL, NULL, 0, "<Separator>"},
528 {N_("/_View"), NULL, NULL, 0, "<Branch>"},
529 {N_("/_View/Open in new _window"),
530 "<control><alt>N", summary_open_msg, 0, NULL},
531 {N_("/_View/Message _source"), "<control>U", summary_view_source, 0, NULL},
533 {N_("/_View/All _headers"), "<control>H", summary_show_all_header_cb, 0, "<ToggleItem>"},
535 {"/---", NULL, NULL, 0, "<Separator>"},
536 {N_("/_Save as..."), "<control>S", summary_save_as, 0, NULL},
538 {N_("/_Print..."), "<control>P", summary_print, 0, NULL},
540 }; /* see also list in menu_connect_identical_items() in menu.c if this changes */
542 static const gchar *const col_label[N_SUMMARY_COLS] = {
544 N_("S"), /* S_COL_STATUS */
546 N_("Subject"), /* S_COL_SUBJECT */
547 N_("From"), /* S_COL_FROM */
548 N_("To"), /* S_COL_TO */
549 N_("Date"), /* S_COL_DATE */
550 N_("Size"), /* S_COL_SIZE */
551 N_("#"), /* S_COL_NUMBER */
552 N_("Score"), /* S_COL_SCORE */
553 "", /* S_COL_LOCKED */
554 N_("Tags"), /* S_COL_TAGS */
557 void summary_freeze(SummaryView *summaryview)
560 gtk_clist_freeze(GTK_CLIST(summaryview->ctree));
563 void summary_thaw(SummaryView *summaryview)
566 gtk_clist_thaw(GTK_CLIST(summaryview->ctree));
569 void summary_grab_focus(SummaryView *summaryview)
572 gtk_widget_grab_focus(summaryview->ctree);
575 GtkWidget *summary_get_main_widget(SummaryView *summaryview)
578 return summaryview->ctree;
583 #define START_LONG_OPERATION(summaryview,force_freeze) { \
584 summary_lock(summaryview); \
585 main_window_cursor_wait(summaryview->mainwin); \
586 if (force_freeze || sc_g_list_bigger(GTK_CLIST(summaryview->ctree)->selection, 1)) {\
588 summary_freeze(summaryview); \
590 folder_item_update_freeze(); \
592 hooks_unregister_hook(MSGINFO_UPDATE_HOOKLIST, \
593 summaryview->msginfo_update_callback_id); \
595 #define END_LONG_OPERATION(summaryview) { \
597 folder_item_update_thaw(); \
599 summary_thaw(summaryview); \
600 main_window_cursor_normal(summaryview->mainwin); \
601 summary_unlock(summaryview); \
602 summaryview->msginfo_update_callback_id = \
603 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST, \
604 summary_update_msg, (gpointer) summaryview); \
607 SummaryView *summary_create(void)
609 SummaryView *summaryview;
611 GtkWidget *scrolledwin;
616 GtkWidget *stat_box2;
617 GtkWidget *stat_vbox;
618 GtkWidget *statlabel_folder;
619 GtkWidget *statlabel_select;
620 GtkWidget *statlabel_msgs;
622 GtkWidget *toggle_eventbox;
624 GtkWidget *multiple_sel_togbtn;
626 GtkWidget *toggle_arrow;
627 GtkWidget *popupmenu;
628 GtkWidget *toggle_search;
629 GtkItemFactory *popupfactory;
631 QuickSearch *quicksearch;
634 debug_print("Creating summary view...\n");
635 summaryview = g_new0(SummaryView, 1);
637 #if !(GTK_CHECK_VERSION(2,12,0))
638 summaryview->tips = tips;
640 #define SUMMARY_VBOX_SPACING 3
641 vbox = gtk_vbox_new(FALSE, SUMMARY_VBOX_SPACING);
643 /* create status label */
644 hbox = gtk_hbox_new(FALSE, 0);
645 gtk_widget_show(hbox);
647 stat_vbox = gtk_vbox_new(FALSE, 0);
648 gtk_widget_show(stat_vbox);
650 stat_box = gtk_hbox_new(FALSE, 0);
651 gtk_widget_show(stat_box);
653 stat_box2 = gtk_hbox_new(FALSE, 0);
654 gtk_widget_show(stat_box2);
656 toggle_search = gtk_toggle_button_new();
657 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(toggle_search),
658 prefs_common.show_searchbar);
659 gtk_widget_show(toggle_search);
661 CLAWS_SET_TIP(toggle_search, _("Toggle quick search bar"));
663 gtk_box_pack_start(GTK_BOX(hbox), toggle_search, FALSE, FALSE, 2);
665 gtk_box_pack_start(GTK_BOX(hbox), stat_vbox, TRUE, TRUE, 0);
666 gtk_box_pack_start(GTK_BOX(stat_vbox), stat_box, TRUE, TRUE, 0);
667 gtk_box_pack_start(GTK_BOX(stat_vbox), stat_box2, TRUE, TRUE, 0);
669 hbox_l = gtk_hbox_new(FALSE, 0);
670 gtk_widget_show(hbox_l);
671 gtk_box_pack_start(GTK_BOX(stat_box), hbox_l, TRUE, TRUE, 0);
673 statlabel_folder = gtk_label_new("");
674 gtk_widget_show(statlabel_folder);
675 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_folder, FALSE, FALSE, 2);
676 statlabel_select = gtk_label_new("");
677 gtk_widget_show(statlabel_select);
678 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_select, FALSE, FALSE, 12);
680 /* toggle view button */
681 toggle_eventbox = gtk_event_box_new();
682 gtk_widget_show(toggle_eventbox);
684 gtk_box_pack_end(GTK_BOX(hbox), toggle_eventbox, FALSE, FALSE, 4);
686 toggle_arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
687 gtk_widget_show(toggle_arrow);
688 gtk_container_add(GTK_CONTAINER(toggle_eventbox), toggle_arrow);
689 g_signal_connect(G_OBJECT(toggle_eventbox), "button_press_event",
690 G_CALLBACK(summary_toggle_pressed),
694 multiple_sel_togbtn = gtk_toggle_button_new();
695 gtk_widget_show(multiple_sel_togbtn);
696 gtk_box_pack_end(GTK_BOX(hbox), multiple_sel_togbtn, FALSE, FALSE, 4);
697 CLAWS_SET_TIP(multiple_sel_togbtn,
698 _("Toggle multiple selection"));
699 g_signal_connect(G_OBJECT(multiple_sel_togbtn), "toggled",
700 G_CALLBACK(summary_toggle_multiple_pressed),
704 statlabel_msgs = gtk_label_new("");
705 gtk_widget_show(statlabel_msgs);
706 gtk_box_pack_end(GTK_BOX(stat_box), statlabel_msgs, FALSE, FALSE, 4);
708 hbox_spc = gtk_hbox_new(FALSE, 0);
709 gtk_widget_show(hbox_spc);
710 gtk_box_pack_end(GTK_BOX(hbox), hbox_spc, FALSE, FALSE, 6);
712 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
713 gtk_widget_show(scrolledwin);
714 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
715 GTK_POLICY_AUTOMATIC,
716 GTK_POLICY_AUTOMATIC);
717 summaryview->mainwidget_book = gtk_notebook_new();
718 gtk_notebook_set_show_tabs(GTK_NOTEBOOK(summaryview->mainwidget_book), FALSE);
719 gtk_notebook_set_show_border(GTK_NOTEBOOK(summaryview->mainwidget_book), FALSE);
721 gtk_container_add(GTK_CONTAINER(summaryview->mainwidget_book),
723 gtk_box_pack_start(GTK_BOX(vbox), summaryview->mainwidget_book, TRUE, TRUE, 0);
725 gtk_widget_set_size_request(vbox,
726 prefs_common.summaryview_width,
727 prefs_common.summaryview_height);
729 ctree = summary_ctree_create(summaryview);
730 gtk_widget_show(ctree);
732 gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
733 GTK_CLIST(ctree)->hadjustment);
734 gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
735 GTK_CLIST(ctree)->vadjustment);
736 gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
739 gtk_widget_show_all(stat_vbox);
740 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
743 quicksearch = quicksearch_new();
744 gtk_box_pack_start(GTK_BOX(vbox), quicksearch_get_widget(quicksearch), FALSE, FALSE, 0);
747 gtk_container_add(GTK_CONTAINER(summaryview->mainwidget_book),
749 gtk_box_pack_start(GTK_BOX(vbox), summaryview->mainwidget_book, TRUE, TRUE, 0);
751 quicksearch_set_execute_callback(quicksearch, quicksearch_execute_cb, summaryview);
753 g_signal_connect (G_OBJECT(toggle_search), "toggled",
754 G_CALLBACK(tog_searchbar_cb), summaryview);
756 /* create popup menu */
757 n_entries = sizeof(summary_popup_entries) /
758 sizeof(summary_popup_entries[0]);
759 popupmenu = menu_create_items(summary_popup_entries, n_entries,
760 "<SummaryView>", &popupfactory,
763 summaryview->vbox = vbox;
764 summaryview->scrolledwin = scrolledwin;
765 summaryview->ctree = ctree;
766 summaryview->hbox = hbox;
767 summaryview->hbox_l = hbox_l;
768 summaryview->hbox_spc = hbox_spc;
769 summaryview->stat_box = stat_box;
770 summaryview->stat_box2 = stat_box2;
771 summaryview->statlabel_folder = statlabel_folder;
772 summaryview->statlabel_select = statlabel_select;
773 summaryview->statlabel_msgs = statlabel_msgs;
774 summaryview->toggle_eventbox = toggle_eventbox;
775 summaryview->toggle_arrow = toggle_arrow;
777 summaryview->multiple_sel_togbtn = multiple_sel_togbtn;
779 summaryview->toggle_search = toggle_search;
780 summaryview->popupmenu = popupmenu;
781 summaryview->popupfactory = popupfactory;
782 summaryview->lock_count = 0;
783 summaryview->msginfo_update_callback_id =
784 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST, summary_update_msg, (gpointer) summaryview);
785 summaryview->folder_item_update_callback_id =
786 hooks_register_hook(FOLDER_ITEM_UPDATE_HOOKLIST,
787 summary_update_folder_item_hook,
788 (gpointer) summaryview);
789 summaryview->folder_update_callback_id =
790 hooks_register_hook(FOLDER_UPDATE_HOOKLIST,
791 summary_update_folder_hook,
792 (gpointer) summaryview);
794 summaryview->target_list = gtk_target_list_new(summary_drag_types, 2);
796 summaryview->quicksearch = quicksearch;
798 /* CLAWS: need this to get the SummaryView * from
800 g_object_set_data(G_OBJECT(ctree), "summaryview", (gpointer)summaryview);
802 gtk_widget_show_all(vbox);
804 gtk_widget_show(vbox);
806 if (prefs_common.show_searchbar)
807 quicksearch_show(quicksearch);
809 quicksearch_hide(quicksearch);
811 if (prefs_common.layout_mode == WIDE_MSGLIST_LAYOUT ||
812 prefs_common.layout_mode == SMALL_LAYOUT)
813 gtk_widget_hide(summaryview->toggle_eventbox);
818 void summary_relayout(SummaryView *summaryview)
820 gtk_widget_realize(summaryview->stat_box);
822 gtk_widget_ref(summaryview->hbox_l);
823 gtk_widget_ref(summaryview->statlabel_msgs);
825 gtkut_container_remove(GTK_CONTAINER(summaryview->hbox_l->parent), summaryview->hbox_l);
826 gtkut_container_remove(GTK_CONTAINER(summaryview->statlabel_msgs->parent), summaryview->statlabel_msgs);
828 switch (prefs_common.layout_mode) {
831 case WIDE_MSGLIST_LAYOUT:
832 gtk_box_pack_start(GTK_BOX(summaryview->stat_box), summaryview->hbox_l, TRUE, TRUE, 0);
833 gtk_box_pack_end(GTK_BOX(summaryview->stat_box), summaryview->statlabel_msgs, FALSE, FALSE, 4);
834 gtk_widget_show_all(summaryview->stat_box);
835 gtk_widget_show_all(summaryview->stat_box2);
836 if (prefs_common.layout_mode == WIDE_MSGLIST_LAYOUT ||
837 prefs_common.layout_mode == SMALL_LAYOUT)
838 gtk_widget_hide(summaryview->toggle_eventbox);
840 gtk_widget_show(summaryview->toggle_eventbox);
842 case VERTICAL_LAYOUT:
844 gtk_box_pack_start(GTK_BOX(summaryview->stat_box), summaryview->hbox_l, TRUE, TRUE, 0);
845 gtk_box_pack_start(GTK_BOX(summaryview->stat_box2), summaryview->statlabel_msgs, FALSE, FALSE, 4);
846 gtk_widget_show_all(summaryview->stat_box);
847 gtk_widget_show_all(summaryview->stat_box2);
848 if (prefs_common.layout_mode == SMALL_LAYOUT) {
849 gtk_widget_hide(summaryview->toggle_eventbox);
850 gtk_widget_hide(summaryview->statlabel_msgs);
852 gtk_widget_show(summaryview->toggle_eventbox);
853 gtk_widget_show(summaryview->statlabel_msgs);
858 summary_set_column_order(summaryview);
860 gtk_widget_unref(summaryview->hbox_l);
861 gtk_widget_unref(summaryview->statlabel_msgs);
862 quicksearch_relayout(summaryview->quicksearch);
863 if (prefs_common.show_searchbar)
864 quicksearch_show(summaryview->quicksearch);
866 quicksearch_hide(summaryview->quicksearch);
869 static void summary_set_fonts(SummaryView *summaryview)
871 PangoFontDescription *font_desc;
874 font_desc = pango_font_description_from_string(NORMAL_FONT);
876 gtk_widget_modify_font(summaryview->ctree, font_desc);
877 pango_font_description_free(font_desc);
881 bold_style = gtk_style_copy
882 (gtk_widget_get_style(summaryview->ctree));
884 if (prefs_common.derive_from_normal_font || !BOLD_FONT) {
885 font_desc = pango_font_description_from_string(NORMAL_FONT);
887 pango_font_description_free(bold_style->font_desc);
888 bold_style->font_desc = font_desc;
890 pango_font_description_set_weight
891 (bold_style->font_desc, PANGO_WEIGHT_BOLD);
893 font_desc = pango_font_description_from_string(BOLD_FONT);
895 pango_font_description_free(bold_style->font_desc);
896 bold_style->font_desc = font_desc;
899 bold_marked_style = gtk_style_copy(bold_style);
900 bold_marked_style->fg[GTK_STATE_NORMAL] =
901 summaryview->color_marked;
902 bold_deleted_style = gtk_style_copy(bold_style);
903 bold_deleted_style->fg[GTK_STATE_NORMAL] =
904 summaryview->color_dim;
907 if (prefs_common.derive_from_normal_font || !SMALL_FONT) {
908 font_desc = pango_font_description_new();
909 size = pango_font_description_get_size
910 (summaryview->ctree->style->font_desc);
911 pango_font_description_set_size(font_desc, size * PANGO_SCALE_SMALL);
913 font_desc = pango_font_description_from_string(SMALL_FONT);
916 gtk_widget_modify_font(summaryview->statlabel_folder, font_desc);
917 gtk_widget_modify_font(summaryview->statlabel_select, font_desc);
918 gtk_widget_modify_font(summaryview->statlabel_msgs, font_desc);
919 pango_font_description_free(font_desc);
924 static void summary_set_folder_pixmap(SummaryView *summaryview, StockPixmap icon)
927 if (!summaryview->folder_pixmap_eventbox) {
928 summaryview->folder_pixmap_eventbox = gtk_event_box_new();
929 gtk_widget_show(summaryview->folder_pixmap_eventbox);
930 gtk_box_pack_start(GTK_BOX(summaryview->hbox_l), summaryview->folder_pixmap_eventbox, FALSE, FALSE, 4);
931 gtk_box_reorder_child(GTK_BOX(summaryview->hbox_l), summaryview->folder_pixmap_eventbox, 0); /* search_toggle before */
932 g_signal_connect(G_OBJECT(summaryview->folder_pixmap_eventbox), "button_press_event",
933 G_CALLBACK(summary_folder_eventbox_pressed),
936 if (summaryview->folder_pixmap)
937 gtk_widget_destroy(summaryview->folder_pixmap);
939 pixmap = stock_pixmap_widget(summaryview->hbox_l, icon);
940 gtk_container_add(GTK_CONTAINER(summaryview->folder_pixmap_eventbox), pixmap);
941 gtk_widget_show(pixmap);
942 summaryview->folder_pixmap = pixmap;
945 void summary_init(SummaryView *summaryview)
949 gtk_widget_realize(summaryview->ctree);
950 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_MARK,
951 &markxpm, &markxpmmask);
952 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_DELETED,
953 &deletedxpm, &deletedxpmmask);
954 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_NEW,
955 &newxpm, &newxpmmask);
956 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_UNREAD,
957 &unreadxpm, &unreadxpmmask);
958 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_REPLIED,
959 &repliedxpm, &repliedxpmmask);
960 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_FORWARDED,
961 &forwardedxpm, &forwardedxpmmask);
962 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP,
963 &clipxpm, &clipxpmmask);
964 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_LOCKED,
965 &lockedxpm, &lockedxpmmask);
966 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_IGNORETHREAD,
967 &ignorethreadxpm, &ignorethreadxpmmask);
968 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_WATCHTHREAD,
969 &watchthreadxpm, &watchthreadxpmmask);
970 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP_KEY,
971 &clipkeyxpm, &clipkeyxpmmask);
972 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_KEY,
973 &keyxpm, &keyxpmmask);
974 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_GPG_SIGNED,
975 &gpgsignedxpm, &gpgsignedxpmmask);
976 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP_GPG_SIGNED,
977 &clipgpgsignedxpm, &clipgpgsignedxpmmask);
978 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_SPAM,
979 &spamxpm, &spamxpmmask);
980 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_MOVED,
981 &movedxpm, &movedxpmmask);
982 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_COPIED,
983 &copiedxpm, &copiedxpmmask);
985 summary_set_fonts(summaryview);
987 summary_set_folder_pixmap(summaryview, STOCK_PIXMAP_DIR_OPEN);
989 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_QUICKSEARCH);
990 gtk_container_add (GTK_CONTAINER(summaryview->toggle_search), pixmap);
991 gtk_widget_show(pixmap);
992 summaryview->quick_search_pixmap = pixmap;
995 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_SELECTION);
996 gtk_container_add(GTK_CONTAINER(summaryview->multiple_sel_togbtn), pixmap);
997 gtk_widget_show(pixmap);
998 summaryview->multiple_sel_image = pixmap;
1001 /* Init summaryview prefs */
1002 summaryview->sort_key = SORT_BY_NONE;
1003 summaryview->sort_type = SORT_ASCENDING;
1005 /* Init summaryview extra data */
1006 summaryview->simplify_subject_preg = NULL;
1008 summary_clear_list(summaryview);
1009 summary_set_column_titles(summaryview);
1010 summary_colorlabel_menu_create(summaryview, FALSE);
1011 summary_tags_menu_create(summaryview, FALSE);
1012 main_create_mailing_list_menu (summaryview->mainwin, NULL);
1013 summary_set_menu_sensitive(summaryview);
1017 #define CURRENTLY_DISPLAYED(m) \
1018 ( (m->msgnum == displayed_msgnum) \
1019 && (!g_ascii_strcasecmp(m->folder->name,item->name)) )
1021 #define FOLDER_SHOWS_TO_HDR(i) \
1022 ( i && (folder_has_parent_of_type(i, F_OUTBOX) \
1023 || folder_has_parent_of_type(i, F_DRAFT) \
1024 || folder_has_parent_of_type(i, F_QUEUE)) )
1026 static void summary_switch_from_to(SummaryView *summaryview, FolderItem *item)
1028 gboolean show_from = FALSE, show_to = FALSE;
1029 gboolean showing_from = FALSE, showing_to = FALSE;
1030 gint from_pos = 0, to_pos = 0;
1031 SummaryColumnState *col_state = summaryview->col_state;
1032 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1034 if (!item || (prefs_common.layout_mode == VERTICAL_LAYOUT && prefs_common.two_line_vert) )
1036 if (FOLDER_SHOWS_TO_HDR(item))
1041 from_pos = summaryview->col_pos[S_COL_FROM];
1042 to_pos = summaryview->col_pos[S_COL_TO];
1043 showing_from = col_state[from_pos].visible;
1044 showing_to = col_state[to_pos].visible;
1046 if (showing_from && showing_to) {
1047 debug_print("showing both\n");
1051 if (!showing_from && !showing_to) {
1052 debug_print("showing none\n");
1056 debug_print("showing %s %s, must show %s %s\n",
1057 showing_from?"From":"",
1059 show_from?"From":"",
1062 if (showing_from == show_from && showing_to == show_to)
1064 /* else we'll switch both */
1066 debug_print("switching columns\n");
1067 col_state[from_pos].type = S_COL_TO;
1068 col_state[from_pos].visible = show_to;
1070 col_state[to_pos].type = S_COL_FROM;
1071 col_state[to_pos].visible = show_from;
1073 summaryview->col_pos[S_COL_TO] = from_pos;
1074 summaryview->col_pos[S_COL_FROM] = to_pos;
1076 gtk_clist_set_column_visibility
1077 (GTK_CLIST(ctree), from_pos, col_state[from_pos].visible);
1078 gtk_clist_set_column_visibility
1079 (GTK_CLIST(ctree), to_pos, col_state[to_pos].visible);
1081 summary_set_column_titles(summaryview);
1084 static gboolean summaryview_quicksearch_recurse(gpointer data)
1086 SummaryView *summaryview = (SummaryView *)data;
1087 main_window_cursor_wait(summaryview->mainwin);
1088 quicksearch_reset_cur_folder_item(summaryview->quicksearch);
1089 quicksearch_search_subfolders(summaryview->quicksearch,
1090 summaryview->folderview,
1091 summaryview->folder_item);
1092 main_window_cursor_normal(summaryview->mainwin);
1096 static gboolean summary_check_consistency(FolderItem *item, GSList *mlist)
1098 int u = 0, n = 0, m = 0, t = 0, r = 0, f = 0, l = 0, i = 0, w = 0;
1101 for(cur = mlist ; cur != NULL && cur->data != NULL ; cur = g_slist_next(cur)) {
1102 MsgInfo * msginfo = (MsgInfo *) cur->data;
1104 if (MSG_IS_NEW(msginfo->flags))
1106 if (MSG_IS_UNREAD(msginfo->flags))
1108 if (MSG_IS_MARKED(msginfo->flags))
1110 if (MSG_IS_REPLIED(msginfo->flags))
1112 if (MSG_IS_FORWARDED(msginfo->flags))
1114 if (MSG_IS_LOCKED(msginfo->flags))
1116 if (MSG_IS_IGNORE_THREAD(msginfo->flags))
1118 if (MSG_IS_WATCH_THREAD(msginfo->flags))
1121 if (t != item->total_msgs
1122 || n != item->new_msgs
1123 || u != item->unread_msgs
1124 || m != item->marked_msgs
1125 || r != item->replied_msgs
1126 || f != item->forwarded_msgs
1127 || l != item->locked_msgs
1128 || i != item->ignored_msgs
1129 || w != item->watched_msgs
1130 || (m == 0 && item->unreadmarked_msgs != 0)
1131 || item->unreadmarked_msgs < 0) {
1132 debug_print("Inconsistency\n");
1133 folder_item_scan_full(item, FALSE);
1141 gboolean summary_show(SummaryView *summaryview, FolderItem *item)
1143 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1144 GtkCTreeNode *node = NULL;
1145 GSList *mlist = NULL;
1147 gboolean is_refresh;
1148 guint selected_msgnum = 0;
1149 guint displayed_msgnum = 0;
1152 gboolean hidden_removed = FALSE;
1154 if (summary_is_locked(summaryview)) return FALSE;
1156 if (!summaryview->mainwin)
1159 summaryview->last_displayed = NULL;
1160 summary_switch_from_to(summaryview, item);
1163 summary_lock(summaryview);
1167 is_refresh = (item == summaryview->folder_item) ? TRUE : FALSE;
1169 if (item && item->folder->klass->item_opened) {
1170 item->folder->klass->item_opened(item);
1174 main_create_mailing_list_menu (summaryview->mainwin, NULL);
1175 if (prefs_common.layout_mode == SMALL_LAYOUT) {
1177 mainwindow_enter_folder(summaryview->mainwin);
1178 gtk_widget_grab_focus(summaryview->ctree);
1182 if (!prefs_common.summary_quicksearch_sticky
1183 && (!prefs_common.summary_quicksearch_recurse
1184 || !quicksearch_is_active(summaryview->quicksearch)
1185 || (item && !quicksearch_is_in_subfolder(summaryview->quicksearch, item)))
1186 && !quicksearch_is_running(summaryview->quicksearch)
1188 quicksearch_set(summaryview->quicksearch, prefs_common.summary_quicksearch_type, "");
1191 /* STATUSBAR_POP(summaryview->mainwin); */
1194 selected_msgnum = summary_get_msgnum(summaryview,
1195 summaryview->selected);
1196 displayed_msgnum = summary_get_msgnum(summaryview,
1197 summaryview->displayed);
1200 /* process the marks if any */
1201 if (summaryview->mainwin->lock_count == 0 &&
1202 (summaryview->moved > 0 || summaryview->copied > 0)) {
1204 gboolean changed = FALSE;
1206 val = alertpanel(_("Process mark"),
1207 _("Some marks are left. Process them?"),
1208 GTK_STOCK_NO, GTK_STOCK_YES, GTK_STOCK_CANCEL);
1209 if (G_ALERTALTERNATE == val) {
1210 summary_unlock(summaryview);
1211 summary_execute(summaryview);
1212 summary_lock(summaryview);
1214 } else if (G_ALERTDEFAULT == val) {
1217 summary_unlock(summaryview);
1222 if (changed || !quicksearch_is_active(summaryview->quicksearch))
1223 folder_update_op_count();
1226 summary_freeze(summaryview);
1228 summary_clear_list(summaryview);
1231 if (!item || !item->path || !folder_item_parent(item) || item->no_select) {
1233 debug_print("empty folder (%p %s %p %d)\n\n",
1235 (item && item->path)?item->path:"(null)",
1236 item?folder_item_parent(item):0x0,
1237 item?item->no_select:FALSE);
1238 summary_set_hide_read_msgs_menu(summaryview, FALSE);
1239 summary_clear_all(summaryview);
1240 summaryview->folder_item = item;
1241 summary_thaw(summaryview);
1242 summary_unlock(summaryview);
1244 if (item && quicksearch_is_running(summaryview->quicksearch)) {
1245 main_window_cursor_wait(summaryview->mainwin);
1246 quicksearch_reset_cur_folder_item(summaryview->quicksearch);
1247 if (quicksearch_is_active(summaryview->quicksearch))
1248 quicksearch_search_subfolders(summaryview->quicksearch,
1249 summaryview->folderview,
1250 summaryview->folder_item);
1251 main_window_cursor_normal(summaryview->mainwin);
1259 messageview_clear(summaryview->messageview);
1261 summaryview->folder_item = item;
1262 item->opened = TRUE;
1264 buf = g_strdup_printf(_("Scanning folder (%s)..."), item->path);
1265 debug_print("%s\n", buf);
1266 STATUSBAR_PUSH(summaryview->mainwin, buf);
1269 main_window_cursor_wait(summaryview->mainwin);
1271 mlist = folder_item_get_msg_list(item);
1273 if (!summary_check_consistency(item, mlist)) {
1274 debug_print("reloading due to inconsistency\n");
1275 procmsg_msg_list_free(mlist);
1276 mlist = folder_item_get_msg_list(item);
1279 if (summaryview->folder_item->hide_read_msgs &&
1280 quicksearch_is_active(summaryview->quicksearch) == FALSE) {
1283 summary_set_hide_read_msgs_menu(summaryview, TRUE);
1285 for(cur = mlist ; cur != NULL && cur->data != NULL ; cur = g_slist_next(cur)) {
1286 MsgInfo * msginfo = (MsgInfo *) cur->data;
1288 if (!msginfo->hidden) {
1289 if (MSG_IS_UNREAD(msginfo->flags) &&
1290 !MSG_IS_IGNORE_THREAD(msginfo->flags))
1291 not_killed = g_slist_prepend(not_killed, msginfo);
1292 else if (MSG_IS_MARKED(msginfo->flags) ||
1293 MSG_IS_LOCKED(msginfo->flags))
1294 not_killed = g_slist_prepend(not_killed, msginfo);
1295 else if (is_refresh &&
1296 (msginfo->msgnum == selected_msgnum ||
1297 msginfo->msgnum == displayed_msgnum))
1298 not_killed = g_slist_prepend(not_killed, msginfo);
1300 procmsg_msginfo_free(msginfo);
1302 procmsg_msginfo_free(msginfo);
1304 hidden_removed = TRUE;
1305 g_slist_free(mlist);
1308 summary_set_hide_read_msgs_menu(summaryview, FALSE);
1311 if (quicksearch_is_active(summaryview->quicksearch)) {
1313 gint interval = quicksearch_is_fast(summaryview->quicksearch) ? 5000:100;
1314 START_TIMING("quicksearch");
1315 gint num = 0, total = summaryview->folder_item->total_msgs;
1316 statusbar_print_all(_("Searching in %s... \n"),
1317 summaryview->folder_item->path ?
1318 summaryview->folder_item->path : "(null)");
1320 folder_item_update_freeze();
1321 for (cur = mlist ; cur != NULL && cur->data != NULL ; cur = g_slist_next(cur)) {
1322 MsgInfo * msginfo = (MsgInfo *) cur->data;
1324 statusbar_progress_all(num++,total, interval);
1326 if (!msginfo->hidden && quicksearch_match(summaryview->quicksearch, msginfo))
1327 not_killed = g_slist_prepend(not_killed, msginfo);
1329 procmsg_msginfo_free(msginfo);
1330 if (num % interval == 0)
1332 if (!quicksearch_is_active(summaryview->quicksearch)) {
1336 folder_item_update_thaw();
1337 statusbar_progress_all(0,0,0);
1338 statusbar_pop_all();
1340 hidden_removed = TRUE;
1341 if (!quicksearch_is_active(summaryview->quicksearch)) {
1342 debug_print("search cancelled!\n");
1343 summary_thaw(summaryview);
1344 STATUSBAR_POP(summaryview->mainwin);
1345 main_window_cursor_normal(summaryview->mainwin);
1346 summary_unlock(summaryview);
1348 summary_show(summaryview, summaryview->folder_item);
1352 g_slist_free(mlist);
1357 if (!hidden_removed) {
1358 START_TIMING("removing hidden");
1360 for(cur = mlist ; cur != NULL && cur->data != NULL ; cur = g_slist_next(cur)) {
1361 MsgInfo * msginfo = (MsgInfo *) cur->data;
1363 if (!msginfo->hidden)
1364 not_killed = g_slist_prepend(not_killed, msginfo);
1366 procmsg_msginfo_free(msginfo);
1368 g_slist_free(mlist);
1373 STATUSBAR_POP(summaryview->mainwin);
1375 /* set ctree and hash table from the msginfo list, and
1376 create the thread */
1377 summary_set_ctree_from_list(summaryview, mlist);
1379 g_slist_free(mlist);
1381 if (quicksearch_is_active(summaryview->quicksearch) &&
1382 quicksearch_is_running(summaryview->quicksearch)) {
1383 /* only scan subfolders when quicksearch changed,
1384 * not when search is the same and folder changed */
1385 g_timeout_add(100, summaryview_quicksearch_recurse, summaryview);
1389 if (!quicksearch_is_in_typing(summaryview->quicksearch)) {
1390 summaryview->last_displayed = summaryview->displayed;
1391 summaryview->displayed =
1392 summary_find_msg_by_msgnum(summaryview,
1394 if (!summaryview->displayed)
1395 messageview_clear(summaryview->messageview);
1396 summary_unlock(summaryview);
1397 summary_select_by_msgnum(summaryview, selected_msgnum);
1398 summary_lock(summaryview);
1399 if (!summaryview->selected) {
1400 /* no selected message - select first unread
1401 message, but do not display it */
1402 node = summary_find_next_flagged_msg(summaryview, NULL,
1404 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL)
1405 node = gtk_ctree_node_nth
1407 item->sort_type == SORT_DESCENDING
1408 ? 0 : GTK_CLIST(ctree)->rows - 1);
1409 summary_unlock(summaryview);
1410 summary_select_node(summaryview, node, FALSE, TRUE);
1411 summary_lock(summaryview);
1414 /* just select first/last */
1415 if (GTK_CLIST(ctree)->row_list != NULL)
1416 node = gtk_ctree_node_nth
1418 item->sort_type == SORT_DESCENDING
1419 ? 0 : GTK_CLIST(ctree)->rows - 1);
1420 gtk_sctree_select(GTK_SCTREE(ctree), node);
1421 summaryview->selected = node;
1422 gtk_ctree_node_moveto(ctree, node, 0, 0.5, 0);
1425 /* backward compat */
1427 gboolean set = FALSE, stop = FALSE;
1428 for (i = 0; i < 6; i++) {
1429 EntryAction act = prefs_common.summary_select_prio[i];
1431 if (act != ACTION_UNSET) {
1437 prefs_summary_open_set_defaults();
1439 for (i = 0; i < 6 && node == NULL; i++) {
1440 EntryAction act = prefs_common.summary_select_prio[i];
1444 node = summary_find_next_flagged_msg(summaryview, NULL,
1448 node = summary_find_next_flagged_msg(summaryview, NULL,
1452 node = summary_find_next_flagged_msg(summaryview, NULL,
1455 case ACTION_LAST_OPENED:
1456 if (summaryview->folder_item) {
1457 node = summary_find_msg_by_msgnum(summaryview,
1458 summaryview->folder_item->last_seen);
1461 case ACTION_LAST_LIST:
1462 if (GTK_CLIST(ctree)->row_list != NULL) {
1463 node = gtk_ctree_node_nth
1465 item->sort_type == SORT_DESCENDING
1466 ? 0 : GTK_CLIST(ctree)->rows - 1);
1469 case ACTION_FIRST_LIST:
1470 if (GTK_CLIST(ctree)->row_list != NULL) {
1471 node = gtk_ctree_node_nth
1473 item->sort_type == SORT_ASCENDING
1474 ? 0 : GTK_CLIST(ctree)->rows - 1);
1477 case ACTION_NOTHING:
1488 summary_unlock(summaryview);
1490 gboolean show = (prefs_common.always_show_msg == OPENMSG_ALWAYS) ||
1491 (prefs_common.always_show_msg == OPENMSG_WHEN_VIEW_VISIBLE &&
1492 messageview_is_visible(summaryview->messageview));
1493 summary_select_node(summaryview, node, show, TRUE);
1495 summary_lock(summaryview);
1498 summary_status_show(summaryview);
1499 summary_set_menu_sensitive(summaryview);
1500 toolbar_main_set_sensitive(summaryview->mainwin);
1502 summary_thaw(summaryview);
1504 STATUSBAR_PUSH(summaryview->mainwin, _("Done."));
1505 STATUSBAR_POP(summaryview->mainwin);
1506 main_window_cursor_normal(summaryview->mainwin);
1507 summary_unlock(summaryview);
1513 #undef CURRENTLY_DISPLAYED
1516 void summary_clear_list(SummaryView *summaryview)
1518 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1521 summary_freeze(summaryview);
1523 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree),
1524 NULL, summary_free_msginfo_func, NULL);
1526 if (summaryview->folder_item) {
1527 summaryview->folder_item->opened = FALSE;
1528 summaryview->folder_item = NULL;
1531 summaryview->display_msg = FALSE;
1533 summaryview->selected = NULL;
1534 summaryview->displayed = NULL;
1535 summaryview->total_size = 0;
1536 summaryview->deleted = summaryview->moved = 0;
1537 summaryview->copied = 0;
1538 if (summaryview->msgid_table) {
1539 g_hash_table_destroy(summaryview->msgid_table);
1540 summaryview->msgid_table = NULL;
1542 if (summaryview->subject_table) {
1543 g_hash_table_destroy(summaryview->subject_table);
1544 summaryview->subject_table = NULL;
1546 summaryview->mlist = NULL;
1548 gtk_clist_clear(clist);
1549 if (summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
1550 optimal_width = gtk_clist_optimal_column_width
1551 (clist, summaryview->col_pos[S_COL_SUBJECT]);
1552 gtk_clist_set_column_width
1553 (clist, summaryview->col_pos[S_COL_SUBJECT],
1557 summary_thaw(summaryview);
1560 void summary_clear_all(SummaryView *summaryview)
1562 messageview_clear(summaryview->messageview);
1563 summary_clear_list(summaryview);
1564 summary_set_menu_sensitive(summaryview);
1565 toolbar_main_set_sensitive(summaryview->mainwin);
1566 summary_status_show(summaryview);
1569 void summary_lock(SummaryView *summaryview)
1571 summaryview->lock_count++;
1574 void summary_unlock(SummaryView *summaryview)
1576 if (summaryview->lock_count)
1577 summaryview->lock_count--;
1580 gboolean summary_is_locked(SummaryView *summaryview)
1582 return summaryview->lock_count > 0;
1585 SummarySelection summary_get_selection_type(SummaryView *summaryview)
1587 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1588 SummarySelection selection;
1590 if (!clist->row_list)
1591 selection = SUMMARY_NONE;
1592 else if (!clist->selection)
1593 selection = SUMMARY_SELECTED_NONE;
1594 else if (!clist->selection->next)
1595 selection = SUMMARY_SELECTED_SINGLE;
1597 selection = SUMMARY_SELECTED_MULTIPLE;
1603 *\return MsgInfo * Selected message if there's one selected;
1604 * if multiple selected, or none, return NULL.
1606 MsgInfo *summary_get_selected_msg(SummaryView *summaryview)
1608 /* summaryview->selected may be valid when multiple
1609 * messages were selected */
1610 GList *sellist = GTK_CLIST(summaryview->ctree)->selection;
1612 if (sellist == NULL || sellist->next)
1615 return GTKUT_CTREE_NODE_GET_ROW_DATA(sellist->data);
1618 GSList *summary_get_selected_msg_list(SummaryView *summaryview)
1620 GSList *mlist = NULL;
1624 for (cur = GTK_CLIST(summaryview->ctree)->selection; cur != NULL && cur->data != NULL;
1626 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(cur->data);
1627 mlist = g_slist_prepend(mlist, msginfo);
1630 mlist = g_slist_reverse(mlist);
1635 void summary_set_menu_sensitive(SummaryView *summaryview)
1637 GtkItemFactory *ifactory = summaryview->popupfactory;
1638 SensitiveCond state;
1640 #ifndef GENERIC_UMPC
1641 GtkWidget *menuitem;
1645 static const struct {
1649 {"/Reply" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1650 #ifndef GENERIC_UMPC
1651 {"/Reply to" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1652 {"/Reply to/all" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1653 {"/Reply to/sender" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1654 {"/Reply to/mailing list" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1657 {"/Forward" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1658 #ifndef GENERIC_UMPC
1659 {"/Forward as attachment" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1660 {"/Redirect" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1663 {"/Move..." , M_TARGET_EXIST|M_ALLOW_DELETE|M_NOT_NEWS},
1664 {"/Copy..." , M_TARGET_EXIST|M_EXEC},
1665 {"/Move to trash" , M_TARGET_EXIST|M_ALLOW_DELETE|M_NOT_NEWS},
1666 #ifndef GENERIC_UMPC
1667 {"/Delete..." , M_TARGET_EXIST|M_ALLOW_DELETE},
1670 {"/Mark" , M_TARGET_EXIST},
1671 {"/Mark/Mark" , M_TARGET_EXIST},
1672 {"/Mark/Unmark" , M_TARGET_EXIST},
1673 {"/Mark/Mark as unread" , M_TARGET_EXIST},
1674 {"/Mark/Mark all read" , M_TARGET_EXIST},
1675 {"/Mark/Ignore thread" , M_TARGET_EXIST},
1676 {"/Mark/Unignore thread" , M_TARGET_EXIST},
1677 {"/Mark/Lock" , M_TARGET_EXIST},
1678 {"/Mark/Unlock" , M_TARGET_EXIST},
1679 {"/Mark/Mark as spam" , M_TARGET_EXIST|M_CAN_LEARN_SPAM},
1680 {"/Mark/Mark as ham" , M_TARGET_EXIST|M_CAN_LEARN_SPAM},
1681 {"/Color label" , M_TARGET_EXIST},
1682 {"/Tags" , M_TARGET_EXIST},
1684 #ifndef GENERIC_UMPC
1685 {"/Add sender to address book" , M_SINGLE_TARGET_EXIST},
1687 {"/Create filter rule" , M_SINGLE_TARGET_EXIST|M_UNLOCKED},
1688 #ifndef GENERIC_UMPC
1689 {"/Create processing rule" , M_SINGLE_TARGET_EXIST|M_UNLOCKED},
1692 {"/View" , M_SINGLE_TARGET_EXIST},
1693 {"/View/Open in new window" , M_SINGLE_TARGET_EXIST},
1694 {"/View/Message source" , M_SINGLE_TARGET_EXIST},
1695 #ifndef GENERIC_UMPC
1696 {"/View/All headers" , M_SINGLE_TARGET_EXIST},
1698 {"/Save as..." , M_TARGET_EXIST},
1699 #ifndef GENERIC_UMPC
1700 {"/Print..." , M_TARGET_EXIST},
1705 main_window_set_menu_sensitive(summaryview->mainwin);
1707 state = main_window_get_current_state(summaryview->mainwin);
1709 for (i = 0; entry[i].entry != NULL; i++) {
1710 sensitive = ((entry[i].cond & state) == entry[i].cond);
1711 menu_set_sensitive(ifactory, entry[i].entry, sensitive);
1715 summary_lock(summaryview);
1716 #ifndef GENERIC_UMPC
1717 menuitem = gtk_item_factory_get_widget(ifactory, "/View/All headers");
1718 if (summaryview->messageview
1719 && summaryview->messageview->mimeview
1720 && summaryview->messageview->mimeview->textview)
1721 gtk_check_menu_item_set_active
1722 (GTK_CHECK_MENU_ITEM(menuitem),
1723 summaryview->messageview->mimeview->textview->show_all_headers);
1725 summary_unlock(summaryview);
1728 void summary_select_prev_unread(SummaryView *summaryview)
1731 gboolean skip_cur = FALSE;
1733 if (summaryview->displayed
1734 && summaryview->selected == summaryview->displayed) {
1735 debug_print("skipping current\n");
1739 node = summary_find_prev_flagged_msg
1740 (summaryview, summaryview->selected, MSG_UNREAD, skip_cur);
1742 if (!node || node == summaryview->selected) {
1745 switch (prefs_common.next_unread_msg_dialog) {
1746 case NEXTUNREADMSGDIALOG_ALWAYS:
1747 val = alertpanel(_("No more unread messages"),
1748 _("No unread message found. "
1749 "Search from the end?"),
1750 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1752 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1753 val = G_ALERTALTERNATE;
1755 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1756 val = !G_ALERTALTERNATE;
1760 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1762 if (val != G_ALERTALTERNATE) return;
1763 node = summary_find_prev_flagged_msg(summaryview, NULL,
1768 alertpanel_notice(_("No unread messages."));
1770 summary_select_node(summaryview, node, TRUE, FALSE);
1773 void summary_select_next_unread(SummaryView *summaryview)
1775 GtkCTreeNode *node = summaryview->selected;
1776 gboolean skip_cur = FALSE;
1778 if (summaryview->displayed
1779 && summaryview->selected == summaryview->displayed) {
1780 debug_print("skipping cur (%p %p)\n",
1781 summaryview->displayed, summaryview->selected);
1786 node = summary_find_next_flagged_msg
1787 (summaryview, node, MSG_UNREAD, skip_cur);
1790 summary_select_node(summaryview, node, TRUE, FALSE);
1792 node = summary_find_next_flagged_msg
1793 (summaryview, NULL, MSG_UNREAD, FALSE);
1794 if (node == NULL || node == summaryview->selected) {
1797 switch (prefs_common.next_unread_msg_dialog) {
1798 case NEXTUNREADMSGDIALOG_ALWAYS:
1799 val = alertpanel(_("No more unread messages"),
1800 _("No unread message found. "
1801 "Go to next folder?"),
1802 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1804 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1805 val = G_ALERTALTERNATE;
1807 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1812 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1815 if (val == G_ALERTALTERNATE) {
1816 folderview_select_next_unread(summaryview->folderview, TRUE);
1822 summary_select_node(summaryview, node, TRUE, FALSE);
1827 void summary_select_prev_new(SummaryView *summaryview)
1830 gboolean skip_cur = FALSE;
1832 if (summaryview->displayed
1833 && summaryview->selected == summaryview->displayed) {
1834 debug_print("skipping current\n");
1838 node = summary_find_prev_flagged_msg
1839 (summaryview, summaryview->selected, MSG_NEW, skip_cur);
1841 if (!node || node == summaryview->selected) {
1844 switch (prefs_common.next_unread_msg_dialog) {
1845 case NEXTUNREADMSGDIALOG_ALWAYS:
1846 val = alertpanel(_("No more new messages"),
1847 _("No new message found. "
1848 "Search from the end?"),
1849 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1851 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1852 val = G_ALERTALTERNATE;
1854 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1855 val = !G_ALERTALTERNATE;
1859 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1861 if (val != G_ALERTALTERNATE) return;
1862 node = summary_find_prev_flagged_msg(summaryview, NULL,
1867 alertpanel_notice(_("No new messages."));
1869 summary_select_node(summaryview, node, TRUE, FALSE);
1872 void summary_select_next_new(SummaryView *summaryview)
1874 GtkCTreeNode *node = summaryview->selected;
1875 gboolean skip_cur = FALSE;
1877 if (summaryview->displayed
1878 && summaryview->selected == summaryview->displayed) {
1879 debug_print("skipping cur (%p %p)\n",
1880 summaryview->displayed, summaryview->selected);
1885 node = summary_find_next_flagged_msg
1886 (summaryview, node, MSG_NEW, skip_cur);
1889 summary_select_node(summaryview, node, TRUE, FALSE);
1891 node = summary_find_next_flagged_msg
1892 (summaryview, NULL, MSG_NEW, FALSE);
1893 if (node == NULL || node == summaryview->selected) {
1896 switch (prefs_common.next_unread_msg_dialog) {
1897 case NEXTUNREADMSGDIALOG_ALWAYS:
1898 val = alertpanel(_("No more new messages"),
1899 _("No new message found. "
1900 "Go to next folder?"),
1901 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1903 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1904 val = G_ALERTALTERNATE;
1906 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1911 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1914 if (val == G_ALERTALTERNATE) {
1915 folderview_select_next_new(summaryview->folderview);
1921 summary_select_node(summaryview, node, TRUE, FALSE);
1926 void summary_select_prev_marked(SummaryView *summaryview)
1930 node = summary_find_prev_flagged_msg
1931 (summaryview, summaryview->selected, MSG_MARKED, TRUE);
1936 val = alertpanel(_("No more marked messages"),
1937 _("No marked message found. "
1938 "Search from the end?"),
1939 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1940 if (val != G_ALERTALTERNATE) return;
1941 node = summary_find_prev_flagged_msg(summaryview, NULL,
1946 alertpanel_notice(_("No marked messages."));
1948 summary_select_node(summaryview, node, TRUE, FALSE);
1951 void summary_select_next_marked(SummaryView *summaryview)
1953 GtkCTreeNode *node = summaryview->selected;
1954 gboolean skip_cur = FALSE;
1956 if (summaryview->displayed
1957 && summaryview->selected == summaryview->displayed) {
1958 debug_print("skipping cur (%p %p)\n",
1959 summaryview->displayed, summaryview->selected);
1964 node = summary_find_next_flagged_msg
1965 (summaryview, node, MSG_MARKED, skip_cur);
1968 summary_select_node(summaryview, node, TRUE, FALSE);
1970 node = summary_find_next_flagged_msg
1971 (summaryview, NULL, MSG_MARKED, FALSE);
1972 if (node == NULL || node == summaryview->selected) {
1975 switch (prefs_common.next_unread_msg_dialog) {
1976 case NEXTUNREADMSGDIALOG_ALWAYS:
1977 val = alertpanel(_("No more marked messages"),
1978 _("No marked message found. "
1979 "Go to next folder?"),
1980 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
1982 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1983 val = G_ALERTALTERNATE;
1985 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1990 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1993 if (val == G_ALERTALTERNATE) {
1994 folderview_select_next_marked(summaryview->folderview);
2000 summary_select_node(summaryview, node, TRUE, FALSE);
2005 void summary_select_prev_labeled(SummaryView *summaryview)
2009 node = summary_find_prev_flagged_msg
2010 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
2015 val = alertpanel(_("No more labeled messages"),
2016 _("No labeled message found. "
2017 "Search from the end?"),
2018 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
2019 if (val != G_ALERTALTERNATE) return;
2020 node = summary_find_prev_flagged_msg(summaryview, NULL,
2021 MSG_CLABEL_FLAG_MASK, TRUE);
2025 alertpanel_notice(_("No labeled messages."));
2027 summary_select_node(summaryview, node, TRUE, FALSE);
2030 void summary_select_next_labeled(SummaryView *summaryview)
2034 node = summary_find_next_flagged_msg
2035 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
2040 val = alertpanel(_("No more labeled messages"),
2041 _("No labeled message found. "
2042 "Search from the beginning?"),
2043 GTK_STOCK_NO, "+"GTK_STOCK_YES, NULL);
2044 if (val != G_ALERTALTERNATE) return;
2045 node = summary_find_next_flagged_msg(summaryview, NULL,
2046 MSG_CLABEL_FLAG_MASK, TRUE);
2050 alertpanel_notice(_("No labeled messages."));
2052 summary_select_node(summaryview, node, TRUE, FALSE);
2055 void summary_select_last_read(SummaryView *summaryview)
2057 if (summaryview->last_displayed)
2058 summary_select_node(summaryview, summaryview->last_displayed, TRUE, FALSE);
2061 void summary_select_parent(SummaryView *summaryview)
2063 GtkCTreeNode *node = NULL;
2065 if (summaryview->selected)
2066 node = GTK_CTREE_ROW(summaryview->selected)->parent;
2068 summary_select_node(summaryview, node, TRUE, FALSE);
2071 void summary_select_by_msgnum(SummaryView *summaryview, guint msgnum)
2075 node = summary_find_msg_by_msgnum(summaryview, msgnum);
2076 summary_select_node(summaryview, node, FALSE, TRUE);
2079 typedef struct _PostponedSelectData
2086 SummaryView *summaryview;
2087 gboolean display_msg;
2088 gboolean do_refresh;
2089 } PostponedSelectData;
2091 static gboolean summary_select_retry(void *data)
2093 PostponedSelectData *psdata = (PostponedSelectData *)data;
2094 debug_print("trying again\n");
2096 summary_selected(psdata->ctree, psdata->row,
2097 psdata->column, psdata->summaryview);
2098 else if (psdata->node)
2099 summary_select_node(psdata->summaryview, psdata->node,
2100 psdata->display_msg, psdata->do_refresh);
2102 summary_step(psdata->summaryview, psdata->type);
2108 * summary_select_node:
2109 * @summaryview: Summary view.
2110 * @node: Summary tree node.
2111 * @display_msg: TRUE to display the selected message.
2112 * @do_refresh: TRUE to refresh the widget.
2114 * Select @node (bringing it into view by scrolling and expanding its
2115 * thread, if necessary) and unselect all others. If @display_msg is
2116 * TRUE, display the corresponding message in the message view.
2117 * If @do_refresh is TRUE, the widget is refreshed.
2119 void summary_select_node(SummaryView *summaryview, GtkCTreeNode *node,
2120 gboolean display_msg, gboolean do_refresh)
2122 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2124 if (summary_is_locked(summaryview)
2125 && !GTK_SCTREE(ctree)->selecting_range
2126 && summaryview->messageview->mimeview
2127 && summaryview->messageview->mimeview->type == MIMEVIEW_TEXT
2128 && summaryview->messageview->mimeview->textview->loading) {
2129 PostponedSelectData *data = g_new0(PostponedSelectData, 1);
2130 summaryview->messageview->mimeview->textview->stop_loading = TRUE;
2132 data->ctree = ctree;
2135 data->summaryview = summaryview;
2136 data->display_msg = display_msg;
2137 data->do_refresh = do_refresh;
2138 debug_print("postponing open of message till end of load\n");
2139 g_timeout_add(100, summary_select_retry, data);
2142 if (summary_is_locked(summaryview)) {
2145 if (!summaryview->folder_item)
2148 gtkut_ctree_expand_parent_all(ctree, node);
2150 summary_lock(summaryview);
2152 summary_unlock(summaryview);
2153 gtk_widget_grab_focus(GTK_WIDGET(ctree));
2154 gtk_ctree_node_moveto(ctree, node, 0, 0.5, 0);
2156 if (display_msg && summaryview->displayed == node)
2157 summaryview->displayed = NULL;
2158 summaryview->display_msg = display_msg;
2159 gtk_sctree_select(GTK_SCTREE(ctree), node);
2160 if (summaryview->selected == NULL)
2161 summaryview->selected = node;
2165 guint summary_get_msgnum(SummaryView *summaryview, GtkCTreeNode *node)
2167 GtkCTree *ctree =NULL;
2172 ctree = GTK_CTREE(summaryview->ctree);
2175 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2177 return msginfo->msgnum;
2182 static GtkCTreeNode *summary_find_prev_msg(SummaryView *summaryview,
2183 GtkCTreeNode *current_node)
2185 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2190 node = current_node;
2192 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
2194 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
2195 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2196 if (msginfo && !MSG_IS_DELETED(msginfo->flags)) break;
2202 static GtkCTreeNode *summary_find_next_msg(SummaryView *summaryview,
2203 GtkCTreeNode *current_node)
2205 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2210 node = current_node;
2212 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2214 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
2215 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2216 if (msginfo && !MSG_IS_DELETED(msginfo->flags)
2217 && !MSG_IS_MOVE(msginfo->flags)) break;
2223 static GtkCTreeNode *summary_find_prev_flagged_msg(SummaryView *summaryview,
2224 GtkCTreeNode *current_node,
2226 gboolean start_from_prev)
2228 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2233 if (start_from_prev)
2234 node = GTK_CTREE_NODE_PREV(current_node);
2236 node = current_node;
2238 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
2240 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
2241 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2242 if (msginfo && (msginfo->flags.perm_flags & flags) != 0) break;
2248 static GtkCTreeNode *summary_find_next_flagged_msg(SummaryView *summaryview,
2249 GtkCTreeNode *current_node,
2251 gboolean start_from_next)
2253 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2258 if (start_from_next)
2259 node = gtkut_ctree_node_next(ctree, current_node);
2261 node = current_node;
2263 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2265 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
2266 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2267 /* Find msg with matching flags but ignore messages with
2268 ignore flags, if searching for new or unread messages */
2269 if ((msginfo && (msginfo->flags.perm_flags & flags) != 0) &&
2270 !(((flags & (MSG_NEW | MSG_UNREAD)) != 0) && MSG_IS_IGNORE_THREAD(msginfo->flags))
2278 static GtkCTreeNode *summary_find_msg_by_msgnum(SummaryView *summaryview,
2281 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2285 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2287 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
2288 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2289 if (msginfo && msginfo->msgnum == msgnum) break;
2295 static guint attract_hash_func(gconstpointer key)
2301 Xstrdup_a(str, (const gchar *)key, return 0);
2308 for (p += 1; *p != '\0'; p++)
2309 h = (h << 5) - h + *p;
2315 static gint attract_compare_func(gconstpointer a, gconstpointer b)
2317 return subject_compare((const gchar *)a, (const gchar *)b) == 0;
2320 void summary_attract_by_subject(SummaryView *summaryview)
2322 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2323 GtkCList *clist = GTK_CLIST(ctree);
2324 GtkCTreeNode *src_node;
2325 GtkCTreeNode *dst_node, *sibling;
2327 MsgInfo *src_msginfo, *dst_msginfo;
2328 GHashTable *subject_table;
2330 debug_print("Attracting messages by subject...");
2331 STATUSBAR_PUSH(summaryview->mainwin,
2332 _("Attracting messages by subject..."));
2334 main_window_cursor_wait(summaryview->mainwin);
2335 summary_freeze(summaryview);
2337 subject_table = g_hash_table_new(attract_hash_func,
2338 attract_compare_func);
2340 for (src_node = GTK_CTREE_NODE(clist->row_list);
2343 tmp = GTK_CTREE_ROW(src_node)->sibling;
2344 src_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(src_node);
2345 if (!src_msginfo) continue;
2346 if (!src_msginfo->subject) continue;
2348 /* find attracting node */
2349 dst_node = g_hash_table_lookup(subject_table,
2350 src_msginfo->subject);
2353 dst_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(dst_node);
2355 /* if the time difference is more than 20 days,
2357 if (ABS(src_msginfo->date_t - dst_msginfo->date_t)
2358 > 60 * 60 * 24 * 20)
2361 sibling = GTK_CTREE_ROW(dst_node)->sibling;
2362 if (src_node != sibling)
2363 gtk_ctree_move(ctree, src_node, NULL, sibling);
2366 g_hash_table_insert(subject_table,
2367 src_msginfo->subject, src_node);
2370 g_hash_table_destroy(subject_table);
2372 gtk_ctree_node_moveto(ctree, summaryview->selected, 0, 0.5, 0);
2374 summary_thaw(summaryview);
2376 debug_print("done.\n");
2377 STATUSBAR_POP(summaryview->mainwin);
2379 main_window_cursor_normal(summaryview->mainwin);
2382 static void summary_free_msginfo_func(GtkCTree *ctree, GtkCTreeNode *node,
2385 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, node);
2388 procmsg_msginfo_free(msginfo);
2391 static void summary_set_marks_func(GtkCTree *ctree, GtkCTreeNode *node,
2394 SummaryView *summaryview = data;
2397 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2399 if (MSG_IS_DELETED(msginfo->flags))
2400 summaryview->deleted++;
2402 summaryview->total_size += msginfo->size;
2404 summary_set_row_marks(summaryview, node);
2407 static void summary_update_status(SummaryView *summaryview)
2409 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2413 summaryview->total_size =
2414 summaryview->deleted = summaryview->moved = summaryview->copied = 0;
2416 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2417 node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
2418 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
2423 if (MSG_IS_DELETED(msginfo->flags))
2424 summaryview->deleted++;
2425 if (MSG_IS_MOVE(msginfo->flags))
2426 summaryview->moved++;
2427 if (MSG_IS_COPY(msginfo->flags))
2428 summaryview->copied++;
2429 summaryview->total_size += msginfo->size;
2433 static void summary_status_show(SummaryView *summaryview)
2436 gchar *del, *mv, *cp;
2440 GList *rowlist, *cur;
2441 guint n_selected = 0, n_new = 0, n_unread = 0, n_total = 0;
2442 guint n_marked = 0, n_replied = 0, n_forwarded = 0, n_locked = 0, n_ignored = 0, n_watched = 0;
2443 goffset sel_size = 0, n_size = 0;
2446 #if GTK_CHECK_VERSION(2, 12, 0)
2450 if (!summaryview->folder_item) {
2451 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_folder), "");
2452 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), "");
2453 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_msgs), "");
2454 toolbar_main_set_sensitive(summaryview->mainwin);
2458 rowlist = GTK_CLIST(summaryview->ctree)->selection;
2459 for (cur = rowlist; cur != NULL && cur->data != NULL; cur = cur->next) {
2460 msginfo = gtk_ctree_node_get_row_data
2461 (GTK_CTREE(summaryview->ctree),
2462 GTK_CTREE_NODE(cur->data));
2464 sel_size += msginfo->size;
2469 if (summaryview->folder_item->hide_read_msgs
2470 || quicksearch_is_active(summaryview->quicksearch)) {
2471 rowlist = GTK_CLIST(summaryview->ctree)->row_list;
2472 for (cur = rowlist; cur != NULL && cur->data != NULL; cur = cur->next) {
2473 msginfo = gtk_ctree_node_get_row_data
2474 (GTK_CTREE(summaryview->ctree),
2475 GTK_CTREE_NODE(cur));
2477 n_size += msginfo->size;
2479 if (MSG_IS_NEW(msginfo->flags))
2481 if (MSG_IS_UNREAD(msginfo->flags))
2483 if (MSG_IS_MARKED(msginfo->flags))
2485 if (MSG_IS_REPLIED(msginfo->flags))
2487 if (MSG_IS_FORWARDED(msginfo->flags))
2489 if (MSG_IS_LOCKED(msginfo->flags))
2491 if (MSG_IS_IGNORE_THREAD(msginfo->flags))
2493 if (MSG_IS_WATCH_THREAD(msginfo->flags))
2498 n_new = summaryview->folder_item->new_msgs;
2499 n_unread = summaryview->folder_item->unread_msgs;
2500 n_marked = summaryview->folder_item->marked_msgs;
2501 n_replied = summaryview->folder_item->replied_msgs;
2502 n_forwarded = summaryview->folder_item->forwarded_msgs;
2503 n_locked = summaryview->folder_item->locked_msgs;
2504 n_ignored = summaryview->folder_item->ignored_msgs;
2505 n_watched = summaryview->folder_item->watched_msgs;
2506 n_total = summaryview->folder_item->total_msgs;
2507 n_size = summaryview->total_size;
2510 name = folder_item_get_name(summaryview->folder_item);
2511 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_folder), name);
2514 if (summaryview->deleted)
2515 del = g_strdup_printf(_("%d deleted"), summaryview->deleted);
2518 if (summaryview->moved)
2519 mv = g_strdup_printf(_("%s%d moved"),
2520 summaryview->deleted ? _(", ") : "",
2521 summaryview->moved);
2524 if (summaryview->copied)
2525 cp = g_strdup_printf(_("%s%d copied"),
2526 summaryview->deleted ||
2527 summaryview->moved ? _(", ") : "",
2528 summaryview->copied);
2532 if (summaryview->deleted || summaryview->moved || summaryview->copied)
2538 sel = g_strdup_printf(" (%s)", to_human_readable((goffset)sel_size));
2539 if (n_selected == 1)
2540 itstr = g_strdup(_(" item selected"));
2542 itstr = g_strdup(_(" items selected"));
2545 itstr = g_strdup("");
2548 if (prefs_common.layout_mode != SMALL_LAYOUT) {
2549 str = g_strconcat(n_selected ? itos(n_selected) : "",
2550 itstr, sel, spc, del, mv, cp, NULL);
2557 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), str);
2560 str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
2561 n_new, n_unread, n_total,
2562 to_human_readable((goffset)n_size));
2565 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_msgs), str);
2567 #if GTK_CHECK_VERSION(2, 12, 0)
2568 tooltip = g_strdup_printf(_("<b>Message summary</b>\n"
2570 "<b>Unread:</b> %d\n"
2571 "<b>Total:</b> %d\n"
2572 "<b>Size:</b> %s\n\n"
2573 "<b>Marked:</b> %d\n"
2574 "<b>Replied:</b> %d\n"
2575 "<b>Forwarded:</b> %d\n"
2576 "<b>Locked:</b> %d\n"
2577 "<b>Ignored:</b> %d\n"
2578 "<b>Watched:</b> %d"),
2579 n_new, n_unread, n_total,
2580 to_human_readable((goffset)n_size),
2581 n_marked,n_replied,n_forwarded,
2582 n_locked,n_ignored,n_watched);
2584 gtk_widget_set_tooltip_markup(GTK_WIDGET(summaryview->statlabel_msgs),
2589 gchar *ssize, *tsize;
2591 ssize = g_strdup(to_human_readable((goffset)sel_size));
2592 tsize = g_strdup(to_human_readable((goffset)n_size));
2593 str = g_strdup_printf(_("%d/%d selected (%s/%s), %d unread"),
2594 n_selected, n_total, ssize, tsize, n_unread);
2598 str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
2599 n_new, n_unread, n_total, to_human_readable((goffset)n_size));
2606 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_select), str);
2610 summary_set_menu_sensitive(summaryview);
2611 toolbar_main_set_sensitive(summaryview->mainwin);
2614 static void summary_set_column_titles(SummaryView *summaryview)
2616 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2617 FolderItem *item = summaryview->folder_item;
2623 SummaryColumnType type;
2624 GtkJustification justify;
2626 static FolderSortKey sort_by[N_SUMMARY_COLS] = {
2641 for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
2642 type = summaryview->col_state[pos].type;
2644 /* CLAWS: mime and unread are single char headers */
2645 justify = (type == S_COL_NUMBER || type == S_COL_SIZE)
2646 ? GTK_JUSTIFY_RIGHT : GTK_JUSTIFY_LEFT;
2654 if(type == S_COL_FROM && item != NULL &&
2655 FOLDER_SHOWS_TO_HDR(item) &&
2656 !summaryview->col_state[summaryview->col_pos[S_COL_TO]].visible)
2658 if(type == S_COL_NUMBER)
2659 title = gettext(col_label[type]);
2661 title = prefs_common_translated_header_name(col_label[type]);
2664 title = gettext(col_label[type]);
2667 if (type == S_COL_MIME) {
2668 label = gtk_image_new_from_pixmap(clipxpm, clipxpmmask);
2669 gtk_widget_show(label);
2670 gtk_clist_set_column_widget(clist, pos, label);
2671 gtk_sctree_set_column_tooltip(GTK_SCTREE(clist), pos, _("Attachment"));
2673 } else if (type == S_COL_MARK) {
2674 label = gtk_image_new_from_pixmap(markxpm, markxpmmask);
2675 gtk_widget_show(label);
2676 gtk_clist_set_column_widget(clist, pos, label);
2677 gtk_sctree_set_column_tooltip(GTK_SCTREE(clist), pos, _("Mark"));
2679 } else if (type == S_COL_LOCKED) {
2680 label = gtk_image_new_from_pixmap(lockedxpm, lockedxpmmask);
2681 gtk_widget_show(label);
2682 gtk_clist_set_column_widget(clist, pos, label);
2683 gtk_sctree_set_column_tooltip(GTK_SCTREE(clist), pos, _("Locked"));
2685 } else if (type == S_COL_STATUS) {
2686 gtk_clist_set_column_title(clist, pos, title);
2687 gtk_sctree_set_column_tooltip(GTK_SCTREE(clist), pos, _("Status"));
2691 hbox = gtk_hbox_new(FALSE, 4);
2692 label = gtk_label_new(title);
2694 gtk_widget_set_size_request(hbox, -1, 20);
2697 if (justify == GTK_JUSTIFY_RIGHT)
2698 gtk_box_pack_end(GTK_BOX(hbox), label,
2701 gtk_box_pack_start(GTK_BOX(hbox), label,
2704 if (summaryview->sort_key == sort_by[type]) {
2705 arrow = gtk_arrow_new
2706 (summaryview->sort_type == SORT_ASCENDING
2707 ? GTK_ARROW_DOWN : GTK_ARROW_UP,
2709 if (justify == GTK_JUSTIFY_RIGHT)
2710 gtk_box_pack_start(GTK_BOX(hbox), arrow,
2713 gtk_box_pack_end(GTK_BOX(hbox), arrow,
2717 gtk_widget_show_all(hbox);
2718 gtk_clist_set_column_widget(clist, pos, hbox);
2722 void summary_reflect_tags_changes(SummaryView *summaryview)
2727 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2728 gboolean froze = FALSE;
2729 gboolean redisplay = FALSE;
2731 /* re-create colorlabel submenu */
2732 menu = GTK_MENU_SHELL(summaryview->tags_menu);
2733 g_return_if_fail(menu != NULL);
2735 /* clear items. get item pointers. */
2736 for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
2737 gtk_menu_item_remove_submenu(GTK_MENU_ITEM(cur->data));
2739 summary_tags_menu_create(summaryview, TRUE);
2741 START_LONG_OPERATION(summaryview, TRUE);
2742 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
2743 node = gtkut_ctree_node_next(ctree, node)) {
2744 redisplay |= summary_set_row_tag(summaryview,
2745 node, TRUE, FALSE, 0);
2747 END_LONG_OPERATION(summaryview);
2749 summary_redisplay_msg(summaryview);
2753 void summary_reflect_prefs(void)
2755 static gchar *last_smallfont = NULL;
2756 static gchar *last_normalfont = NULL;
2757 static gchar *last_boldfont = NULL;
2758 gboolean update_font = FALSE;
2759 SummaryView *summaryview = NULL;
2761 if (!mainwindow_get_mainwindow())
2763 summaryview = mainwindow_get_mainwindow()->summaryview;
2765 if (!last_smallfont || strcmp(last_smallfont, SMALL_FONT) ||
2766 !last_normalfont || strcmp(last_normalfont, NORMAL_FONT) ||
2767 !last_boldfont || strcmp(last_boldfont, BOLD_FONT))
2770 g_free(last_smallfont);
2771 last_smallfont = g_strdup(SMALL_FONT);
2772 g_free(last_normalfont);
2773 last_normalfont = g_strdup(NORMAL_FONT);
2774 g_free(last_boldfont);
2775 last_boldfont = g_strdup(BOLD_FONT);
2778 bold_style = bold_marked_style = bold_deleted_style =
2779 small_style = small_marked_style = small_deleted_style = NULL;
2780 summary_set_fonts(summaryview);
2783 summary_set_column_titles(summaryview);
2784 summary_relayout(summaryview);
2786 if (summaryview->folder_item)
2787 summary_show(summaryview, summaryview->folder_item);
2790 void summary_sort(SummaryView *summaryview,
2791 FolderSortKey sort_key, FolderSortType sort_type)
2793 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2794 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2795 GtkCListCompareFunc cmp_func = NULL;
2797 g_signal_handlers_block_by_func(G_OBJECT(summaryview->ctree),
2798 G_CALLBACK(summary_tree_expanded), summaryview);
2799 summary_freeze(summaryview);
2803 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mark;
2805 case SORT_BY_STATUS:
2806 cmp_func = (GtkCListCompareFunc)summary_cmp_by_status;
2809 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mime;
2811 case SORT_BY_NUMBER:
2812 cmp_func = (GtkCListCompareFunc)summary_cmp_by_num;
2815 cmp_func = (GtkCListCompareFunc)summary_cmp_by_size;
2818 cmp_func = (GtkCListCompareFunc)summary_cmp_by_date;
2820 case SORT_BY_THREAD_DATE:
2821 cmp_func = (GtkCListCompareFunc)summary_cmp_by_thread_date;
2824 cmp_func = (GtkCListCompareFunc)summary_cmp_by_from;
2826 case SORT_BY_SUBJECT:
2827 if (summaryview->simplify_subject_preg)
2828 cmp_func = (GtkCListCompareFunc)summary_cmp_by_simplified_subject;
2830 cmp_func = (GtkCListCompareFunc)summary_cmp_by_subject;
2833 cmp_func = (GtkCListCompareFunc)summary_cmp_by_score;
2836 cmp_func = (GtkCListCompareFunc)summary_cmp_by_label;
2839 cmp_func = (GtkCListCompareFunc)summary_cmp_by_to;
2841 case SORT_BY_LOCKED:
2842 cmp_func = (GtkCListCompareFunc)summary_cmp_by_locked;
2845 cmp_func = (GtkCListCompareFunc)summary_cmp_by_tags;
2853 summaryview->sort_key = sort_key;
2854 summaryview->sort_type = sort_type;
2856 summary_set_column_titles(summaryview);
2857 summary_set_menu_sensitive(summaryview);
2859 /* allow fallback to don't sort */
2860 if (summaryview->sort_key == SORT_BY_NONE)
2863 if (cmp_func != NULL) {
2864 debug_print("Sorting summary...");
2865 STATUSBAR_PUSH(summaryview->mainwin, _("Sorting summary..."));
2867 main_window_cursor_wait(summaryview->mainwin);
2869 gtk_clist_set_compare_func(clist, cmp_func);
2871 gtk_clist_set_sort_type(clist, (GtkSortType)sort_type);
2872 gtk_sctree_sort_recursive(ctree, NULL);
2874 gtk_ctree_node_moveto(ctree, summaryview->selected, 0, 0.5, 0);
2876 main_window_cursor_normal(summaryview->mainwin);
2878 debug_print("done.\n");
2879 STATUSBAR_POP(summaryview->mainwin);
2882 summary_thaw(summaryview);
2883 g_signal_handlers_unblock_by_func(G_OBJECT(summaryview->ctree),
2884 G_CALLBACK(summary_tree_expanded), summaryview);
2888 static gboolean summary_update_thread_age(GNode *node, gpointer data)
2890 MsgInfo *msginfo = node->data;
2891 time_t *most_recent = (time_t *)data;
2893 if (msginfo->date_t > *most_recent) {
2894 *most_recent = msginfo->date_t;
2899 static void summary_find_thread_age(GNode *gnode)
2901 MsgInfo *msginfo = (MsgInfo *)gnode->data;
2906 most_recent = msginfo->thread_date = msginfo->date_t;
2908 g_node_traverse(gnode, G_IN_ORDER, G_TRAVERSE_ALL, -1, summary_update_thread_age, &most_recent);
2910 msginfo->thread_date = most_recent;
2913 static gboolean summary_insert_gnode_func(GtkCTree *ctree, guint depth, GNode *gnode,
2914 GtkCTreeNode *cnode, gpointer data)
2916 SummaryView *summaryview = (SummaryView *)data;
2917 MsgInfo *msginfo = (MsgInfo *)gnode->data;
2918 gchar *text[N_SUMMARY_COLS];
2919 gint *col_pos = summaryview->col_pos;
2920 const gchar *msgid = msginfo->msgid;
2921 GHashTable *msgid_table = summaryview->msgid_table;
2922 gboolean vert = (prefs_common.layout_mode == VERTICAL_LAYOUT);
2924 summary_set_header(summaryview, text, msginfo);
2926 gtk_sctree_set_node_info(ctree, cnode, text[col_pos[S_COL_SUBJECT]], 2,
2927 NULL, NULL, NULL, NULL, FALSE, summaryview->threaded && !summaryview->thread_collapsed);
2928 #define SET_TEXT(col) { \
2929 gtk_ctree_node_set_text(ctree, cnode, col_pos[col], \
2930 text[col_pos[col]]); \
2933 if (summaryview->col_state[summaryview->col_pos[S_COL_NUMBER]].visible)
2934 SET_TEXT(S_COL_NUMBER);
2935 if (summaryview->col_state[summaryview->col_pos[S_COL_SCORE]].visible)
2936 SET_TEXT(S_COL_SCORE);
2937 if (summaryview->col_state[summaryview->col_pos[S_COL_SIZE]].visible)
2938 SET_TEXT(S_COL_SIZE);
2939 if (summaryview->col_state[summaryview->col_pos[S_COL_DATE]].visible)
2940 SET_TEXT(S_COL_DATE);
2941 if (summaryview->col_state[summaryview->col_pos[S_COL_FROM]].visible)
2942 SET_TEXT(S_COL_FROM);
2943 if (summaryview->col_state[summaryview->col_pos[S_COL_TO]].visible)
2945 if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible)
2946 SET_TEXT(S_COL_TAGS);
2948 if (vert && prefs_common.two_line_vert)
2949 g_free(text[summaryview->col_pos[S_COL_SUBJECT]]);
2953 GTKUT_CTREE_NODE_SET_ROW_DATA(cnode, msginfo);
2954 summary_set_marks_func(ctree, cnode, summaryview);
2956 if (msgid && msgid[0] != '\0')
2957 g_hash_table_insert(msgid_table, (gchar *)msgid, cnode);
2962 static void summary_set_ctree_from_list(SummaryView *summaryview,
2965 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2967 GtkCTreeNode *node = NULL;
2968 GHashTable *msgid_table;
2969 GHashTable *subject_table = NULL;
2971 gboolean vert = (prefs_common.layout_mode == VERTICAL_LAYOUT);
2977 debug_print("\tSetting summary from message data...\n");
2978 STATUSBAR_PUSH(summaryview->mainwin,
2979 _("Setting summary from message data..."));
2982 g_signal_handlers_block_by_func(G_OBJECT(ctree),
2983 G_CALLBACK(summary_tree_expanded), summaryview);
2985 msgid_table = g_hash_table_new(g_str_hash, g_str_equal);
2986 summaryview->msgid_table = msgid_table;
2988 if (prefs_common.thread_by_subject) {
2989 subject_table = g_hash_table_new(g_str_hash, g_str_equal);
2990 summaryview->subject_table = subject_table;
2992 summaryview->subject_table = NULL;
2995 if (prefs_common.use_addr_book)
2996 start_address_completion(NULL);
2998 if (summaryview->threaded) {
2999 GNode *root, *gnode;
3000 START_TIMING("threaded");
3001 root = procmsg_get_thread_tree(mlist);
3004 for (gnode = root->children; gnode != NULL;
3005 gnode = gnode->next) {
3006 summary_find_thread_age(gnode);
3007 node = gtk_sctree_insert_gnode
3008 (ctree, NULL, node, gnode,
3009 summary_insert_gnode_func, summaryview);
3012 g_node_destroy(root);
3016 gchar *text[N_SUMMARY_COLS];
3017 START_TIMING("unthreaded");
3019 for (; mlist != NULL; mlist = mlist->next) {
3020 msginfo = (MsgInfo *)mlist->data;
3022 summary_set_header(summaryview, text, msginfo);
3024 node = gtk_sctree_insert_node
3025 (ctree, NULL, node, text, 2,
3026 NULL, NULL, NULL, NULL,
3028 if (vert && prefs_common.two_line_vert)
3029 g_free(text[summaryview->col_pos[S_COL_SUBJECT]]);
3031 GTKUT_CTREE_NODE_SET_ROW_DATA(node, msginfo);
3032 summary_set_marks_func(ctree, node, summaryview);
3034 if (msginfo->msgid && msginfo->msgid[0] != '\0')
3035 g_hash_table_insert(msgid_table,
3036 msginfo->msgid, node);
3038 if (prefs_common.thread_by_subject)
3039 subject_table_insert(subject_table,
3047 if (prefs_common.enable_hscrollbar &&
3048 summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
3051 optimal_width = gtk_clist_optimal_column_width
3052 (GTK_CLIST(ctree), summaryview->col_pos[S_COL_SUBJECT]);
3053 gtk_clist_set_column_width(GTK_CLIST(ctree),
3054 summaryview->col_pos[S_COL_SUBJECT],
3058 if (prefs_common.use_addr_book)
3059 end_address_completion();
3061 debug_print("done.\n");
3062 STATUSBAR_POP(summaryview->mainwin);
3063 if (debug_get_mode()) {
3064 debug_print("\tmsgid hash table size = %d\n",
3065 g_hash_table_size(msgid_table));
3066 if (prefs_common.thread_by_subject)
3067 debug_print("\tsubject hash table size = %d\n",
3068 g_hash_table_size(subject_table));
3071 summary_sort(summaryview, summaryview->sort_key, summaryview->sort_type);
3073 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3075 if (prefs_common.bold_unread) {
3076 START_TIMING("bold_unread");
3078 GtkCTreeNode *next = GTK_CTREE_NODE_NEXT(node);
3079 if (GTK_CTREE_ROW(node)->children)
3080 summary_set_row_marks(summaryview, node);
3086 g_signal_handlers_unblock_by_func(G_OBJECT(ctree),
3087 G_CALLBACK(summary_tree_expanded), summaryview);
3091 static gchar *summary_complete_address(const gchar *addr)
3094 gchar *res, *tmp, *email_addr;
3096 if (addr == NULL || !strchr(addr, '@'))
3099 Xstrdup_a(email_addr, addr, return NULL);
3100 extract_address(email_addr);
3105 * completion stuff must be already initialized
3108 if (1 < (count = complete_address(email_addr))) {
3109 tmp = get_complete_address(1);
3110 res = procheader_get_fromname(tmp);
3117 static inline void summary_set_header(SummaryView *summaryview, gchar *text[],
3120 static gchar date_modified[80];
3121 static gchar col_score[11];
3122 static gchar buf[BUFFSIZE], tmp1[BUFFSIZE], tmp2[BUFFSIZE], tmp3[BUFFSIZE];
3123 gint *col_pos = summaryview->col_pos;
3124 gchar *from_text = NULL, *to_text = NULL, *tags_text = NULL;
3125 gboolean should_swap = FALSE;
3126 gboolean vert = (prefs_common.layout_mode == VERTICAL_LAYOUT);
3127 #if GTK_CHECK_VERSION(2,12,0)
3128 static const gchar *color_dim_rgb = NULL;
3130 color_dim_rgb = gdk_color_to_string(&summaryview->color_dim);
3132 static const gchar *color_dim_rgb = "#888888";
3134 text[col_pos[S_COL_FROM]] = "";
3135 text[col_pos[S_COL_TO]] = "";
3136 text[col_pos[S_COL_SUBJECT]]= "";
3137 text[col_pos[S_COL_MARK]] = "";
3138 text[col_pos[S_COL_STATUS]] = "";
3139 text[col_pos[S_COL_MIME]] = "";
3140 text[col_pos[S_COL_LOCKED]] = "";
3141 text[col_pos[S_COL_DATE]] = "";
3142 text[col_pos[S_COL_TAGS]] = "";
3143 if (summaryview->col_state[summaryview->col_pos[S_COL_NUMBER]].visible)
3144 text[col_pos[S_COL_NUMBER]] = itos(msginfo->msgnum);
3146 text[col_pos[S_COL_NUMBER]] = "";
3149 if (summaryview->col_state[summaryview->col_pos[S_COL_SIZE]].visible)
3150 text[col_pos[S_COL_SIZE]] = to_human_readable(msginfo->size);
3152 text[col_pos[S_COL_SIZE]] = "";
3154 if (summaryview->col_state[summaryview->col_pos[S_COL_SCORE]].visible)
3155 text[col_pos[S_COL_SCORE]] = itos_buf(col_score, msginfo->score);
3157 text[col_pos[S_COL_SCORE]] = "";
3159 if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible) {
3160 tags_text = procmsg_msginfo_get_tags_str(msginfo);
3162 text[col_pos[S_COL_TAGS]] = "-";
3164 strncpy2(tmp1, tags_text, sizeof(tmp1));
3165 tmp1[sizeof(tmp1)-1]='\0';
3167 text[col_pos[S_COL_TAGS]] = tmp1;
3170 text[col_pos[S_COL_TAGS]] = "";
3173 if (summaryview->col_state[summaryview->col_pos[S_COL_DATE]].visible ||
3174 (vert && prefs_common.two_line_vert)) {
3175 if (msginfo->date_t) {
3176 procheader_date_get_localtime(date_modified,
3177 sizeof(date_modified),
3179 text[col_pos[S_COL_DATE]] = date_modified;
3180 } else if (msginfo->date)
3181 text[col_pos[S_COL_DATE]] = msginfo->date;
3183 text[col_pos[S_COL_DATE]] = _("(No Date)");
3186 if (prefs_common.swap_from && msginfo->from && msginfo->to
3187 && !summaryview->col_state[summaryview->col_pos[S_COL_TO]].visible) {
3190 addr = g_strdup(msginfo->from);
3193 extract_address(addr);
3194 if (account_find_from_address(addr, FALSE)) {
3201 if (!prefs_common.use_addr_book) {
3202 from_text = msginfo->fromname ?
3206 gchar *tmp = summary_complete_address(msginfo->from);
3208 strncpy2(buf, tmp, sizeof(buf));
3212 from_text = (msginfo->fromname) ?
3218 to_text = msginfo->to ? msginfo->to :
3219 (msginfo->cc ? msginfo->cc :
3220 (msginfo->newsgroups ? msginfo->newsgroups : _("(No Recipient)")
3224 text[col_pos[S_COL_TO]] = to_text;
3226 text[col_pos[S_COL_FROM]] = from_text;
3228 snprintf(tmp2, BUFFSIZE-1, "--> %s", to_text);
3229 tmp2[BUFFSIZE-1]='\0';
3230 text[col_pos[S_COL_FROM]] = tmp2;
3233 if (summaryview->simplify_subject_preg != NULL)
3234 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ?
3235 string_remove_match(tmp3, BUFFSIZE, msginfo->subject,
3236 summaryview->simplify_subject_preg) :
3239 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ? msginfo->subject :
3241 if (vert && prefs_common.two_line_vert) {
3242 if (!FOLDER_SHOWS_TO_HDR(summaryview->folder_item)) {
3243 gchar *tmp = g_markup_printf_escaped(_("%s\n<span color='%s' style='italic'>From: %s, on %s</span>"),
3244 text[col_pos[S_COL_SUBJECT]],
3246 text[col_pos[S_COL_FROM]],
3247 text[col_pos[S_COL_DATE]]);
3248 text[col_pos[S_COL_SUBJECT]] = tmp;
3250 gchar *tmp = g_markup_printf_escaped(_("%s\n<span color='%s' style='italic'>To: %s, on %s</span>"),
3251 text[col_pos[S_COL_SUBJECT]],
3253 text[col_pos[S_COL_TO]],
3254 text[col_pos[S_COL_DATE]]);
3255 text[col_pos[S_COL_SUBJECT]] = tmp;
3260 static void summary_display_msg(SummaryView *summaryview, GtkCTreeNode *row)
3262 summary_display_msg_full(summaryview, row, FALSE, FALSE);
3265 static gboolean defer_change(gpointer data);
3266 typedef struct _ChangeData {
3268 gint op; /* 0, 1, 2 for unset, set, change */
3269 MsgPermFlags set_flags;
3270 MsgTmpFlags set_tmp_flags;
3271 MsgPermFlags unset_flags;
3272 MsgTmpFlags unset_tmp_flags;
3275 static void summary_msginfo_unset_flags(MsgInfo *msginfo, MsgPermFlags flags, MsgTmpFlags tmp_flags)
3277 if (!msginfo->folder || !msginfo->folder->processing_pending) {
3278 debug_print("flags: doing unset now\n");
3279 procmsg_msginfo_unset_flags(msginfo, flags, tmp_flags);
3281 ChangeData *unset_data = g_new0(ChangeData, 1);
3282 unset_data->info = msginfo;
3284 unset_data->unset_flags = flags;
3285 unset_data->unset_tmp_flags = tmp_flags;
3286 debug_print("flags: deferring unset\n");
3287 g_timeout_add(100, defer_change, unset_data);
3291 static void summary_msginfo_set_flags(MsgInfo *msginfo, MsgPermFlags flags, MsgTmpFlags tmp_flags)
3293 if (!msginfo->folder || !msginfo->folder->processing_pending) {
3294 debug_print("flags: doing set now\n");
3295 procmsg_msginfo_set_flags(msginfo, flags, tmp_flags);
3297 ChangeData *set_data = g_new0(ChangeData, 1);
3298 set_data->info = msginfo;
3300 set_data->set_flags = flags;
3301 set_data->set_tmp_flags = tmp_flags;
3302 debug_print("flags: deferring set\n");
3303 g_timeout_add(100, defer_change, set_data);
3307 static void summary_msginfo_change_flags(MsgInfo *msginfo,
3308 MsgPermFlags add_flags, MsgTmpFlags add_tmp_flags,
3309 MsgPermFlags rem_flags, MsgTmpFlags rem_tmp_flags)
3311 if (!msginfo->folder || !msginfo->folder->processing_pending) {
3312 debug_print("flags: doing change now\n");
3313 procmsg_msginfo_change_flags(msginfo, add_flags, add_tmp_flags,
3314 rem_flags, rem_tmp_flags);
3316 ChangeData *change_data = g_new0(ChangeData, 1);
3317 change_data->info = msginfo;
3318 change_data->op = 2;
3319 change_data->set_flags = add_flags;
3320 change_data->set_tmp_flags = add_tmp_flags;
3321 change_data->unset_flags = rem_flags;
3322 change_data->unset_tmp_flags = rem_tmp_flags;
3323 debug_print("flags: deferring change\n");
3324 g_timeout_add(100, defer_change, change_data);
3328 gboolean defer_change(gpointer data)
3330 ChangeData *chg = (ChangeData *)data;
3331 if (chg->info->folder && chg->info->folder->processing_pending) {
3332 debug_print("flags: trying later\n");
3333 return TRUE; /* try again */
3335 debug_print("flags: finally doing it\n");
3338 procmsg_msginfo_unset_flags(chg->info, chg->unset_flags, chg->unset_tmp_flags);
3341 procmsg_msginfo_set_flags(chg->info, chg->set_flags, chg->set_tmp_flags);
3344 procmsg_msginfo_change_flags(chg->info, chg->set_flags, chg->set_tmp_flags,
3345 chg->unset_flags, chg->unset_tmp_flags);
3348 g_warning("shouldn't happen\n");
3355 static void msginfo_mark_as_read (SummaryView *summaryview, MsgInfo *msginfo,
3358 g_return_if_fail(summaryview != NULL);
3359 g_return_if_fail(msginfo != NULL);
3360 g_return_if_fail(row != NULL);
3362 if (MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)) {
3363 summary_msginfo_unset_flags
3364 (msginfo, MSG_NEW | MSG_UNREAD, 0);
3365 summary_set_row_marks(summaryview, row);
3366 summary_status_show(summaryview);
3372 SummaryView *summaryview;
3375 static int msginfo_mark_as_read_timeout(void *data)
3377 MarkAsReadData *mdata = (MarkAsReadData *)data;
3381 if (mdata->msginfo == summary_get_selected_msg(mdata->summaryview))
3382 msginfo_mark_as_read(mdata->summaryview, mdata->msginfo,
3383 mdata->summaryview->selected);
3390 static void summary_display_msg_full(SummaryView *summaryview,
3392 gboolean new_window, gboolean all_headers)
3394 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3399 if (summaryview->displayed == row)
3401 else if (summaryview->messageview)
3402 summaryview->messageview->filtered = FALSE;
3404 g_return_if_fail(row != NULL);
3406 if (summary_is_locked(summaryview)) return;
3407 summary_lock(summaryview);
3409 STATUSBAR_POP(summaryview->mainwin);
3412 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3414 g_return_if_fail(msginfo);
3416 if (new_window && prefs_common.layout_mode != SMALL_LAYOUT) {
3417 MessageView *msgview;
3419 msgview = messageview_create_with_new_window(summaryview->mainwin);
3420 val = messageview_show(msgview, msginfo, all_headers);
3422 MessageView *msgview;
3424 if (prefs_common.layout_mode == SMALL_LAYOUT) {
3425 if (summaryview->ext_messageview == NULL)
3426 summaryview->ext_messageview = messageview_create_with_new_window(summaryview->mainwin);
3428 gtkut_window_popup(summaryview->ext_messageview->window);
3429 msgview = summaryview->ext_messageview;
3430 summaryview->last_displayed = summaryview->displayed;
3431 summaryview->displayed = row;
3432 val = messageview_show(msgview, msginfo, all_headers);
3433 if (GTK_CLIST(msgview->mimeview->ctree)->row_list == NULL)
3434 gtk_widget_grab_focus(summaryview->ctree);
3435 gtkut_ctree_node_move_if_on_the_edge(ctree, row,
3436 GTK_CLIST(summaryview->ctree)->focus_row);
3438 msgview = summaryview->messageview;
3439 summaryview->last_displayed = summaryview->displayed;
3440 summaryview->displayed = row;
3441 if (!messageview_is_visible(msgview)) {
3442 main_window_toggle_message_view(summaryview->mainwin);
3445 val = messageview_show(msgview, msginfo, all_headers);
3446 if (GTK_CLIST(msgview->mimeview->ctree)->row_list == NULL)
3447 gtk_widget_grab_focus(summaryview->ctree);
3448 gtkut_ctree_node_move_if_on_the_edge(ctree, row,
3449 GTK_CLIST(summaryview->ctree)->focus_row);
3453 if (val == 0 && MSG_IS_UNREAD(msginfo->flags)) {
3454 if (!prefs_common.mark_as_read_on_new_window &&
3455 prefs_common.mark_as_read_delay) {
3456 MarkAsReadData *data = g_new0(MarkAsReadData, 1);
3457 data->summaryview = summaryview;
3458 data->msginfo = msginfo;
3459 g_timeout_add(prefs_common.mark_as_read_delay * 1000,
3460 msginfo_mark_as_read_timeout, data);
3461 } else if (new_window || !prefs_common.mark_as_read_on_new_window) {
3462 msginfo_mark_as_read(summaryview, msginfo, row);
3466 summary_set_menu_sensitive(summaryview);
3467 toolbar_main_set_sensitive(summaryview->mainwin);
3468 messageview_set_menu_sensitive(summaryview->messageview);
3470 summary_unlock(summaryview);
3474 void summary_display_msg_selected(SummaryView *summaryview,
3475 gboolean all_headers)
3477 if (summary_is_locked(summaryview)) return;
3478 summaryview->displayed = NULL;
3479 summary_display_msg_full(summaryview, summaryview->selected, FALSE,
3483 void summary_redisplay_msg(SummaryView *summaryview)
3487 if (summaryview->displayed) {
3488 node = summaryview->displayed;
3489 summaryview->displayed = NULL;
3490 summary_display_msg(summaryview, node);
3494 void summary_open_msg(SummaryView *summaryview)
3496 if (!summaryview->selected) return;
3498 /* CLAWS: if separate message view, don't open a new window
3499 * but rather use the current separated message view */
3500 summary_display_msg_full(summaryview, summaryview->selected,
3504 void summary_view_source(SummaryView * summaryview)
3506 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3508 SourceWindow *srcwin;
3510 if (!summaryview->selected) return;
3512 srcwin = source_window_create();
3513 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
3514 source_window_show_msg(srcwin, msginfo);
3515 source_window_show(srcwin);
3517 maemo_window_full_screen_if_needed(GTK_WINDOW(srcwin->window));
3518 maemo_connect_key_press_to_mainwindow(GTK_WINDOW(srcwin->window));
3522 void summary_reedit(SummaryView *summaryview)
3526 if (!summaryview->selected) return;
3527 if (!summaryview->folder_item) return;
3528 if (!FOLDER_SHOWS_TO_HDR(summaryview->folder_item))
3531 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
3532 summaryview->selected);
3533 if (!msginfo) return;
3535 compose_reedit(msginfo, FALSE);
3538 gboolean summary_step(SummaryView *summaryview, GtkScrollType type)
3540 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3543 if (summary_is_locked(summaryview)
3544 && !GTK_SCTREE(ctree)->selecting_range
3545 && summaryview->messageview->mimeview
3546 && summaryview->messageview->mimeview->type == MIMEVIEW_TEXT
3547 && summaryview->messageview->mimeview->textview->loading) {
3548 PostponedSelectData *data = g_new0(PostponedSelectData, 1);
3549 summaryview->messageview->mimeview->textview->stop_loading = TRUE;
3551 data->ctree = ctree;
3555 data->summaryview = summaryview;
3556 debug_print("postponing open of message till end of load\n");
3557 g_timeout_add(100, summary_select_retry, data);
3560 if (summary_is_locked(summaryview))
3562 if (type == GTK_SCROLL_STEP_FORWARD) {
3563 node = gtkut_ctree_node_next(ctree, summaryview->selected);
3565 gtkut_ctree_expand_parent_all(ctree, node);
3569 if (summaryview->selected) {
3570 node = GTK_CTREE_NODE_PREV(summaryview->selected);
3571 if (!node) return FALSE;
3575 if (messageview_is_visible(summaryview->messageview))
3576 summaryview->display_msg = TRUE;
3578 g_signal_emit_by_name(G_OBJECT(ctree), "scroll_vertical", type, 0.0);
3580 if (GTK_CLIST(ctree)->selection)
3581 gtk_sctree_set_anchor_row
3583 GTK_CTREE_NODE(GTK_CLIST(ctree)->selection->data));
3588 gboolean summary_is_list(SummaryView *summaryview)
3590 return (gtk_notebook_get_current_page(
3591 GTK_NOTEBOOK(summaryview->mainwidget_book)) == 0);
3594 void summary_toggle_view(SummaryView *summaryview)
3596 if (prefs_common.layout_mode == SMALL_LAYOUT)
3598 if (!messageview_is_visible(summaryview->messageview) &&
3599 summaryview->selected && summary_is_list(summaryview))
3600 summary_display_msg(summaryview,
3601 summaryview->selected);
3603 main_window_toggle_message_view(summaryview->mainwin);
3606 static gboolean summary_search_unread_recursive(GtkCTree *ctree,
3612 msginfo = gtk_ctree_node_get_row_data(ctree, node);
3613 if (msginfo && MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
3615 node = GTK_CTREE_ROW(node)->children;
3617 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3620 if (summary_search_unread_recursive(ctree, node) == TRUE)
3622 node = GTK_CTREE_ROW(node)->sibling;
3628 static gboolean summary_have_unread_children(SummaryView *summaryview,
3631 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3633 if (!node) return FALSE;
3635 node = GTK_CTREE_ROW(node)->children;
3638 if (summary_search_unread_recursive(ctree, node) == TRUE)
3640 node = GTK_CTREE_ROW(node)->sibling;
3645 static void summary_set_row_marks(SummaryView *summaryview, GtkCTreeNode *row)
3647 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3648 GtkStyle *style = NULL;
3651 gint *col_pos = summaryview->col_pos;
3653 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3654 if (!msginfo) return;
3656 flags = msginfo->flags;
3658 gtk_ctree_node_set_foreground(ctree, row, NULL);
3660 /* set new/unread column */
3661 if (MSG_IS_IGNORE_THREAD(flags)) {
3662 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3663 ignorethreadxpm, ignorethreadxpmmask);
3664 } else if (MSG_IS_WATCH_THREAD(flags)) {
3665 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3666 watchthreadxpm, watchthreadxpmmask);
3667 } else if (MSG_IS_SPAM(flags)) {
3668 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3669 spamxpm, spamxpmmask);
3670 } else if (MSG_IS_NEW(flags)) {
3671 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3672 newxpm, newxpmmask);
3673 } else if (MSG_IS_UNREAD(flags)) {
3674 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3675 unreadxpm, unreadxpmmask);
3676 } else if (MSG_IS_REPLIED(flags)) {
3677 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3678 repliedxpm, repliedxpmmask);
3679 } else if (MSG_IS_FORWARDED(flags)) {
3680 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
3681 forwardedxpm, forwardedxpmmask);
3683 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_STATUS],
3687 if (prefs_common.bold_unread &&
3688 ((MSG_IS_UNREAD(flags) && !MSG_IS_IGNORE_THREAD(flags)) ||
3689 (!GTK_CTREE_ROW(row)->expanded &&
3690 GTK_CTREE_ROW(row)->children &&
3691 summary_have_unread_children(summaryview, row))))
3694 /* set mark column */
3695 if (MSG_IS_DELETED(flags)) {
3696 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
3697 deletedxpm, deletedxpmmask);
3699 style = bold_deleted_style;
3701 style = small_deleted_style;
3703 gtk_ctree_node_set_foreground
3704 (ctree, row, &summaryview->color_dim);
3705 } else if (MSG_IS_MARKED(flags)) {
3706 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
3707 markxpm, markxpmmask);
3708 } else if (MSG_IS_MOVE(flags)) {
3709 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
3710 movedxpm, movedxpmmask);
3712 style = bold_marked_style;
3714 style = small_marked_style;
3716 gtk_ctree_node_set_foreground
3717 (ctree, row, &summaryview->color_marked);
3718 } else if (MSG_IS_COPY(flags)) {
3719 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
3720 copiedxpm, copiedxpmmask);
3722 style = bold_marked_style;
3724 style = small_marked_style;
3726 gtk_ctree_node_set_foreground
3727 (ctree, row, &summaryview->color_marked);
3729 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], "");
3732 if (MSG_IS_LOCKED(flags)) {
3733 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_LOCKED],
3734 lockedxpm, lockedxpmmask);
3737 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_LOCKED], "");
3740 if (MSG_IS_WITH_ATTACHMENT(flags) && MSG_IS_SIGNED(flags)) {
3741 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
3742 clipgpgsignedxpm, clipgpgsignedxpmmask);
3743 } else if (MSG_IS_SIGNED(flags)) {
3744 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
3745 gpgsignedxpm, gpgsignedxpmmask);
3746 } else if (MSG_IS_WITH_ATTACHMENT(flags) && MSG_IS_ENCRYPTED(flags)) {
3747 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
3748 clipkeyxpm, clipkeyxpmmask);
3749 } else if (MSG_IS_ENCRYPTED(flags)) {
3750 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
3751 keyxpm, keyxpmmask);
3752 } else if (MSG_IS_WITH_ATTACHMENT(flags)) {
3753 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
3754 clipxpm, clipxpmmask);
3756 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MIME], "");
3759 style = small_style;
3761 gtk_ctree_node_set_row_style(ctree, row, style);
3763 if (MSG_GET_COLORLABEL(flags))
3764 summary_set_colorlabel_color(ctree, row, MSG_GET_COLORLABEL_VALUE(flags));
3767 static void summary_mark_row(SummaryView *summaryview, GtkCTreeNode *row)
3769 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3772 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3773 g_return_if_fail(msginfo);
3774 if (MSG_IS_DELETED(msginfo->flags))
3775 summaryview->deleted--;
3776 if (MSG_IS_MOVE(msginfo->flags))
3777 summaryview->moved--;
3778 if (MSG_IS_COPY(msginfo->flags))
3779 summaryview->copied--;
3781 procmsg_msginfo_set_to_folder(msginfo, NULL);
3782 summary_msginfo_change_flags(msginfo, MSG_MARKED, 0, MSG_DELETED,
3783 MSG_MOVE | MSG_COPY | MSG_MOVE_DONE);
3784 summary_set_row_marks(summaryview, row);
3785 debug_print("Message %s/%d is marked\n", msginfo->folder->path, msginfo->msgnum);
3788 static void summary_lock_row(SummaryView *summaryview, GtkCTreeNode *row)
3790 gboolean changed = FALSE;
3791 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3794 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3795 g_return_if_fail(msginfo);
3796 if (MSG_IS_DELETED(msginfo->flags))
3797 summaryview->deleted--;
3798 if (MSG_IS_MOVE(msginfo->flags)) {
3799 summaryview->moved--;
3802 if (MSG_IS_COPY(msginfo->flags)) {
3803 summaryview->copied--;
3806 procmsg_msginfo_set_to_folder(msginfo, NULL);
3807 summary_msginfo_change_flags(msginfo, MSG_LOCKED, 0, MSG_DELETED,
3808 MSG_MOVE | MSG_COPY | MSG_MOVE_DONE);
3810 summary_set_row_marks(summaryview, row);
3811 debug_print("Message %d is locked\n", msginfo->msgnum);
3814 static void summary_unlock_row(SummaryView *summaryview, GtkCTreeNode *row)
3816 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3819 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3820 g_return_if_fail(msginfo);
3821 if (!MSG_IS_LOCKED(msginfo->flags))
3823 procmsg_msginfo_set_to_folder(msginfo, NULL);
3824 summary_msginfo_unset_flags(msginfo, MSG_LOCKED, 0);
3825 summary_set_row_marks(summaryview, row);
3826 debug_print("Message %d is unlocked\n", msginfo->msgnum);
3829 void summary_mark(SummaryView *summaryview)
3831 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3833 gboolean froze = FALSE;
3835 if (summary_is_locked(summaryview))
3837 START_LONG_OPERATION(summaryview, FALSE);
3838 folder_item_set_batch(summaryview->folder_item, TRUE);
3839 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
3840 summary_mark_row(summaryview, GTK_CTREE_NODE(cur->data));
3841 folder_item_set_batch(summaryview->folder_item, FALSE);
3842 END_LONG_OPERATION(summaryview);
3844 summary_status_show(summaryview);
3847 static void summary_mark_row_as_read(SummaryView *summaryview,
3850 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3853 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3854 g_return_if_fail(msginfo);
3856 if(!(MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)))
3859 summary_msginfo_unset_flags(msginfo, MSG_NEW | MSG_UNREAD, 0);
3860 summary_set_row_marks(summaryview, row);
3861 debug_print("Message %d is marked as read\n",
3865 void summary_mark_as_read(SummaryView *summaryview)
3867 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3869 gboolean froze = FALSE;
3871 if (summary_is_locked(summaryview))
3873 START_LONG_OPERATION(summaryview, FALSE);
3874 folder_item_set_batch(summaryview->folder_item, TRUE);
3875 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
3876 summary_mark_row_as_read(summaryview,
3877 GTK_CTREE_NODE(cur->data));
3878 folder_item_set_batch(summaryview->folder_item, FALSE);
3879 END_LONG_OPERATION(summaryview);
3881 summary_status_show(summaryview);
3884 void summary_msgs_lock(SummaryView *summaryview)
3886 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3888 gboolean froze = FALSE;
3890 if (summary_is_locked(summaryview))
3892 START_LONG_OPERATION(summaryview, FALSE);
3893 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
3894 summary_lock_row(summaryview,
3895 GTK_CTREE_NODE(cur->data));
3896 END_LONG_OPERATION(summaryview);
3898 summary_status_show(summaryview);
3901 void summary_msgs_unlock(SummaryView *summaryview)
3903 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3905 gboolean froze = FALSE;
3907 if (summary_is_locked(summaryview))
3909 START_LONG_OPERATION(summaryview, FALSE);
3910 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
3911 summary_unlock_row(summaryview,
3912 GTK_CTREE_NODE(cur->data));
3913 END_LONG_OPERATION(summaryview);
3915 summary_status_show(summaryview);
3918 void summary_mark_all_read(SummaryView *summaryview)
3920 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3923 gboolean froze = FALSE;
3925 if (prefs_common.ask_mark_all_read) {
3926 val = alertpanel_full(_("Mark all as read"),
3927 _("Do you really want to mark all mails in this "
3928 "folder as read ?"), GTK_STOCK_NO, GTK_STOCK_YES, NULL,
3929 TRUE, NULL, ALERT_QUESTION, G_ALERTDEFAULT);
3931 if ((val & ~G_ALERTDISABLE) != G_ALERTALTERNATE)
3933 else if (val & G_ALERTDISABLE)
3934 prefs_common.ask_mark_all_read = FALSE;
3937 if (summary_is_locked(summaryview))
3939 START_LONG_OPERATION(summaryview, TRUE);
3940 folder_item_set_batch(summaryview->folder_item, TRUE);
3941 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
3942 node = gtkut_ctree_node_next(ctree, node))
3943 summary_mark_row_as_read(summaryview, node);
3944 folder_item_set_batch(summaryview->folder_item, FALSE);
3945 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
3946 node = gtkut_ctree_node_next(ctree, node)) {
3947 if (!GTK_CTREE_ROW(node)->expanded)
3948 summary_set_row_marks(summaryview, node);
3950 END_LONG_OPERATION(summaryview);
3952 summary_status_show(summaryview);
3955 void summary_mark_as_spam(SummaryView *summaryview, guint action, GtkWidget *widget)
3957 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3959 gboolean is_spam = action;
3960 GSList *msgs = NULL;
3961 gboolean immediate_exec = prefs_common.immediate_exec;
3962 gboolean moved = FALSE;
3963 gboolean froze = FALSE;
3966 if (summary_is_locked(summaryview))
3969 prefs_common.immediate_exec = FALSE;
3970 START_LONG_OPERATION(summaryview, FALSE);
3971 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next) {
3972 GtkCTreeNode *row = GTK_CTREE_NODE(cur->data);
3973 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, row);
3975 msgs = g_slist_prepend(msgs, msginfo);
3978 if (procmsg_spam_learner_learn(NULL, msgs, is_spam) == 0) {
3979 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next) {
3980 GtkCTreeNode *row = GTK_CTREE_NODE(cur->data);
3981 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, row);
3985 summary_msginfo_change_flags(msginfo, MSG_SPAM, 0, MSG_NEW|MSG_UNREAD, 0);
3986 if (procmsg_spam_get_folder(msginfo) != summaryview->folder_item) {
3987 summary_move_row_to(summaryview, row,
3988 procmsg_spam_get_folder(msginfo));
3992 summary_msginfo_unset_flags(msginfo, MSG_SPAM, 0);
3994 summaryview->display_msg = prefs_common.always_show_msg;
3996 summary_set_row_marks(summaryview, row);
3999 log_error(LOG_PROTOCOL, _("An error happened while learning.\n"));
4002 prefs_common.immediate_exec = immediate_exec;
4004 END_LONG_OPERATION(summaryview);
4006 if (prefs_common.immediate_exec && moved) {
4007 summary_execute(summaryview);
4010 if (!moved && msgs) {
4011 MsgInfo *msginfo = (MsgInfo *)msgs->data;
4012 toolbar_set_learn_button
4013 (summaryview->mainwin->toolbar,
4014 MSG_IS_SPAM(msginfo->flags)?LEARN_HAM:LEARN_SPAM);
4018 summary_status_show(summaryview);
4022 static void summary_mark_row_as_unread(SummaryView *summaryview,
4025 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4028 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4029 g_return_if_fail(msginfo);
4030 if (MSG_IS_DELETED(msginfo->flags)) {
4031 procmsg_msginfo_set_to_folder(msginfo, NULL);
4032 summary_msginfo_unset_flags(msginfo, MSG_DELETED, 0);
4033 summaryview->deleted--;
4036 summary_msginfo_set_flags(msginfo, MSG_UNREAD, 0);
4037 debug_print("Message %d is marked as unread\n",
4040 summary_set_row_marks(summaryview, row);
4043 void summary_mark_as_unread(SummaryView *summaryview)
4045 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4047 gboolean froze = FALSE;
4049 if (summary_is_locked(summaryview))
4051 START_LONG_OPERATION(summaryview, FALSE);
4052 folder_item_set_batch(summaryview->folder_item, TRUE);
4053 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL;
4055 summary_mark_row_as_unread(summaryview,
4056 GTK_CTREE_NODE(cur->data));
4057 folder_item_set_batch(summaryview->folder_item, FALSE);
4058 END_LONG_OPERATION(summaryview);
4060 summary_status_show(summaryview);
4063 static gboolean check_permission(SummaryView *summaryview, MsgInfo * msginfo)
4068 switch (FOLDER_TYPE(summaryview->folder_item->folder)) {
4073 security : checks if one the accounts correspond to
4074 the author of the post
4078 for(cur = account_get_list() ; cur != NULL ; cur = cur->next) {
4079 PrefsAccount * account;
4082 account = cur->data;
4083 if (account->name && *account->name)
4085 g_strdup_printf("%s <%s>",
4090 g_strdup_printf("%s",
4093 if (g_utf8_collate(from_name, msginfo->from) == 0) {
4102 alertpanel_error(_("You're not the author of the article.\n"));
4112 static void summary_delete_row(SummaryView *summaryview, GtkCTreeNode *row)
4114 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4117 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4118 g_return_if_fail(msginfo);
4120 if (MSG_IS_LOCKED(msginfo->flags)) return;
4122 if (MSG_IS_DELETED(msginfo->flags)) return;
4124 if (MSG_IS_MOVE(msginfo->flags))
4125 summaryview->moved--;
4126 if (MSG_IS_COPY(msginfo->flags))
4127 summaryview->copied--;
4129 procmsg_msginfo_set_to_folder(msginfo, NULL);
4130 summary_msginfo_change_flags(msginfo, MSG_DELETED, 0, MSG_MARKED,
4131 MSG_MOVE | MSG_COPY | MSG_MOVE_DONE);
4132 summaryview->deleted++;
4134 if (!prefs_common.immediate_exec &&
4135 !folder_has_parent_of_type(summaryview->folder_item, F_TRASH))
4136 summary_set_row_marks(summaryview, row);
4138 debug_print("Message %s/%d is set to delete\n",
4139 msginfo->folder->path, msginfo->msgnum);
4142 void summary_cancel(SummaryView *summaryview)
4146 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
4147 summaryview->selected);
4148 if (!msginfo) return;
4150 if (!check_permission(summaryview, msginfo))
4153 news_cancel_article(summaryview->folder_item->folder, msginfo);
4155 if (summary_is_locked(summaryview)) return;
4157 summary_lock(summaryview);
4159 summary_freeze(summaryview);
4161 summary_update_status(summaryview);
4162 summary_status_show(summaryview);
4164 summary_thaw(summaryview);
4166 summary_unlock(summaryview);
4169 void summary_delete(SummaryView *summaryview)
4171 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4172 FolderItem *item = summaryview->folder_item;
4174 GtkCTreeNode *sel_last = NULL;
4178 gboolean froze = FALSE;
4182 if (summary_is_locked(summaryview)) return;
4184 if (!summaryview->folder_item) return;
4186 if (!prefs_common.live_dangerously) {
4188 int num = g_list_length(GTK_CLIST(summaryview->ctree)->selection);
4189 buf = g_strdup_printf(ngettext(
4190 "Do you really want to delete the selected message?",
4191 "Do you really want to delete the %d selected messages?", num),
4193 aval = alertpanel(_("Delete message(s)"),
4195 GTK_STOCK_CANCEL, "+"GTK_STOCK_DELETE, NULL);
4197 if (aval != G_ALERTALTERNATE) return;
4200 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL;
4202 GtkCTreeNode *row = GTK_CTREE_NODE(cur->data);
4203 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4204 if (msginfo && msginfo->total_size != 0 &&
4205 msginfo->size != (goffset)msginfo->total_size)
4206 partial_mark_for_delete(msginfo);
4209 main_window_cursor_wait(summaryview->mainwin);
4211 /* next code sets current row focus right. We need to find a row
4212 * that is not deleted. */
4213 START_LONG_OPERATION(summaryview, FALSE);
4214 folder_item_set_batch(summaryview->folder_item, TRUE);
4215 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next) {
4216 sel_last = GTK_CTREE_NODE(cur->data);
4217 summary_delete_row(summaryview, sel_last);
4219 folder_item_set_batch(summaryview->folder_item, FALSE);
4220 END_LONG_OPERATION(summaryview);
4222 node = summary_find_next_msg(summaryview, sel_last);
4224 node = summary_find_prev_msg(summaryview, sel_last);
4226 summary_select_node(summaryview, node, prefs_common.always_show_msg, TRUE);
4228 if (prefs_common.immediate_exec || folder_has_parent_of_type(item, F_TRASH)) {
4229 summary_execute(summaryview);
4230 /* after deleting, the anchor may be at an invalid row
4231 * so reset it to the node we found earlier */
4232 gtk_sctree_set_anchor_row(GTK_SCTREE(ctree), node);
4234 summary_status_show(summaryview);
4237 main_window_cursor_normal(summaryview->mainwin);
4240 void summary_delete_trash(SummaryView *summaryview)
4242 FolderItem *to_folder = NULL;
4244 if (!summaryview->folder_item ||
4245 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
4247 if (NULL != (ac = account_find_from_item(summaryview->folder_item)))
4248 to_folder = account_get_special_folder(ac, F_TRASH);
4250 if (to_folder == NULL)
4251 to_folder = summaryview->folder_item->folder->trash;
4253 if (to_folder == NULL || to_folder == summaryview->folder_item
4254 || folder_has_parent_of_type(summaryview->folder_item, F_TRASH))
4255 summary_delete(summaryview);
4257 summary_move_selected_to(summaryview, to_folder);
4261 static void summary_unmark_row(SummaryView *summaryview, GtkCTreeNode *row)
4263 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4266 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4267 g_return_if_fail(msginfo);
4268 if (MSG_IS_DELETED(msginfo->flags))
4269 summaryview->deleted--;
4270 if (MSG_IS_MOVE(msginfo->flags))
4271 summaryview->moved--;
4272 if (MSG_IS_COPY(msginfo->flags))
4273 summaryview->copied--;
4275 procmsg_msginfo_set_to_folder(msginfo, NULL);
4276 summary_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED,
4277 MSG_MOVE | MSG_COPY | MSG_MOVE_DONE);
4278 summary_set_row_marks(summaryview, row);
4280 debug_print("Message %s/%d is unmarked\n",
4281 msginfo->folder->path, msginfo->msgnum);
4284 void summary_unmark(SummaryView *summaryview)
4286 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4288 gboolean froze = FALSE;
4290 if (summary_is_locked(summaryview))
4292 START_LONG_OPERATION(summaryview, FALSE);
4293 folder_item_set_batch(summaryview->folder_item, TRUE);
4294 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
4295 summary_unmark_row(summaryview, GTK_CTREE_NODE(cur->data));
4296 folder_item_set_batch(summaryview->folder_item, FALSE);
4297 END_LONG_OPERATION(summaryview);
4299 summary_status_show(summaryview);
4302 static void summary_move_row_to(SummaryView *summaryview, GtkCTreeNode *row,
4303 FolderItem *to_folder)
4305 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4308 g_return_if_fail(to_folder != NULL);
4310 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4311 g_return_if_fail(msginfo);
4312 if (MSG_IS_LOCKED(msginfo->flags))
4315 procmsg_msginfo_set_to_folder(msginfo, to_folder);
4316 if (MSG_IS_DELETED(msginfo->flags))
4317 summaryview->deleted--;
4318 if (MSG_IS_COPY(msginfo->flags)) {
4319 summaryview->copied--;
4321 if (!MSG_IS_MOVE(msginfo->flags)) {
4322 summary_msginfo_change_flags(msginfo, 0, MSG_MOVE, MSG_DELETED,
4323 MSG_COPY | MSG_MOVE_DONE);
4324 summaryview->moved++;
4326 summary_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_COPY);
4329 if (!prefs_common.immediate_exec) {
4330 summary_set_row_marks(summaryview, row);
4333 debug_print("Message %d is set to move to %s\n",
4334 msginfo->msgnum, to_folder->path);
4337 void summary_move_selected_to(SummaryView *summaryview, FolderItem *to_folder)
4340 GtkCTreeNode *sel_last = NULL;
4341 gboolean froze = FALSE;
4343 if (!to_folder) return;
4344 if (!summaryview->folder_item ||
4345 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
4347 if (summary_is_locked(summaryview)) return;
4349 if (summaryview->folder_item == to_folder) {
4350 alertpanel_error(_("Destination is same as current folder."));
4354 if (to_folder->no_select) {
4355 alertpanel_error(_("The destination folder can only be used to "
4356 "store subfolders."));
4360 START_LONG_OPERATION(summaryview, FALSE);
4362 for (cur = GTK_CLIST(summaryview->ctree)->selection;
4363 cur != NULL && cur->data != NULL; cur = cur->next) {
4364 sel_last = GTK_CTREE_NODE(cur->data);
4366 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
4368 END_LONG_OPERATION(summaryview);
4370 summaryview->display_msg = (prefs_common.always_show_msg == OPENMSG_ALWAYS) ||
4371 ((prefs_common.always_show_msg == OPENMSG_WHEN_VIEW_VISIBLE &&
4372 messageview_is_visible(summaryview->messageview)));
4374 if (prefs_common.immediate_exec) {
4375 summary_execute(summaryview);
4377 GtkCTreeNode *node = summary_find_next_msg(summaryview, sel_last);
4379 node = summary_find_prev_msg(summaryview, sel_last);
4380 summary_select_node(summaryview, node, summaryview->display_msg, TRUE);
4381 summary_status_show(summaryview);
4384 if (!summaryview->selected) { /* this was the last message */
4385 GtkCTreeNode *node = gtk_ctree_node_nth (GTK_CTREE(summaryview->ctree),
4386 GTK_CLIST(summaryview->ctree)->rows - 1);
4388 summary_select_node(summaryview, node, summaryview->display_msg, TRUE);
4393 void summary_move_to(SummaryView *summaryview)
4395 FolderItem *to_folder;
4397 if (!summaryview->folder_item ||
4398 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
4400 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
4401 FOLDER_SEL_MOVE, NULL, FALSE);
4402 summary_move_selected_to(summaryview, to_folder);
4405 static void summary_copy_row_to(SummaryView *summaryview, GtkCTreeNode *row,
4406 FolderItem *to_folder)
4408 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4411 g_return_if_fail(to_folder != NULL);
4413 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4414 g_return_if_fail(msginfo);
4415 procmsg_msginfo_set_to_folder(msginfo, to_folder);
4416 if (MSG_IS_DELETED(msginfo->flags))
4417 summaryview->deleted--;
4418 if (MSG_IS_MOVE(msginfo->flags)) {
4419 summaryview->moved--;
4422 if (!MSG_IS_COPY(msginfo->flags)) {
4423 summary_msginfo_change_flags(msginfo, 0, MSG_COPY, MSG_DELETED,
4424 MSG_MOVE | MSG_MOVE_DONE);
4425 summaryview->copied++;
4427 summary_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_MOVE);
4429 if (!prefs_common.immediate_exec) {
4430 summary_set_row_marks(summaryview, row);
4433 debug_print("Message %d is set to copy to %s\n",
4434 msginfo->msgnum, to_folder->path);
4437 void summary_copy_selected_to(SummaryView *summaryview, FolderItem *to_folder)
4440 gboolean froze = FALSE;
4442 if (!to_folder) return;
4443 if (!summaryview->folder_item) return;
4445 if (summary_is_locked(summaryview)) return;
4447 if (summaryview->folder_item == to_folder) {
4449 (_("Destination to copy is same as current folder."));
4453 if (to_folder->no_select) {
4454 alertpanel_error(_("The destination folder can only be used to "
4455 "store subfolders."));
4459 START_LONG_OPERATION(summaryview, FALSE);
4461 for (cur = GTK_CLIST(summaryview->ctree)->selection;
4462 cur != NULL && cur->data != NULL; cur = cur->next)
4464 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
4466 END_LONG_OPERATION(summaryview);
4468 if (prefs_common.immediate_exec)
4469 summary_execute(summaryview);
4471 summary_status_show(summaryview);
4475 void summary_copy_to(SummaryView *summaryview)
4477 FolderItem *to_folder;
4479 if (!summaryview->folder_item) return;
4481 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
4482 FOLDER_SEL_COPY, NULL, FALSE);
4483 summary_copy_selected_to(summaryview, to_folder);
4486 void summary_add_address(SummaryView *summaryview)
4488 MsgInfo *msginfo, *full_msginfo;
4490 GtkWidget *image = NULL;
4491 GdkPixbuf *picture = NULL;
4493 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
4494 summaryview->selected);
4495 if (!msginfo || !msginfo->from)
4498 Xstrdup_a(from, msginfo->from, return);
4499 eliminate_address_comment(from);
4500 extract_address(from);
4502 full_msginfo = procmsg_msginfo_get_full_info(msginfo);
4504 full_msginfo->extradata &&
4505 full_msginfo->extradata->face) {
4506 image = face_get_from_header(full_msginfo->extradata->face);
4508 #if HAVE_LIBCOMPFACE
4509 else if (full_msginfo &&
4510 full_msginfo->extradata &&
4511 full_msginfo->extradata->xface) {
4512 image = xface_get_from_header(full_msginfo->extradata->xface,
4513 &summaryview->ctree->style->white,
4514 summaryview->mainwin->window->window);
4517 procmsg_msginfo_free(full_msginfo);
4519 picture = gtk_image_get_pixbuf(GTK_IMAGE(image));
4521 addressbook_add_contact(msginfo->fromname, from, NULL, picture);
4524 gtk_widget_destroy(image);
4527 void summary_select_all(SummaryView *summaryview)
4529 if (!summaryview->folder_item) return;
4531 summary_lock(summaryview);
4532 gtk_clist_select_all(GTK_CLIST(summaryview->ctree));
4533 summary_unlock(summaryview);
4534 summary_status_show(summaryview);
4537 void summary_unselect_all(SummaryView *summaryview)
4539 summary_lock(summaryview);
4540 gtk_sctree_unselect_all(GTK_SCTREE(summaryview->ctree));
4541 summary_unlock(summaryview);
4542 summary_status_show(summaryview);
4545 void summary_select_thread(SummaryView *summaryview, gboolean delete_thread)
4547 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4548 GtkCTreeNode *node = NULL;
4549 gboolean froze = FALSE;
4552 if (!GTK_CLIST(summaryview->ctree)->selection)
4556 START_LONG_OPERATION(summaryview, FALSE);
4557 copy = g_list_copy(GTK_CLIST(summaryview->ctree)->selection);
4558 for (cur = copy; cur != NULL && cur->data != NULL;
4560 node = GTK_CTREE_NODE(cur->data);
4563 while (GTK_CTREE_ROW(node)->parent != NULL)
4564 node = GTK_CTREE_ROW(node)->parent;
4566 gtk_ctree_select_recursive(ctree, node);
4569 END_LONG_OPERATION(summaryview);
4571 if (delete_thread) {
4572 if (FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS)
4573 summary_delete(summaryview);
4575 summary_delete_trash(summaryview);
4577 summary_status_show(summaryview);
4580 void summary_save_as(SummaryView *summaryview)
4582 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4584 gchar *filename = NULL;
4588 AlertValue aval = 0;
4590 if (!summaryview->selected) return;
4591 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
4592 if (!msginfo) return;
4594 if (msginfo->subject) {
4595 Xstrdup_a(filename, msginfo->subject, return);
4596 subst_for_filename(filename);
4598 if (filename && !g_utf8_validate(filename, -1, NULL)) {
4599 gchar *oldstr = filename;
4600 filename = conv_codeset_strdup(filename,
4601 conv_get_locale_charset_str(),
4604 g_warning("summary_save_as(): failed to convert character set.");
4605 filename = g_strdup(oldstr);
4607 dest = filesel_select_file_save(_("Save as"), filename);
4610 dest = filesel_select_file_save(_("Save as"), filename);
4613 if (is_file_exist(dest)) {
4614 aval = alertpanel(_("Append or Overwrite"),
4615 _("Append or overwrite existing file?"),
4616 _("_Append"), _("_Overwrite"),
4618 if (aval != 0 && aval != 1)
4622 src = procmsg_get_message_file(msginfo);
4623 tmp = g_path_get_basename(dest);
4625 if ( aval==0 ) { /* append */
4626 if (append_file(src, dest, TRUE) < 0)
4627 alertpanel_error(_("Couldn't save the file '%s'."), tmp);
4628 } else { /* overwrite */
4629 if (copy_file(src, dest, TRUE) < 0)
4630 alertpanel_error(_("Couldn't save the file '%s'."), tmp);
4635 * If two or more msgs are selected,
4636 * append them to the output file.
4638 if (GTK_CLIST(ctree)->selection->next) {
4640 for (item = GTK_CLIST(ctree)->selection->next; item != NULL; item=item->next) {
4641 msginfo = gtk_ctree_node_get_row_data(ctree, GTK_CTREE_NODE(item->data));
4642 if (!msginfo) break;
4643 src = procmsg_get_message_file(msginfo);
4644 if (append_file(src, dest, TRUE) < 0)
4645 alertpanel_error(_("Couldn't save the file '%s'."), tmp);
4653 void summary_print(SummaryView *summaryview)
4655 GtkCList *clist = GTK_CLIST(summaryview->ctree);
4656 #if !defined(USE_GNOMEPRINT) && !GTK_CHECK_VERSION(2,10,0)
4657 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4659 gchar *cmdline = NULL;
4663 gchar *msg = g_strdup_printf(_("You are about to print %d "
4664 "messages, one by one. Do you "
4665 "want to continue?"),
4666 g_list_length(clist->selection));
4667 if (g_list_length(clist->selection) > 9
4668 && alertpanel(_("Warning"), msg, GTK_STOCK_CANCEL, "+" GTK_STOCK_YES, NULL)
4669 != G_ALERTALTERNATE) {
4675 if (clist->selection == NULL) return;
4676 #if !defined(USE_GNOMEPRINT) && !GTK_CHECK_VERSION(2,10,0)
4677 cmdline = input_dialog(_("Print"),
4678 _("Enter the print command line:\n"
4679 "('%s' will be replaced with file name)"),
4680 prefs_common.print_cmd);
4681 if (!cmdline) return;
4682 if (!(p = strchr(cmdline, '%')) || *(p + 1) != 's' ||
4683 strchr(p + 2, '%')) {
4684 alertpanel_error(_("Print command line is invalid:\n'%s'"),
4689 for (cur = clist->selection;
4690 cur != NULL && cur->data != NULL;
4692 msginfo = gtk_ctree_node_get_row_data
4693 (ctree, GTK_CTREE_NODE(cur->data));
4695 procmsg_print_message(msginfo, cmdline);
4700 for (cur = clist->selection;
4701 cur != NULL && cur->data != NULL;
4703 GtkCTreeNode *node = GTK_CTREE_NODE(cur->data);
4704 MsgInfo *msginfo = gtk_ctree_node_get_row_data(
4705 GTK_CTREE(summaryview->ctree),
4707 gint sel_start = -1, sel_end = -1, partnum = 0;
4709 if (node == summaryview->displayed) {
4710 partnum = mimeview_get_selected_part_num(summaryview->messageview->mimeview);
4711 textview_get_selection_offsets(summaryview->messageview->mimeview->textview,
4712 &sel_start, &sel_end);
4714 messageview_print(msginfo, summaryview->messageview->all_headers,
4715 sel_start, sel_end, partnum);
4720 gboolean summary_execute(SummaryView *summaryview)
4722 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4723 GtkCList *clist = GTK_CLIST(summaryview->ctree);
4724 GtkCTreeNode *node, *next;
4725 GtkCTreeNode *new_selected = NULL;
4728 if (!summaryview->folder_item) return FALSE;
4730 if (summary_is_locked(summaryview)) return FALSE;
4731 summary_lock(summaryview);
4733 summary_freeze(summaryview);
4735 main_window_cursor_wait(summaryview->mainwin);
4737 if (summaryview->threaded)
4738 summary_unthread_for_exec(summaryview);
4740 folder_item_update_freeze();
4741 move_val = summary_execute_move(summaryview);
4742 summary_execute_copy(summaryview);
4743 summary_execute_delete(summaryview);
4745 node = GTK_CTREE_NODE(clist->row_list);
4746 for (; node != NULL; node = next) {
4747 next = gtkut_ctree_node_next(ctree, node);
4748 if (gtk_ctree_node_get_row_data(ctree, node) != NULL) continue;
4750 if (node == summaryview->displayed) {
4751 messageview_clear(summaryview->messageview);
4752 summaryview->displayed = NULL;
4754 if (GTK_CTREE_ROW(node)->children != NULL) {
4756 if (GTK_CTREE_ROW(node)->sibling) {
4757 next = GTK_CTREE_ROW(node)->sibling;
4759 GtkCTreeNode *parent = NULL;
4760 for (parent = GTK_CTREE_ROW(node)->parent; parent != NULL;
4761 parent = GTK_CTREE_ROW(parent)->parent) {
4762 if (GTK_CTREE_ROW(parent)->sibling) {
4763 next = GTK_CTREE_ROW(parent)->sibling;
4769 if (!new_selected &&
4770 gtkut_ctree_node_is_selected(ctree, node)) {
4771 summary_unselect_all(summaryview);
4772 new_selected = summary_find_next_msg(summaryview, node);
4774 new_selected = summary_find_prev_msg
4775 (summaryview, node);
4778 gtk_sctree_remove_node((GtkSCTree *)ctree, node);
4781 folder_item_update_thaw();
4784 summary_unlock(summaryview);
4786 (GTK_SCTREE(ctree), new_selected);
4787 summary_lock(summaryview);
4790 if (summaryview->threaded) {
4791 summary_thread_build(summaryview);
4792 summary_thread_init(summaryview);
4795 summary_thaw(summaryview);
4797 summaryview->selected = clist->selection ?
4798 GTK_CTREE_NODE(clist->selection->data) : NULL;
4800 if (!GTK_CLIST(summaryview->ctree)->row_list) {
4801 menu_set_insensitive_all
4802 (GTK_MENU_SHELL(summaryview->popupmenu));
4803 gtk_widget_grab_focus(summaryview->folderview->ctree);
4805 gtk_widget_grab_focus(summaryview->ctree);
4807 summary_update_status(summaryview);
4808 summary_status_show(summaryview);
4810 gtk_ctree_node_moveto(ctree, summaryview->selected, 0, 0.5, 0);
4812 summary_unlock(summaryview);
4814 main_window_cursor_normal(summaryview->mainwin);
4817 summary_show(summaryview, summaryview->folder_item);
4821 static gint summary_execute_move(SummaryView *summaryview)
4823 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4826 /* search moving messages and execute */
4827 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_move_func,
4830 if (summaryview->mlist) {
4831 hooks_unregister_hook(MSGINFO_UPDATE_HOOKLIST,
4832 summaryview->msginfo_update_callback_id);
4833 val = procmsg_move_messages(summaryview->mlist);
4834 summaryview->msginfo_update_callback_id =
4835 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST,
4836 summary_update_msg, (gpointer) summaryview);
4838 for (cur = summaryview->mlist; cur != NULL && cur->data != NULL; cur = cur->next)
4839 procmsg_msginfo_free((MsgInfo *)cur->data);
4840 g_slist_free(summaryview->mlist);
4841 summaryview->mlist = NULL;
4847 static void summary_execute_move_func(GtkCTree *ctree, GtkCTreeNode *node,
4850 SummaryView *summaryview = data;
4853 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
4855 if (msginfo && MSG_IS_MOVE(msginfo->flags) && msginfo->to_folder) {
4856 summaryview->mlist =
4857 g_slist_prepend(summaryview->mlist, msginfo);
4858 gtk_ctree_node_set_row_data(ctree, node, NULL);
4860 if (msginfo->msgid && *msginfo->msgid &&
4861 node == g_hash_table_lookup(summaryview->msgid_table,
4863 g_hash_table_remove(summaryview->msgid_table,
4865 if (prefs_common.thread_by_subject &&
4866 msginfo->subject && *msginfo->subject &&
4867 node == subject_table_lookup(summaryview->subject_table,
4868 msginfo->subject)) {
4869 subject_table_remove(summaryview->subject_table,
4875 static void summary_execute_copy(SummaryView *summaryview)
4877 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4879 /* search copying messages and execute */
4880 hooks_unregister_hook(MSGINFO_UPDATE_HOOKLIST,
4881 summaryview->msginfo_update_callback_id);
4882 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_copy_func,
4885 if (summaryview->mlist) {
4886 summaryview->mlist = g_slist_reverse(summaryview->mlist);
4887 procmsg_copy_messages(summaryview->mlist);
4889 g_slist_free(summaryview->mlist);
4890 summaryview->mlist = NULL;
4892 summaryview->msginfo_update_callback_id =
4893 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST,
4894 summary_update_msg, (gpointer) summaryview);
4897 static void summary_execute_copy_func(GtkCTree *ctree, GtkCTreeNode *node,
4900 SummaryView *summaryview = data;
4903 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
4905 if (msginfo && MSG_IS_COPY(msginfo->flags) && msginfo->to_folder) {
4906 summaryview->mlist =
4907 g_slist_prepend(summaryview->mlist, msginfo);
4909 summary_msginfo_unset_flags(msginfo, 0, MSG_COPY);
4910 summary_set_row_marks(summaryview, node);
4914 static void summary_execute_delete(SummaryView *summaryview)
4916 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4919 /* search deleting messages and execute */
4920 gtk_ctree_pre_recursive
4921 (ctree, NULL, summary_execute_delete_func, summaryview);
4923 if (!summaryview->mlist) return;
4925 hooks_unregister_hook(MSGINFO_UPDATE_HOOKLIST,
4926 summaryview->msginfo_update_callback_id);
4928 folder_item_remove_msgs(summaryview->folder_item,
4929 summaryview->mlist);
4931 summaryview->msginfo_update_callback_id =
4932 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST,
4933 summary_update_msg, (gpointer) summaryview);
4934 for (cur = summaryview->mlist; cur != NULL && cur->data != NULL; cur = cur->next)
4935 procmsg_msginfo_free((MsgInfo *)cur->data);
4937 g_slist_free(summaryview->mlist);
4938 summaryview->mlist = NULL;
4941 static void summary_execute_delete_func(GtkCTree *ctree, GtkCTreeNode *node,
4944 SummaryView *summaryview = data;
4947 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
4949 if (msginfo && MSG_IS_DELETED(msginfo->flags)) {
4950 summaryview->mlist =
4951 g_slist_prepend(summaryview->mlist, msginfo);
4952 gtk_ctree_node_set_row_data(ctree, node, NULL);
4954 if (msginfo->msgid && *msginfo->msgid &&
4955 node == g_hash_table_lookup(summaryview->msgid_table,
4957 g_hash_table_remove(summaryview->msgid_table,
4960 if (prefs_common.thread_by_subject &&
4961 msginfo->subject && *msginfo->subject &&
4962 node == subject_table_lookup(summaryview->subject_table,
4963 msginfo->subject)) {
4964 subject_table_remove(summaryview->subject_table,
4970 /* thread functions */
4972 static void summary_thread_build(SummaryView *summaryview)
4974 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4977 GtkCTreeNode *parent;
4981 summary_lock(summaryview);
4983 debug_print("Building threads...");
4984 STATUSBAR_PUSH(summaryview->mainwin, _("Building threads..."));
4985 main_window_cursor_wait(summaryview->mainwin);
4987 g_signal_handlers_block_by_func(G_OBJECT(ctree),
4988 G_CALLBACK(summary_tree_expanded), summaryview);
4989 summary_freeze(summaryview);
4991 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
4993 next = GTK_CTREE_ROW(node)->sibling;
4995 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
4999 if (msginfo && msginfo->inreplyto) {
5000 parent = g_hash_table_lookup(summaryview->msgid_table,
5001 msginfo->inreplyto);
5003 if (!parent && msginfo->references) {
5004 for (reflist = msginfo->references;
5005 reflist != NULL; reflist = reflist->next)
5006 if ((parent = g_hash_table_lookup
5007 (summaryview->msgid_table,
5013 if (prefs_common.thread_by_subject && parent == NULL) {
5014 parent = subject_table_lookup
5015 (summaryview->subject_table,
5019 if (parent && parent != node && parent != GTK_CTREE_ROW(node)->parent) {
5020 gtk_ctree_move(ctree, node, parent, NULL);
5026 gtkut_ctree_set_focus_row(ctree, summaryview->selected);
5028 summary_thaw(summaryview);
5029 g_signal_handlers_unblock_by_func(G_OBJECT(ctree),
5030 G_CALLBACK(summary_tree_expanded), summaryview);
5032 debug_print("done.\n");
5033 STATUSBAR_POP(summaryview->mainwin);
5034 main_window_cursor_normal(summaryview->mainwin);
5036 summaryview->threaded = TRUE;
5038 summary_unlock(summaryview);
5041 static void summary_thread_init(SummaryView *summaryview)
5043 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5044 GtkCTreeNode *node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
5047 if (!summaryview->thread_collapsed) {
5048 g_signal_handlers_block_by_func(G_OBJECT(ctree),
5049 G_CALLBACK(summary_tree_expanded), summaryview);
5051 next = GTK_CTREE_ROW(node)->sibling;
5052 if (GTK_CTREE_ROW(node)->children)
5053 gtk_ctree_expand_recursive(ctree, node);
5056 g_signal_handlers_unblock_by_func(G_OBJECT(ctree),
5057 G_CALLBACK(summary_tree_expanded), summaryview);
5062 static void summary_unthread_for_exec(SummaryView *summaryview)
5065 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5066 gboolean froze = FALSE;
5068 debug_print("Unthreading for execution...");
5070 START_LONG_OPERATION(summaryview, TRUE);
5071 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
5072 node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
5073 summary_unthread_for_exec_func(ctree, node, summaryview);
5076 END_LONG_OPERATION(summaryview);
5078 debug_print("done.\n");
5081 static void summary_unthread_for_exec_func(GtkCTree *ctree, GtkCTreeNode *node,
5085 GtkCTreeNode *top_parent;
5086 GtkCTreeNode *child;
5087 GtkCTreeNode *sibling;
5088 SummaryView * summaryview = (SummaryView *)data;
5089 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
5092 (!MSG_IS_MOVE(msginfo->flags) &&
5093 !MSG_IS_DELETED(msginfo->flags)))
5095 child = GTK_CTREE_ROW(node)->children;
5098 if (node == summaryview->selected)
5099 summaryview->selected = NULL;
5100 if (node == summaryview->displayed)
5101 summaryview->displayed = NULL;
5103 for (top_parent = node;
5104 GTK_CTREE_ROW(top_parent)->parent != NULL;
5105 top_parent = GTK_CTREE_ROW(top_parent)->parent)
5107 sibling = GTK_CTREE_ROW(top_parent)->sibling;
5109 GTK_SCTREE(ctree)->sorting = TRUE;
5110 while (child != NULL) {
5111 GtkCTreeNode *next_child;
5112 MsgInfo *cinfo = GTKUT_CTREE_NODE_GET_ROW_DATA(child);
5114 next_child = GTK_CTREE_ROW(child)->sibling;
5116 if (!MSG_IS_MOVE(cinfo->flags) && !MSG_IS_DELETED(cinfo->flags)) {
5117 gtk_ctree_move(ctree, child,
5121 if (child == summaryview->displayed) {
5122 messageview_clear(summaryview->messageview);
5123 summaryview->displayed = NULL;
5125 if (child == summaryview->selected) {
5126 messageview_clear(summaryview->messageview);
5127 summaryview->selected = NULL;
5132 GTK_SCTREE(ctree)->sorting = FALSE;
5135 void summary_expand_threads(SummaryView *summaryview)
5137 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5138 GtkCTreeNode *node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
5140 g_signal_handlers_block_by_func(G_OBJECT(ctree),
5141 G_CALLBACK(summary_tree_expanded), summaryview);
5142 summary_freeze(summaryview);
5145 if (GTK_CTREE_ROW(node)->children) {
5146 gtk_ctree_expand(ctree, node);
5147 summary_set_row_marks(summaryview, node);
5149 node = GTK_CTREE_NODE_NEXT(node);
5152 summary_thaw(summaryview);
5153 g_signal_handlers_unblock_by_func(G_OBJECT(ctree),
5154 G_CALLBACK(summary_tree_expanded), summaryview);
5156 summaryview->thread_collapsed = FALSE;
5158 gtk_ctree_node_moveto(ctree, summaryview->selected, 0, 0.5, 0);
5161 void summary_collapse_threads(SummaryView *summaryview)
5163 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5164 GtkCTreeNode *node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
5166 summary_freeze(summaryview);
5169 if (GTK_CTREE_ROW(node)->children)
5170 gtk_ctree_collapse(ctree, node);
5171 node = GTK_CTREE_ROW(node)->sibling;
5174 summary_thaw(summaryview);
5176 summaryview->thread_collapsed = TRUE;
5178 gtk_ctree_node_moveto(ctree, summaryview->selected, 0, 0.5, 0);
5181 static void account_rules_radio_button_toggled_cb(GtkToggleButton *btn, gpointer data)
5183 prefs_common.apply_per_account_filtering_rules = GPOINTER_TO_INT(data);
5186 static gboolean summary_filter_get_mode(void)
5187 /* ask what to do w/ them: skip them, apply them regardless to the account,
5188 use the current account */
5190 /* TODO: eventually also propose to use the current folder's default account,
5192 /* TODO: eventually allow to select the account to use from a optmenu */
5195 GtkWidget *account_rules_skip;
5196 GtkWidget *account_rules_force;
5197 GtkWidget *account_rules_user_current;
5200 vbox = gtk_vbox_new (FALSE, 0);
5202 account_rules_skip = gtk_radio_button_new_with_label
5203 (NULL, _("Skip these rules"));
5204 account_rules_force = gtk_radio_button_new_with_label_from_widget
5205 (GTK_RADIO_BUTTON(account_rules_skip),
5206 _("Apply these rules regardless of the account they belong to"));
5207 account_rules_user_current = gtk_radio_button_new_with_label_from_widget
5208 (GTK_RADIO_BUTTON(account_rules_skip),
5209 _("Apply these rules if they apply to the current account"));
5210 gtk_box_pack_start (GTK_BOX (vbox), account_rules_skip, FALSE, FALSE, 0);
5211 gtk_box_pack_start (GTK_BOX (vbox), account_rules_force, FALSE, FALSE, 0);
5212 gtk_box_pack_start (GTK_BOX (vbox), account_rules_user_current, FALSE, FALSE, 0);
5213 g_signal_connect(G_OBJECT(account_rules_skip), "toggled",
5214 G_CALLBACK(account_rules_radio_button_toggled_cb),
5215 GINT_TO_POINTER(FILTERING_ACCOUNT_RULES_SKIP));
5216 g_signal_connect(G_OBJECT(account_rules_force), "toggled",
5217 G_CALLBACK(account_rules_radio_button_toggled_cb),
5218 GINT_TO_POINTER(FILTERING_ACCOUNT_RULES_FORCE));
5219 g_signal_connect(G_OBJECT(account_rules_user_current), "toggled",
5220 G_CALLBACK(account_rules_radio_button_toggled_cb),
5221 GINT_TO_POINTER(FILTERING_ACCOUNT_RULES_USE_CURRENT));
5222 switch (prefs_common.apply_per_account_filtering_rules) {
5223 case FILTERING_ACCOUNT_RULES_SKIP:
5224 gtk_toggle_button_set_active(
5225 GTK_TOGGLE_BUTTON(account_rules_skip), TRUE);
5227 case FILTERING_ACCOUNT_RULES_FORCE:
5228 gtk_toggle_button_set_active(
5229 GTK_TOGGLE_BUTTON(account_rules_force), TRUE);
5231 case FILTERING_ACCOUNT_RULES_USE_CURRENT:
5232 gtk_toggle_button_set_active(
5233 GTK_TOGGLE_BUTTON(account_rules_user_current), TRUE);
5237 val = alertpanel_with_widget(
5239 _("There are some filtering rules that belong to an account.\n"
5240 "Please choose what to do with these rules:"),
5241 GTK_STOCK_CANCEL, _("_Filter"), NULL, TRUE, G_ALERTALTERNATE, vbox);
5243 if ((val & ~G_ALERTDISABLE) != G_ALERTALTERNATE) {
5245 } else if (val & G_ALERTDISABLE)
5246 prefs_common.ask_apply_per_account_filtering_rules = FALSE;
5251 void summary_filter(SummaryView *summaryview, gboolean selected_only)
5253 GSList *mlist = NULL, *cur_list;
5254 summary_lock(summaryview);
5256 /* are there any per-account filtering rules? */
5257 if (prefs_common.ask_apply_per_account_filtering_rules == TRUE &&
5258 filtering_peek_per_account_rules(filtering_rules)) {
5260 if (summary_filter_get_mode() == FALSE) {
5261 summary_unlock(summaryview);
5266 folder_item_update_freeze();
5268 debug_print("filtering...");
5269 STATUSBAR_PUSH(summaryview->mainwin, _("Filtering..."));
5270 main_window_cursor_wait(summaryview->mainwin);
5272 summary_freeze(summaryview);
5274 if (selected_only) {
5277 for (cur = GTK_CLIST(summaryview->ctree)->selection;
5278 cur != NULL && cur->data != NULL; cur = cur->next) {
5279 mlist = g_slist_prepend(mlist,
5280 procmsg_msginfo_new_ref(
5281 GTKUT_CTREE_NODE_GET_ROW_DATA(cur->data)));
5283 mlist = g_slist_reverse(mlist);
5285 mlist = folder_item_get_msg_list(summaryview->folder_item);
5288 folder_item_set_batch(summaryview->folder_item, TRUE);
5289 for (cur_list = mlist; cur_list; cur_list = cur_list->next) {
5290 summary_filter_func((MsgInfo *)cur_list->data);
5292 folder_item_set_batch(summaryview->folder_item, FALSE);
5294 filtering_move_and_copy_msgs(mlist);
5296 for (cur_list = mlist; cur_list; cur_list = cur_list->next) {
5297 procmsg_msginfo_free((MsgInfo *)cur_list->data);
5299 g_slist_free(mlist);
5301 summary_thaw(summaryview);
5303 folder_item_update_thaw();
5304 debug_print("done.\n");
5305 STATUSBAR_POP(summaryview->mainwin);
5306 main_window_cursor_normal(summaryview->mainwin);
5308 summary_unlock(summaryview);
5311 * CLAWS: summary_show() only valid after having a lock. ideally
5312 * we want the lock to be context aware...
5314 summary_show(summaryview, summaryview->folder_item);
5317 static void summary_filter_func(MsgInfo *msginfo)
5319 MailFilteringData mail_filtering_data;
5321 mail_filtering_data.msginfo = msginfo;
5322 mail_filtering_data.msglist = NULL;
5323 mail_filtering_data.filtered = NULL;
5324 mail_filtering_data.unfiltered = NULL;
5325 if (hooks_invoke(MAIL_MANUAL_FILTERING_HOOKLIST, &mail_filtering_data))
5328 filter_message_by_msginfo(filtering_rules, msginfo, NULL,
5329 FILTERING_MANUALLY, NULL);
5332 void summary_msginfo_filter_open(FolderItem * item, MsgInfo *msginfo,
5333 PrefsFilterType type, gint processing_rule)
5335 gchar *header = NULL;
5338 procmsg_get_filter_keyword(msginfo, &header, &key, type);
5340 if (processing_rule) {
5342 prefs_filtering_open(&pre_global_processing,
5343 _("Processing rules to apply before folder rules"),
5344 MANUAL_ANCHOR_PROCESSING,
5345 header, key, FALSE);
5347 prefs_filtering_open(&item->prefs->processing,
5348 _("Processing configuration"),
5349 MANUAL_ANCHOR_PROCESSING,
5350 header, key, FALSE);
5353 prefs_filtering_open(&filtering_rules,
5354 _("Filtering configuration"),
5355 MANUAL_ANCHOR_FILTERING,
5363 void summary_filter_open(SummaryView *summaryview, PrefsFilterType type,
5364 gint processing_rule)
5369 if (!summaryview->selected) return;
5371 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
5372 summaryview->selected);
5373 if (!msginfo) return;
5375 item = summaryview->folder_item;
5376 summary_msginfo_filter_open(item, msginfo, type, processing_rule);
5381 #define N_COLOR_LABELS colorlabel_get_color_count()
5383 static void summary_colorlabel_menu_item_activate_cb(GtkWidget *widget,
5386 guint color = GPOINTER_TO_UINT(data);
5387 SummaryView *summaryview;
5389 summaryview = g_object_get_data(G_OBJECT(widget), "summaryview");
5390 g_return_if_fail(summaryview != NULL);
5392 /* "dont_toggle" state set? */
5393 if (g_object_get_data(G_OBJECT(summaryview->colorlabel_menu),
5397 summary_set_colorlabel(summaryview, color, NULL);
5400 /* summary_set_colorlabel_color() - labelcolor parameter is the color *flag*
5401 * for the messsage; not the color index */
5402 void summary_set_colorlabel_color(GtkCTree *ctree, GtkCTreeNode *node,
5406 GtkStyle *style, *prev_style, *ctree_style;
5410 msginfo = gtk_ctree_node_get_row_data(ctree, node);
5411 g_return_if_fail(msginfo);
5413 color_index = labelcolor == 0 ? -1 : (gint)labelcolor - 1;
5414 ctree_style = gtk_widget_get_style(GTK_WIDGET(ctree));
5415 prev_style = gtk_ctree_node_get_row_style(ctree, node);
5417 if (color_index < 0 || color_index >= N_COLOR_LABELS) {
5418 if (!prev_style) return;
5419 style = gtk_style_copy(prev_style);
5420 color = ctree_style->fg[GTK_STATE_NORMAL];
5421 style->fg[GTK_STATE_NORMAL] = color;
5422 color = ctree_style->fg[GTK_STATE_SELECTED];
5423 style->fg[GTK_STATE_SELECTED] = color;
5426 style = gtk_style_copy(prev_style);
5428 style = gtk_style_copy(ctree_style);
5429 color = colorlabel_get_color(color_index);
5430 style->fg[GTK_STATE_NORMAL] = color;
5431 /* get the average of label color and selected fg color
5433 style->fg[GTK_STATE_SELECTED].red = (color.red + 3*ctree_style->fg[GTK_STATE_SELECTED].red ) / 4;
5434 style->fg[GTK_STATE_SELECTED].green = (color.green + 3*ctree_style->fg[GTK_STATE_SELECTED].green) / 4;
5435 style->fg[GTK_STATE_SELECTED].blue = (color.blue + 3*ctree_style->fg[GTK_STATE_SELECTED].blue ) / 4;
5438 gtk_ctree_node_set_row_style(ctree, node, style);
5439 gtk_style_unref(style);
5442 static void summary_set_row_colorlabel(SummaryView *summaryview, GtkCTreeNode *row, guint labelcolor)
5444 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5447 msginfo = gtk_ctree_node_get_row_data(ctree, row);
5448 g_return_if_fail(msginfo);
5450 summary_msginfo_change_flags(msginfo, MSG_COLORLABEL_TO_FLAGS(labelcolor), 0,
5451 MSG_CLABEL_FLAG_MASK, 0);
5452 summary_set_row_marks(summaryview, row);
5455 void summary_set_colorlabel(SummaryView *summaryview, guint labelcolor,
5458 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5460 gboolean froze = FALSE;
5462 START_LONG_OPERATION(summaryview, FALSE);
5463 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
5464 summary_set_row_colorlabel(summaryview,
5465 GTK_CTREE_NODE(cur->data), labelcolor);
5466 END_LONG_OPERATION(summaryview);
5469 static gboolean summary_set_row_tag(SummaryView *summaryview, GtkCTreeNode *row, gboolean refresh, gboolean set, gint id)
5471 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5473 gchar *tags_str = NULL;
5474 msginfo = gtk_ctree_node_get_row_data(ctree, row);
5475 g_return_val_if_fail(msginfo, FALSE);
5477 procmsg_msginfo_update_tags(msginfo, set, id);
5479 if (summaryview->col_state[summaryview->col_pos[S_COL_TAGS]].visible) {
5480 tags_str = procmsg_msginfo_get_tags_str(msginfo);
5481 gtk_ctree_node_set_text(ctree, row,
5482 summaryview->col_pos[S_COL_TAGS],
5483 tags_str?tags_str:"-");
5487 summary_set_row_marks(summaryview, row);
5488 if (row == summaryview->displayed) {
5494 void summary_set_tag(SummaryView *summaryview, gint tag_id,
5497 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
5499 gboolean set = tag_id > 0;
5500 gint real_id = set? tag_id:-tag_id;
5501 gboolean froze = FALSE;
5502 gboolean redisplay = FALSE;
5504 if (summary_is_locked(summaryview))
5506 START_LONG_OPERATION(summaryview, FALSE);
5507 folder_item_set_batch(summaryview->folder_item, TRUE);
5508 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next) {
5509 redisplay |= summary_set_row_tag(summaryview,
5510 GTK_CTREE_NODE(cur->data), FALSE, set, real_id);
5512 folder_item_set_batch(summaryview->folder_item, FALSE);
5513 END_LONG_OPERATION(summaryview);
5515 summary_redisplay_msg(summaryview);
5518 static void summary_tags_menu_item_activate_cb(GtkWidget *widget,
5521 gint id = GPOINTER_TO_INT(data);
5522 gboolean set = gtk_check_menu_item_get_active(GTK_CHECK_MENU_ITEM(widget));
5523 SummaryView *summaryview;
5525 summaryview = g_object_get_data(G_OBJECT(widget), "summaryview");
5526 g_return_if_fail(summaryview != NULL);
5528 /* "dont_toggle" state set? */
5529 if (g_object_get_data(G_OBJECT(summaryview->tags_menu),
5535 summary_set_tag(summaryview, id, NULL);
5538 static void summary_colorlabel_menu_item_activate_item_cb(GtkMenuItem *menu_item,
5541 SummaryView *summaryview;
5543 GtkCheckMenuItem **items;
5547 summaryview = (SummaryView *)data;
5548 g_return_if_fail(summaryview != NULL);
5550 sel = GTK_CLIST(summaryview->ctree)->selection;
5553 menu = GTK_MENU_SHELL(summaryview->colorlabel_menu);
5555 g_return_if_fail(menu != NULL);
5557 Xalloca(items, (N_COLOR_LABELS + 1) * sizeof(GtkWidget *), return);
5559 /* NOTE: don't return prematurely because we set the "dont_toggle"
5560 * state for check menu items */
5561 g_object_set_data(G_OBJECT(menu), "dont_toggle",
5562 GINT_TO_POINTER(1));
5564 /* clear items. get item pointers. */
5565 for (n = 0, cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
5566 if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
5567 gtk_check_menu_item_set_active
5568 (GTK_CHECK_MENU_ITEM(cur->data), FALSE);
5569 items[n] = GTK_CHECK_MENU_ITEM(cur->data);
5574 if (n == (N_COLOR_LABELS + 1)) {
5575 /* iterate all messages and set the state of the appropriate
5577 for (; sel != NULL; sel = sel->next) {
5581 msginfo = gtk_ctree_node_get_row_data
5582 (GTK_CTREE(summaryview->ctree),
5583 GTK_CTREE_NODE(sel->data));
5585 clabel = MSG_GET_COLORLABEL_VALUE(msginfo->flags);
5586 if (!items[clabel]->active)
5587 gtk_check_menu_item_set_active
5588 (items[clabel], TRUE);
5592 g_warning("invalid number of color elements (%d)\n", n);
5594 /* reset "dont_toggle" state */
5595 g_object_set_data(G_OBJECT(menu), "dont_toggle",
5596 GINT_TO_POINTER(0));
5599 static void summary_colorlabel_menu_create(SummaryView *summaryview, gboolean refresh)
5601 GtkWidget *label_menuitem;
5606 label_menuitem = gtk_item_factory_get_item(summaryview->popupfactory,
5608 g_signal_connect(G_OBJECT(label_menuitem), "activate",
5609 G_CALLBACK(summary_colorlabel_menu_item_activate_item_cb),
5611 gtk_widget_show(label_menuitem);
5613 menu = gtk_menu_new();
5615 /* create sub items. for the menu item activation callback we pass the
5616 * index of label_colors[] as data parameter. for the None color we
5617 * pass an invalid (high) value. also we attach a data pointer so we
5618 * can always get back the SummaryView pointer. */
5620 item = gtk_check_menu_item_new_with_label(_("None"));
5621 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5622 g_signal_connect(G_OBJECT(item), "activate",
5623 G_CALLBACK(summary_colorlabel_menu_item_activate_cb),
5624 GUINT_TO_POINTER(0));
5625 g_object_set_data(G_OBJECT(item), "summaryview", summaryview);
5626 gtk_widget_show(item);
5628 gtk_widget_add_accelerator(item, "activate",
5629 summaryview->popupfactory->accel_group,
5630 GDK_0, GDK_CONTROL_MASK,
5631 GTK_ACCEL_LOCKED | GTK_ACCEL_VISIBLE);
5633 item = gtk_menu_item_new();
5634 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5635 gtk_widget_show(item);
5637 /* create pixmap/label menu items */
5638 for (i = 0; i < N_COLOR_LABELS; i++) {
5639 item = colorlabel_create_check_color_menu_item(
5640 i, refresh, SUMMARY_COLORMENU);
5641 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5642 g_signal_connect(G_OBJECT(item), "activate",
5643 G_CALLBACK(summary_colorlabel_menu_item_activate_cb),
5644 GUINT_TO_POINTER(i + 1));
5645 g_object_set_data(G_OBJECT(item), "summaryview",
5647 gtk_widget_show(item);
5650 gtk_widget_show(menu);
5651 gtk_menu_item_set_submenu(GTK_MENU_ITEM(label_menuitem), menu);
5652 summaryview->colorlabel_menu = menu;
5655 static void summary_tags_menu_item_activate_item_cb(GtkMenuItem *menu_item,
5661 GHashTable *menu_table = g_hash_table_new_full(
5665 GHashTable *menu_allsel_table = g_hash_table_new_full(
5670 SummaryView *summaryview = (SummaryView *)data;
5671 g_return_if_fail(summaryview != NULL);
5673 sel = GTK_CLIST(summaryview->ctree)->selection;
5676 menu = GTK_MENU_SHELL(summaryview->tags_menu);
5677 g_return_if_fail(menu != NULL);
5679 /* NOTE: don't return prematurely because we set the "dont_toggle"
5680 * state for check menu items */
5681 g_object_set_data(G_OBJECT(menu), "dont_toggle",
5682 GINT_TO_POINTER(1));
5684 /* clear items. get item pointers. */
5685 for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
5686 if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
5687 gint id = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(cur->data),
5689 gtk_check_menu_item_set_active
5690 (GTK_CHECK_MENU_ITEM(cur->data), FALSE);
5692 g_hash_table_insert(menu_table, GINT_TO_POINTER(id), GTK_CHECK_MENU_ITEM(cur->data));
5693 g_hash_table_insert(menu_allsel_table, GINT_TO_POINTER(id), GINT_TO_POINTER(0));
5697 /* iterate all messages and set the state of the appropriate
5700 for (; sel != NULL; sel = sel->next) {
5702 GSList *tags = NULL;
5704 GtkCheckMenuItem *item;
5705 msginfo = gtk_ctree_node_get_row_data
5706 (GTK_CTREE(summaryview->ctree),
5707 GTK_CTREE_NODE(sel->data));
5710 tags = msginfo->tags;
5714 for (; tags; tags = tags->next) {
5715 gint num_checked = GPOINTER_TO_INT(g_hash_table_lookup(menu_allsel_table, tags->data));
5716 id = GPOINTER_TO_INT(tags->data);
5717 item = g_hash_table_lookup(menu_table, GINT_TO_POINTER(tags->data));
5718 if (item && !item->active) {
5719 gtk_check_menu_item_set_active
5723 g_hash_table_replace(menu_allsel_table, tags->data, GINT_TO_POINTER(num_checked));
5728 for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
5729 if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
5730 gint id = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(cur->data),
5732 gint num_checked = GPOINTER_TO_INT(g_hash_table_lookup(menu_allsel_table, GINT_TO_POINTER(id)));
5733 if (num_checked < sel_len && num_checked > 0)
5734 gtk_check_menu_item_set_inconsistent(GTK_CHECK_MENU_ITEM(cur->data), TRUE);
5736 gtk_check_menu_item_set_inconsistent(GTK_CHECK_MENU_ITEM(cur->data), FALSE);
5739 g_hash_table_destroy(menu_table);
5740 g_hash_table_destroy(menu_allsel_table);
5741 /* reset "dont_toggle" state */
5742 g_object_set_data(G_OBJECT(menu), "dont_toggle",
5743 GINT_TO_POINTER(0));
5747 void summaryview_destroy(SummaryView *summaryview)
5749 if(summaryview->simplify_subject_preg) {
5750 regfree(summaryview->simplify_subject_preg);
5751 g_free(summaryview->simplify_subject_preg);
5752 summaryview->simplify_subject_preg = NULL;
5755 static void summary_tags_menu_item_apply_tags_activate_cb(GtkWidget *widget,
5758 SummaryView *summaryview;
5760 summaryview = g_object_get_data(G_OBJECT(widget), "summaryview");
5761 g_return_if_fail(summaryview != NULL);
5763 /* "dont_toggle" state set? */
5764 if (g_object_get_data(G_OBJECT(summaryview->tags_menu),
5768 tag_apply_open(summary_get_selection(summaryview));
5771 static gint summary_tag_cmp_list(gconstpointer a, gconstpointer b)
5773 gint id_a = GPOINTER_TO_INT(a);
5774 gint id_b = GPOINTER_TO_INT(b);
5775 const gchar *tag_a = tags_get_tag(id_a);
5776 const gchar *tag_b = tags_get_tag(id_b);
5779 return tag_b == NULL ? 0:1;
5782 return tag_a == NULL ? 0:1;
5784 return g_utf8_collate(tag_a, tag_b);
5787 static void summary_tags_menu_create(SummaryView *summaryview, gboolean refresh)
5789 GtkWidget *label_menuitem;
5792 GSList *cur = tags_get_list();
5793 GSList *orig = NULL;
5794 gboolean existing_tags = FALSE;
5796 cur = orig = g_slist_sort(cur, summary_tag_cmp_list);
5797 label_menuitem = gtk_item_factory_get_item(summaryview->popupfactory,
5799 g_signal_connect(G_OBJECT(label_menuitem), "activate",
5800 G_CALLBACK(summary_tags_menu_item_activate_item_cb),
5803 gtk_widget_show(label_menuitem);
5805 menu = gtk_menu_new();
5807 /* create tags menu items */
5808 for (; cur; cur = cur->next) {
5809 gint id = GPOINTER_TO_INT(cur->data);
5810 const gchar *tag = tags_get_tag(id);
5812 item = gtk_check_menu_item_new_with_label(tag);
5813 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5814 g_signal_connect(G_OBJECT(item), "activate",
5815 G_CALLBACK(summary_tags_menu_item_activate_cb),
5816 GINT_TO_POINTER(id));
5817 g_object_set_data(G_OBJECT(item), "summaryview",
5819 g_object_set_data(G_OBJECT(item), "tag_id",
5820 GINT_TO_POINTER(id));
5821 gtk_widget_show(item);
5822 existing_tags = TRUE;
5824 if (existing_tags) {
5826 item = gtk_menu_item_new();
5827 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5828 gtk_widget_show(item);
5831 item = gtk_menu_item_new_with_label(_("Apply tags..."));
5832 gtk_widget_add_accelerator(item, "activate",
5833 summaryview->popupfactory->accel_group,
5834 GDK_T, GDK_CONTROL_MASK|GDK_SHIFT_MASK,
5835 GTK_ACCEL_LOCKED | GTK_ACCEL_VISIBLE);
5836 gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
5837 g_signal_connect(G_OBJECT(item), "activate",
5838 G_CALLBACK(summary_tags_menu_item_apply_tags_activate_cb),
5840 g_object_set_data(G_OBJECT(item), "summaryview",
5842 gtk_widget_show(item);
5845 gtk_widget_show(menu);
5846 gtk_menu_item_set_submenu(GTK_MENU_ITEM(label_menuitem), menu);
5847 summaryview->tags_menu = menu;
5850 static gboolean summary_popup_menu(GtkWidget *widget, gpointer data)
5852 SummaryView *summaryview = (SummaryView *)data;
5853 summaryview->display_msg = messageview_is_visible(summaryview->messageview);
5855 gtk_menu_popup(GTK_MENU(summaryview->popupmenu), NULL, NULL,
5856 NULL, NULL, 0, gtk_get_current_event_time());
5860 #if GTK_CHECK_VERSION(2,12,0) && !GENERIC_UMPC
5861 static gchar *summaryview_get_tooltip_text(SummaryView *summaryview, MsgInfo *info, gint column)
5867 flags = info->flags;
5869 switch(summaryview->col_state[column].type) {
5871 if (MSG_IS_IGNORE_THREAD(flags)) {
5872 return _("Ignored thread");
5873 } else if (MSG_IS_WATCH_THREAD(flags)) {
5874 return _("Watched thread");
5875 } else if (MSG_IS_SPAM(flags)) {
5877 } else if (MSG_IS_NEW(flags)) {
5879 } else if (MSG_IS_UNREAD(flags)) {
5881 } else if (MSG_IS_REPLIED(flags)) {
5882 return _("Replied - click to see reply");
5883 } else if (MSG_IS_FORWARDED(flags)) {
5884 return _("Forwarded");
5889 if (MSG_IS_DELETED(flags)) {
5890 return _("Deleted");
5891 } else if (MSG_IS_MARKED(flags)) {
5893 } else if (MSG_IS_MOVE(flags)) {
5894 return _("To be moved");
5895 } else if (MSG_IS_COPY(flags)) {
5896 return _("To be copied");
5901 if (MSG_IS_LOCKED(flags)) {
5907 if (MSG_IS_WITH_ATTACHMENT(flags) && MSG_IS_SIGNED(flags)) {
5908 return _("Signed, has attachment(s)");
5909 } else if (MSG_IS_SIGNED(flags)) {
5911 } else if (MSG_IS_WITH_ATTACHMENT(flags) && MSG_IS_ENCRYPTED(flags)) {
5912 return _("Encrypted, has attachment(s)");
5913 } else if (MSG_IS_ENCRYPTED(flags)) {
5914 return _("Encrypted");
5915 } else if (MSG_IS_WITH_ATTACHMENT(flags)) {
5916 return _("Has attachment(s)");
5924 static gboolean tooltip_cb (GtkWidget *widget,
5927 gboolean keyboard_mode,
5928 GtkTooltip *tooltip,
5931 GtkCTree *ctree = GTK_CTREE(widget);
5932 SummaryView *summaryview = (SummaryView *)user_data;
5933 gint row = -1, column = -1;
5934 int offset = prefs_common.show_col_headers ? 24:0;
5935 GtkCTreeNode *node = NULL;
5937 gchar *formatted = NULL;
5938 MsgInfo *info = NULL;
5940 gboolean vert = (prefs_common.layout_mode == VERTICAL_LAYOUT);
5942 if (!prefs_common.show_tooltips)
5948 if (!gtk_clist_get_selection_info(GTK_CLIST(ctree), x, y - offset,
5952 if ((node = gtk_ctree_node_nth(ctree, row)) == NULL)
5955 if ((info = gtk_ctree_node_get_row_data(ctree, node)) == NULL)
5958 switch (gtk_ctree_node_get_cell_type(ctree, node, column)) {
5960 if (gtk_ctree_node_get_text(ctree, node, column, &text) != TRUE)
5963 case GTK_CELL_PIXTEXT:
5964 if (gtk_ctree_node_get_pixtext(ctree, node, column, &text,
5965 NULL, NULL, NULL) != TRUE)
5969 if ((text = summaryview_get_tooltip_text(summaryview, info, column)) == NULL)
5973 if (!text || !*text)
5976 formatted = g_strdup(text);
5977 g_strstrip(formatted);
5980 gtk_tooltip_set_text (tooltip, formatted);
5981 else if (prefs_common.two_line_vert)
5982 gtk_tooltip_set_markup (tooltip, formatted);
5989 gtk_tooltip_set_tip_area(tooltip, &rect);
5994 static GtkWidget *summary_ctree_create(SummaryView *summaryview)
5997 gint *col_pos = summaryview->col_pos;
5998 SummaryColumnState *col_state;
5999 gchar *titles[N_SUMMARY_COLS];
6000 SummaryColumnType type;
6002 gboolean vert = (prefs_common.layout_mode == VERTICAL_LAYOUT);
6004 memset(titles, 0, sizeof(titles));
6006 col_state = prefs_summary_column_get_config();
6007 for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
6008 summaryview->col_state[pos] = col_state[pos];
6009 type = col_state[pos].type;
6010 col_pos[type] = pos;
6011 titles[pos] = "dummy";
6013 col_state = summaryview->col_state;
6015 ctree = gtk_sctree_new_with_titles
6016 (N_SUMMARY_COLS, col_pos[S_COL_SUBJECT], titles);
6018 /* get normal row height */
6019 gtk_clist_set_row_height(GTK_CLIST(ctree), 0);
6020 normal_row_height = GTK_CLIST(ctree)->row_height;
6022 if (prefs_common.show_col_headers == FALSE)
6023 gtk_clist_column_titles_hide(GTK_CLIST(ctree));
6025 gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_EXTENDED);
6026 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_MARK],
6027 GTK_JUSTIFY_CENTER);
6028 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_STATUS],
6029 GTK_JUSTIFY_CENTER);
6030 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_LOCKED],
6031 GTK_JUSTIFY_CENTER);
6032 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_MIME],
6033 GTK_JUSTIFY_CENTER);
6034 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_SIZE],
6036 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_NUMBER],
6038 gtk_clist_set_column_justification(GTK_CLIST(ctree), col_pos[S_COL_SCORE],
6040 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_MARK],
6041 prefs_common.summary_col_size[S_COL_MARK]);
6042 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_STATUS],
6043 prefs_common.summary_col_size[S_COL_STATUS]);
6044 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_LOCKED],
6045 prefs_common.summary_col_size[S_COL_LOCKED]);
6046 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_MIME],
6047 prefs_common.summary_col_size[S_COL_MIME]);
6048 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_SUBJECT],
6049 prefs_common.summary_col_size[S_COL_SUBJECT]);
6050 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_FROM],
6051 prefs_common.summary_col_size[S_COL_FROM]);
6052 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_TO],
6053 prefs_common.summary_col_size[S_COL_TO]);
6054 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_DATE],
6055 prefs_common.summary_col_size[S_COL_DATE]);
6056 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_SIZE],
6057 prefs_common.summary_col_size[S_COL_SIZE]);
6058 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_NUMBER],
6059 prefs_common.summary_col_size[S_COL_NUMBER]);
6060 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_SCORE],
6061 prefs_common.summary_col_size[S_COL_SCORE]);
6062 gtk_clist_set_column_width(GTK_CLIST(ctree), col_pos[S_COL_TAGS],
6063 prefs_common.summary_col_size[S_COL_TAGS]);
6065 if (prefs_common.enable_dotted_lines) {
6066 gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
6067 gtk_ctree_set_expander_style(GTK_CTREE(ctree),
6068 GTK_CTREE_EXPANDER_SQUARE);
6070 gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_NONE);
6071 gtk_ctree_set_expander_style(GTK_CTREE(ctree),
6072 GTK_CTREE_EXPANDER_TRIANGLE);
6075 gtk_sctree_set_stripes(GTK_SCTREE(ctree), prefs_common.use_stripes_in_summaries);
6077 gtk_ctree_set_indent(GTK_CTREE(ctree), 12);
6078 g_object_set_data(G_OBJECT(ctree), "summaryview", (gpointer)summaryview);
6080 for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
6081 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(ctree)->column[pos].button,
6083 if (((pos == summaryview->col_pos[S_COL_FROM] && !FOLDER_SHOWS_TO_HDR(summaryview->folder_item)) ||
6084 (pos == summaryview->col_pos[S_COL_TO] && FOLDER_SHOWS_TO_HDR(summaryview->folder_item)) ||
6085 pos == summaryview->col_pos[S_COL_DATE]) && vert &&
6086 prefs_common.two_line_vert)
6087 gtk_clist_set_column_visibility
6088 (GTK_CLIST(ctree), pos, FALSE);
6090 gtk_clist_set_column_visibility
6091 (GTK_CLIST(ctree), pos, col_state[pos].visible);
6093 if (prefs_common.two_line_vert)
6094 gtk_sctree_set_use_markup(GTK_SCTREE(ctree), summaryview->col_pos[S_COL_SUBJECT], vert);
6096 /* connect signal to the buttons for sorting */
6097 #define CLIST_BUTTON_SIGNAL_CONNECT(col, func) \
6099 (G_OBJECT(GTK_CLIST(ctree)->column[col_pos[col]].button), \
6104 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_MARK , summary_mark_clicked);
6105 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_STATUS , summary_status_clicked);
6106 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_MIME , summary_mime_clicked);
6107 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_NUMBER , summary_num_clicked);
6108 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_SIZE , summary_size_clicked);
6109 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_DATE , summary_date_clicked);
6110 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_FROM , summary_from_clicked);
6111 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_TO , summary_to_clicked);
6112 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_SUBJECT, summary_subject_clicked);
6113 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_SCORE, summary_score_clicked);
6114 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_LOCKED, summary_locked_clicked);
6115 CLIST_BUTTON_SIGNAL_CONNECT(S_COL_TAGS, summary_tags_clicked);
6117 #undef CLIST_BUTTON_SIGNAL_CONNECT
6119 g_signal_connect(G_OBJECT(ctree), "tree_select_row",
6120 G_CALLBACK(summary_selected), summaryview);
6121 g_signal_connect(G_OBJECT(ctree), "tree_unselect_row",
6122 G_CALLBACK(summary_unselected), summaryview);
6123 g_signal_connect(G_OBJECT(ctree), "button_press_event",
6124 G_CALLBACK(summary_button_pressed),
6127 g_signal_connect(G_OBJECT(ctree), "popup-menu",
6128 G_CALLBACK(summary_popup_menu), summaryview);
6130 gtk_widget_tap_and_hold_setup(GTK_WIDGET(ctree), NULL, NULL,
6131 GTK_TAP_AND_HOLD_NONE | GTK_TAP_AND_HOLD_NO_INTERNALS);
6132 g_signal_connect(G_OBJECT(ctree), "tap-and-hold",
6133 G_CALLBACK(summary_popup_menu), summaryview);
6135 g_signal_connect(G_OBJECT(ctree), "button_release_event",
6136 G_CALLBACK(summary_button_released),
6138 g_signal_connect(G_OBJECT(ctree), "key_press_event",
6139 G_CALLBACK(summary_key_pressed), summaryview);
6140 g_signal_connect(G_OBJECT(ctree), "resize_column",
6141 G_CALLBACK(summary_col_resized), summaryview);
6142 g_signal_connect(G_OBJECT(ctree), "open_row",
6143 G_CALLBACK(summary_open_row), summaryview);
6145 g_signal_connect_after(G_OBJECT(ctree), "tree_expand",
6146 G_CALLBACK(summary_tree_expanded),
6148 g_signal_connect_after(G_OBJECT(ctree), "tree_collapse",
6149 G_CALLBACK(summary_tree_collapsed),
6152 g_signal_connect(G_OBJECT(ctree), "start_drag",
6153 G_CALLBACK(summary_start_drag),
6155 g_signal_connect(G_OBJECT(ctree), "drag_data_get",
6156 G_CALLBACK(summary_drag_data_get),
6158 g_signal_connect(G_OBJECT(ctree), "drag_end",
6159 G_CALLBACK(summary_drag_end),
6162 gtk_drag_dest_set(ctree, GTK_DEST_DEFAULT_ALL & ~GTK_DEST_DEFAULT_HIGHLIGHT,
6163 summary_drag_types, 2,
6164 GDK_ACTION_MOVE | GDK_ACTION_COPY | GDK_ACTION_DEFAULT);
6166 g_signal_connect(G_OBJECT(ctree), "drag_data_received",
6167 G_CALLBACK(summary_drag_data_received),
6170 g_signal_connect(G_OBJECT(ctree), "drag_motion",
6171 G_CALLBACK(summary_drag_motion_cb),
6174 #if GTK_CHECK_VERSION(2,12,0) && !GENERIC_UMPC
6175 g_object_set (G_OBJECT(ctree), "has-tooltip", TRUE, NULL);
6176 g_signal_connect(G_OBJECT(ctree), "query-tooltip",
6177 G_CALLBACK(tooltip_cb),
6183 void summary_set_column_order(SummaryView *summaryview)
6186 GtkWidget *scrolledwin = summaryview->scrolledwin;
6188 guint selected_msgnum = summary_get_msgnum(summaryview, summaryview->selected);
6189 guint displayed_msgnum = summary_get_msgnum(summaryview, summaryview->displayed);
6191 item = summaryview->folder_item;
6193 summary_clear_all(summaryview);
6194 gtk_widget_destroy(summaryview->ctree);
6196 summaryview->ctree = ctree = summary_ctree_create(summaryview);
6197 summary_set_fonts(summaryview);
6198 summary_set_column_titles(summaryview);
6199 gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
6200 GTK_CLIST(ctree)->hadjustment);
6201 gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
6202 GTK_CLIST(ctree)->vadjustment);
6203 gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
6204 gtk_widget_show(ctree);
6206 summary_show(summaryview, item);
6208 summary_select_by_msgnum(summaryview, selected_msgnum);
6210 summaryview->selected = summary_find_msg_by_msgnum(summaryview, selected_msgnum);
6211 summaryview->displayed = summary_find_msg_by_msgnum(summaryview, displayed_msgnum);
6212 summaryview->last_displayed = summaryview->displayed;
6213 if (!summaryview->displayed)
6214 messageview_clear(summaryview->messageview);
6216 summary_redisplay_msg(summaryview);
6218 if (prefs_common.layout_mode == VERTICAL_LAYOUT &&
6219 prefs_common.two_line_vert) {
6220 gtk_clist_set_row_height(GTK_CLIST(summaryview->ctree), 2*normal_row_height + 2);
6222 gtk_clist_set_row_height(GTK_CLIST(summaryview->ctree), 0);
6227 /* callback functions */
6229 static gint summary_folder_eventbox_pressed(GtkWidget *eventbox, GdkEventButton *event,
6230 SummaryView *summaryview)
6233 gtk_widget_grab_focus(summaryview->folderview->ctree);
6234 mainwindow_exit_folder(summaryview->mainwin);
6239 static gint summary_toggle_pressed(GtkWidget *eventbox, GdkEventButton *event,
6240 SummaryView *summaryview)
6243 summary_toggle_view(summaryview);
6247 static void summary_toggle_multiple_pressed(GtkWidget *widget,
6248 SummaryView *summaryview)
6250 GTK_SCTREE(summaryview->ctree)->force_additive_sel =
6251 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget));
6254 static gboolean summary_button_pressed(GtkWidget *ctree, GdkEventButton *event,
6255 SummaryView *summaryview)
6257 if (!event) return FALSE;
6259 if (event->button == 3) {
6260 summaryview->display_msg = messageview_is_visible(summaryview->messageview);
6262 gtk_menu_popup(GTK_MENU(summaryview->popupmenu), NULL, NULL,
6263 NULL, NULL, event->button, event->time);
6264 } else if (event->button == 2) {
6265 summaryview->display_msg = messageview_is_visible(summaryview->messageview);
6266 } else if (event->button == 1) {
6267 if (!prefs_common.emulate_emacs &&
6268 messageview_is_visible(summaryview->messageview))
6269 summaryview->display_msg = TRUE;
6275 static gboolean summary_button_released(GtkWidget *ctree, GdkEventButton *event,
6276 SummaryView *summaryview)
6281 gboolean summary_pass_key_press_event(SummaryView *summaryview, GdkEventKey *event)
6283 if (summary_is_list(summaryview))
6284 return summary_key_pressed(summaryview->ctree, event, summaryview);
6289 #define BREAK_ON_MODIFIER_KEY() \
6290 if ((event->state & (GDK_MOD1_MASK|GDK_CONTROL_MASK)) != 0) break
6292 static gboolean summary_key_pressed(GtkWidget *widget, GdkEventKey *event,
6293 SummaryView *summaryview)
6295 GtkCTree *ctree = GTK_CTREE(widget);
6297 MessageView *messageview;
6300 gboolean mod_pressed;
6305 if (quicksearch_has_focus(summaryview->quicksearch))
6308 messageview = summaryview->messageview;
6309 textview = messageview->mimeview->textview;
6312 ((event->state & (GDK_SHIFT_MASK|GDK_MOD1_MASK)) != 0);
6314 if (summaryview->selected) {
6315 gboolean handled = FALSE;
6316 switch (event->keyval) {
6317 case GDK_space: /* Page down or go to the next */
6319 if (event->state & GDK_CONTROL_MASK)
6320 event->state &= ~GDK_CONTROL_MASK;
6321 if (event->state & GDK_SHIFT_MASK)
6322 mimeview_scroll_page(messageview->mimeview, TRUE);
6324 if (summaryview->displayed != summaryview->selected) {
6325 summary_display_msg(summaryview,
6326 summaryview->selected);
6330 if (!mimeview_scroll_page(messageview->mimeview, TRUE))
6331 summary_select_prev_unread(summaryview);
6333 if (!mimeview_scroll_page(messageview->mimeview, FALSE))
6334 summary_select_next_unread(summaryview);
6338 case GDK_BackSpace: /* Page up */
6340 mimeview_scroll_page(messageview->mimeview, TRUE);
6342 case GDK_Return: /* Scroll up/down one line */
6345 if (summaryview->displayed != summaryview->selected) {
6346 #ifndef GENERIC_UMPC
6347 summary_display_msg(summaryview,
6348 summaryview->selected);
6350 summary_open_row(NULL, summaryview);
6354 mimeview_scroll_one_line(messageview->mimeview, mod_pressed);
6361 if (summary_is_locked(summaryview))
6364 switch (event->keyval) {
6365 case GDK_Left: /* Move focus */
6366 adj = gtk_scrolled_window_get_hadjustment
6367 (GTK_SCROLLED_WINDOW(summaryview->scrolledwin));
6368 if (adj->lower != adj->value)
6372 gtk_widget_grab_focus(summaryview->folderview->ctree);
6373 mainwindow_exit_folder(summaryview->mainwin);
6377 if ((node = summaryview->selected) != NULL) {
6378 GtkCTreeNode *next = NULL;
6379 next = (event->keyval == GDK_Home)
6380 ? gtk_ctree_node_nth(ctree, 0)
6381 : gtk_ctree_node_nth(ctree,
6382 g_list_length(GTK_CLIST(ctree)->row_list)-1);
6384 gtk_sctree_select_with_state
6385 (GTK_SCTREE(ctree), next, (event->state & ~GDK_CONTROL_MASK) );
6387 /* Deprecated - what are the non-deprecated equivalents? */
6388 if (gtk_ctree_node_is_visible(GTK_CTREE(ctree), next) != GTK_VISIBILITY_FULL)
6389 gtk_ctree_node_moveto(GTK_CTREE(ctree), next, 0, 0, 0);
6390 summaryview->selected = next;
6398 if (!summaryview->selected) {
6399 node = gtk_ctree_node_nth(ctree, 0);
6401 gtk_sctree_select(GTK_SCTREE(ctree), node);
6406 switch (event->keyval) {
6408 BREAK_ON_MODIFIER_KEY();
6409 summary_delete_trash(summaryview);
6417 if ((event->state & (GDK_MOD1_MASK|GDK_CONTROL_MASK)) == 0) {
6418 g_signal_stop_emission_by_name(G_OBJECT(widget),
6420 mimeview_pass_key_press_event(messageview->mimeview,
6430 static void quicksearch_execute_cb(QuickSearch *quicksearch, gpointer data)
6432 SummaryView *summaryview = data;
6434 summary_show(summaryview, summaryview->folder_item);
6437 static void tog_searchbar_cb(GtkWidget *w, gpointer data)
6439 SummaryView *summaryview = (SummaryView *)data;
6441 if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(w))) {
6442 prefs_common.show_searchbar = TRUE;
6443 quicksearch_show(summaryview->quicksearch);
6445 prefs_common.show_searchbar = FALSE;
6446 quicksearch_hide(summaryview->quicksearch);
6450 void summaryview_activate_quicksearch(SummaryView *summaryview, gboolean show)
6452 prefs_common.show_searchbar = show;
6453 gtk_toggle_button_set_active(
6454 GTK_TOGGLE_BUTTON(summaryview->toggle_search),
6457 quicksearch_show(summaryview->quicksearch);
6459 quicksearch_hide(summaryview->quicksearch);
6460 summary_grab_focus(summaryview);
6464 void summary_open_row(GtkSCTree *sctree, SummaryView *summaryview)
6466 if (FOLDER_SHOWS_TO_HDR(summaryview->folder_item))
6467 summary_reedit(summaryview);
6469 summary_open_msg(summaryview);
6471 summaryview->display_msg = FALSE;
6474 static void summary_tree_expanded(GtkCTree *ctree, GtkCTreeNode *node,
6475 SummaryView *summaryview)
6477 summary_set_row_marks(summaryview, node);
6478 if (prefs_common.bold_unread) {
6480 GtkCTreeNode *next = GTK_CTREE_NODE_NEXT(node);
6481 if (GTK_CTREE_ROW(node)->children)
6482 summary_set_row_marks(summaryview, node);
6488 static void summary_tree_collapsed(GtkCTree *ctree, GtkCTreeNode *node,
6489 SummaryView *summaryview)
6491 summary_set_row_marks(summaryview, node);
6494 static void summary_unselected(GtkCTree *ctree, GtkCTreeNode *row,
6495 gint column, SummaryView *summaryview)
6497 if (summary_is_locked(summaryview)
6498 || GTK_SCTREE(ctree)->selecting_range) {
6502 summary_status_show(summaryview);
6505 static void summary_selected(GtkCTree *ctree, GtkCTreeNode *row,
6506 gint column, SummaryView *summaryview)
6509 gboolean marked_unread = FALSE;
6511 if (summary_is_locked(summaryview)
6512 && !GTK_SCTREE(ctree)->selecting_range
6513 && summaryview->messageview
6514 && summaryview->messageview->mimeview
6515 && summaryview->messageview->mimeview->type == MIMEVIEW_TEXT
6516 && summaryview->messageview->mimeview->textview->loading) {
6517 PostponedSelectData *data = g_new0(PostponedSelectData, 1);
6518 summaryview->messageview->mimeview->textview->stop_loading = TRUE;
6520 data->ctree = ctree;
6523 data->column = column;
6524 data->summaryview = summaryview;
6525 debug_print("postponing open of message till end of load\n");
6526 g_timeout_add(100, summary_select_retry, data);
6529 if (summary_is_locked(summaryview)
6530 || GTK_SCTREE(ctree)->selecting_range) {
6534 summary_status_show(summaryview);
6536 if (GTK_CLIST(ctree)->selection &&
6537 GTK_CLIST(ctree)->selection->next) {
6538 summaryview->display_msg = FALSE;
6539 summary_set_menu_sensitive(summaryview);
6540 toolbar_main_set_sensitive(summaryview->mainwin);
6544 summaryview->selected = row;
6546 msginfo = gtk_ctree_node_get_row_data(ctree, row);
6547 g_return_if_fail(msginfo != NULL);
6549 main_create_mailing_list_menu (summaryview->mainwin, msginfo);
6550 toolbar_set_learn_button
6551 (summaryview->mainwin->toolbar,
6552 MSG_IS_SPAM(msginfo->flags)?LEARN_HAM:LEARN_SPAM);
6554 switch (column < 0 ? column : summaryview->col_state[column].type) {
6556 if (!MSG_IS_DELETED(msginfo->flags) &&
6557 !MSG_IS_MOVE(msginfo->flags) &&
6558 !MSG_IS_COPY(msginfo->flags)) {
6559 if (MSG_IS_MARKED(msginfo->flags)) {
6560 summary_unmark_row(summaryview, row);
6561 summary_status_show(summaryview);
6563 summary_mark_row(summaryview, row);
6564 summary_status_show(summaryview);
6569 if (MSG_IS_UNREAD(msginfo->flags)) {
6570 summary_mark_row_as_read(summaryview, row);
6571 summary_status_show(summaryview);
6572 } else if (MSG_IS_SPAM(msginfo->flags)) {
6573 if (procmsg_spam_learner_learn(msginfo, NULL, FALSE) == 0)
6574 summary_msginfo_unset_flags(msginfo, MSG_SPAM, 0);
6576 log_error(LOG_PROTOCOL, _("An error happened while learning.\n"));
6577 } else if (!MSG_IS_REPLIED(msginfo->flags) &&
6578 !MSG_IS_FORWARDED(msginfo->flags)) {
6579 marked_unread = TRUE;
6580 } else if (MSG_IS_REPLIED(msginfo->flags)) {
6581 summary_find_answers(summaryview, msginfo);
6586 if (MSG_IS_LOCKED(msginfo->flags)) {
6587 summary_unlock_row(summaryview, row);
6588 summary_status_show(summaryview);
6591 summary_lock_row(summaryview, row);
6592 summary_status_show(summaryview);
6599 if (summaryview->display_msg ||
6600 (prefs_common.always_show_msg &&
6601 messageview_is_visible(summaryview->messageview))) {
6602 summaryview->display_msg = FALSE;
6603 if (summaryview->displayed != row) {
6604 summary_display_msg(summaryview, row);
6605 if (marked_unread) {
6606 summary_mark_row_as_unread(summaryview, row);
6607 summary_status_show(summaryview);
6613 if (marked_unread) {
6614 summary_mark_row_as_unread(summaryview, row);
6615 summary_status_show(summaryview);
6618 summary_set_menu_sensitive(summaryview);
6619 toolbar_main_set_sensitive(summaryview->mainwin);
6622 static void summary_col_resized(GtkCList *clist, gint column, gint width,
6623 SummaryView *summaryview)
6625 SummaryColumnType type = summaryview->col_state[column].type;
6627 prefs_common.summary_col_size[type] = width;
6632 * \brief get List of msginfo selected in SummaryView
6634 * \param summaryview
6636 * \return GSList holding MsgInfo
6638 GSList *summary_get_selection(SummaryView *summaryview)
6641 GSList *msginfo_list = NULL;
6643 g_return_val_if_fail(summaryview != NULL, NULL);
6645 sel = GTK_CLIST(summaryview->ctree)->selection;
6647 g_return_val_if_fail(sel != NULL, NULL);
6649 for ( ; sel != NULL; sel = sel->next)
6651 g_slist_prepend(msginfo_list,
6652 gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
6653 GTK_CTREE_NODE(sel->data)));
6654 return g_slist_reverse(msginfo_list);
6657 static void summary_reply_cb(SummaryView *summaryview, guint action,
6660 MessageView *msgview = (MessageView*)summaryview->messageview;
6661 GSList *msginfo_list;
6663 g_return_if_fail(msgview != NULL);
6665 msginfo_list = summary_get_selection(summaryview);
6666 g_return_if_fail(msginfo_list != NULL);
6667 compose_reply_from_messageview(msgview, msginfo_list, action);
6668 g_slist_free(msginfo_list);
6670 #ifndef GENERIC_UMPC
6671 static void summary_show_all_header_cb(SummaryView *summaryview,
6672 guint action, GtkWidget *widget)
6674 summaryview->messageview->all_headers =
6675 GTK_CHECK_MENU_ITEM(widget)->active;
6676 summary_display_msg_selected(summaryview,
6677 GTK_CHECK_MENU_ITEM(widget)->active);
6680 static void summary_add_address_cb(SummaryView *summaryview,
6681 guint action, GtkWidget *widget)
6683 summary_add_address(summaryview);
6686 static void summary_create_filter_cb(SummaryView *summaryview,
6687 guint action, GtkWidget *widget)
6689 summary_filter_open(summaryview, (PrefsFilterType)action, 0);
6691 #ifndef GENERIC_UMPC
6692 static void summary_create_processing_cb(SummaryView *summaryview,
6693 guint action, GtkWidget *widget)
6695 summary_filter_open(summaryview, (PrefsFilterType)action, 1);
6698 static void summary_sort_by_column_click(SummaryView *summaryview,
6699 FolderSortKey sort_key)
6701 GtkCTreeNode *node = NULL;
6703 if (summaryview->sort_key == sort_key)
6704 summary_sort(summaryview, sort_key,
6705 summaryview->sort_type == SORT_ASCENDING
6706 ? SORT_DESCENDING : SORT_ASCENDING);
6708 summary_sort(summaryview, sort_key, SORT_ASCENDING);
6710 node = GTK_CTREE_NODE(GTK_CLIST(summaryview->ctree)->row_list);
6712 summary_freeze(summaryview);
6713 if (prefs_common.bold_unread) {
6715 GtkCTreeNode *next = GTK_CTREE_NODE_NEXT(node);
6716 if (GTK_CTREE_ROW(node)->children)
6717 summary_set_row_marks(summaryview, node);
6721 summary_thaw(summaryview);
6725 static void summary_mark_clicked(GtkWidget *button, SummaryView *summaryview)
6727 summary_sort_by_column_click(summaryview, SORT_BY_MARK);
6730 static void summary_status_clicked(GtkWidget *button, SummaryView *summaryview)
6732 summary_sort_by_column_click(summaryview, SORT_BY_STATUS);
6735 static void summary_mime_clicked(GtkWidget *button, SummaryView *summaryview)
6737 summary_sort_by_column_click(summaryview, SORT_BY_MIME);
6740 static void summary_num_clicked(GtkWidget *button, SummaryView *summaryview)
6742 summary_sort_by_column_click(summaryview, SORT_BY_NUMBER);
6745 static void summary_size_clicked(GtkWidget *button, SummaryView *summaryview)
6747 summary_sort_by_column_click(summaryview, SORT_BY_SIZE);
6750 static void summary_date_clicked(GtkWidget *button, SummaryView *summaryview)
6752 summary_sort_by_column_click(summaryview, SORT_BY_DATE);
6755 static void summary_from_clicked(GtkWidget *button, SummaryView *summaryview)
6757 if (summaryview->col_state[summaryview->col_pos[S_COL_FROM]].visible)
6758 summary_sort_by_column_click(summaryview, SORT_BY_FROM);
6760 summary_sort_by_column_click(summaryview, SORT_BY_TO);
6763 static void summary_to_clicked(GtkWidget *button, SummaryView *summaryview)
6765 if (summaryview->col_state[summaryview->col_pos[S_COL_TO]].visible)
6766 summary_sort_by_column_click(summaryview, SORT_BY_TO);
6768 summary_sort_by_column_click(summaryview, SORT_BY_FROM);
6771 static void summary_subject_clicked(GtkWidget *button,
6772 SummaryView *summaryview)
6774 summary_sort_by_column_click(summaryview, SORT_BY_SUBJECT);
6777 static void summary_score_clicked(GtkWidget *button,
6778 SummaryView *summaryview)
6780 summary_sort_by_column_click(summaryview, SORT_BY_SCORE);
6783 static void summary_locked_clicked(GtkWidget *button,
6784 SummaryView *summaryview)
6786 summary_sort_by_column_click(summaryview, SORT_BY_LOCKED);
6789 static void summary_tags_clicked(GtkWidget *button,
6790 SummaryView *summaryview)
6792 summary_sort_by_column_click(summaryview, SORT_BY_TAGS);
6795 static void summary_start_drag(GtkWidget *widget, gint button, GdkEvent *event,
6796 SummaryView *summaryview)
6798 GdkDragContext *context;
6800 g_return_if_fail(summaryview != NULL);
6801 g_return_if_fail(summaryview->folder_item != NULL);
6802 g_return_if_fail(summaryview->folder_item->folder != NULL);
6804 if (summaryview->selected == NULL) return;
6806 context = gtk_drag_begin(widget, summaryview->target_list,
6807 GDK_ACTION_MOVE|GDK_ACTION_COPY|GDK_ACTION_DEFAULT, button, event);
6808 gtk_drag_set_icon_default(context);
6809 if (prefs_common.layout_mode == SMALL_LAYOUT) {
6810 GtkWidget *paned = GTK_WIDGET_PTR(summaryview)->parent;
6811 if (paned && GTK_IS_PANED(paned)) {
6812 mainwindow_reset_paned(GTK_PANED(paned));
6817 static gboolean summary_return_to_list(void *data)
6819 SummaryView *summaryview = (SummaryView *)data;
6820 mainwindow_enter_folder(summaryview->mainwin);
6824 static void summary_drag_end (GtkWidget *widget,
6825 GdkDragContext *drag_context,
6826 SummaryView *summaryview)
6828 if (prefs_common.layout_mode == SMALL_LAYOUT) {
6829 g_timeout_add(250, summary_return_to_list, summaryview);
6833 static void summary_drag_data_get(GtkWidget *widget,
6834 GdkDragContext *drag_context,
6835 GtkSelectionData *selection_data,
6838 SummaryView *summaryview)
6840 if (info == TARGET_MAIL_URI_LIST) {
6841 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
6844 gchar *mail_list = NULL, *tmp1, *tmp2;
6846 for (cur = GTK_CLIST(ctree)->selection;
6847 cur != NULL && cur->data != NULL; cur = cur->next) {
6848 msginfo = gtk_ctree_node_get_row_data
6849 (ctree, GTK_CTREE_NODE(cur->data));
6850 tmp2 = procmsg_get_message_file(msginfo);
6851 if (!tmp2) continue;
6852 if (msginfo->subject) {
6853 gchar *san_subject = g_strdup(msginfo->subject);
6855 subst_for_filename(san_subject);
6856 dest = g_strdup_printf("%s%s%s.%d.txt",
6859 san_subject, msginfo->msgnum);
6860 g_free(san_subject);
6861 san_subject = g_filename_from_utf8(dest, -1, NULL, NULL, NULL);
6864 if (copy_file(tmp2, dest, TRUE) == 0) {
6869 tmp1 = g_filename_to_uri(tmp2, NULL, NULL);
6871 tmp2 = g_strconcat(tmp1, "\r\n", NULL);
6878 tmp2 = g_strconcat(mail_list, tmp1, NULL);
6885 if (mail_list != NULL) {
6886 gtk_selection_data_set(selection_data,
6887 selection_data->target, 8,
6888 mail_list, strlen(mail_list));
6891 } else if (info == TARGET_DUMMY) {
6892 if (GTK_CLIST(summaryview->ctree)->selection)
6893 gtk_selection_data_set(selection_data,
6894 selection_data->target, 8,
6895 "Dummy-Summaryview",
6896 strlen("Dummy-Summaryview")+1);
6900 static gboolean summary_drag_motion_cb(GtkWidget *widget,
6901 GdkDragContext *context,
6905 SummaryView *summaryview)
6907 FolderItem *item = summaryview->folder_item;
6908 if (!(item && item->folder && folder_item_parent(item) != NULL
6909 && FOLDER_CLASS(item->folder)->add_msg != NULL)) {
6910 gdk_drag_status(context, 0, time);
6912 } else if (gtk_drag_get_source_widget(context) ==
6913 mainwindow_get_mainwindow()->folderview->ctree) {
6915 gdk_drag_status(context, 0, time);
6917 } else if (gtk_drag_get_source_widget(context) ==
6918 summaryview->ctree) {
6919 /* not from same folder */
6920 gdk_drag_status(context, 0, time);
6923 gdk_drag_status(context, GDK_ACTION_COPY, time);
6928 static void summary_drag_data_received(GtkWidget *widget,
6929 GdkDragContext *drag_context,
6932 GtkSelectionData *data,
6935 SummaryView *summaryview)
6937 if (info == TARGET_MAIL_URI_LIST) {
6938 FolderItem *item = summaryview->folder_item;
6940 gtk_drag_finish(drag_context, FALSE, FALSE, time);
6943 folderview_finish_dnd(data->data, drag_context, time, item);
6949 /* custom compare functions for sorting */
6951 #define CMP_FUNC_DEF(func_name, val) \
6952 static gint func_name(GtkCList *clist, \
6953 gconstpointer ptr1, gconstpointer ptr2) \
6955 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data; \
6956 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data; \
6957 if (!msginfo1 || !msginfo2) \
6963 CMP_FUNC_DEF(summary_cmp_by_mark,
6964 MSG_IS_MARKED(msginfo1->flags) - MSG_IS_MARKED(msginfo2->flags))
6965 CMP_FUNC_DEF(summary_cmp_by_status,
6966 (-(MSG_IS_SPAM(msginfo1->flags))+(MSG_IS_UNREAD(msginfo1->flags)<<1)+(MSG_IS_NEW(msginfo1->flags)<<2))
6967 - (-(MSG_IS_SPAM(msginfo2->flags))+(MSG_IS_UNREAD(msginfo2->flags)<<1)+(MSG_IS_NEW(msginfo2->flags)<<2)) )
6968 CMP_FUNC_DEF(summary_cmp_by_mime,
6969 MSG_IS_WITH_ATTACHMENT(msginfo1->flags) - MSG_IS_WITH_ATTACHMENT(msginfo2->flags))
6970 CMP_FUNC_DEF(summary_cmp_by_label,
6971 MSG_GET_COLORLABEL(msginfo1->flags) -
6972 MSG_GET_COLORLABEL(msginfo2->flags))
6973 CMP_FUNC_DEF(summary_cmp_by_locked,
6974 MSG_IS_LOCKED(msginfo1->flags) - MSG_IS_LOCKED(msginfo2->flags))
6976 CMP_FUNC_DEF(summary_cmp_by_num, msginfo1->msgnum - msginfo2->msgnum)
6977 CMP_FUNC_DEF(summary_cmp_by_size, msginfo1->size - msginfo2->size)
6978 CMP_FUNC_DEF(summary_cmp_by_date, msginfo1->date_t - msginfo2->date_t)
6982 static gint summary_cmp_by_subject(GtkCList *clist,
6986 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
6987 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
6990 if (!msginfo1->subject)
6991 return (msginfo2->subject != NULL);
6992 if (!msginfo2->subject)
6995 res = subject_compare_for_sort
6996 (msginfo1->subject, msginfo2->subject);
6997 return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
7000 static gint summary_cmp_by_thread_date(GtkCList *clist,
7004 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
7005 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
7006 gint thread_diff = msginfo1->thread_date - msginfo2->thread_date;
7008 if (msginfo1->thread_date > 0 && msginfo2->thread_date > 0)
7011 return msginfo1->date_t - msginfo2->date_t;
7014 static gint summary_cmp_by_from(GtkCList *clist, gconstpointer ptr1,
7017 const gchar *str1, *str2;
7018 const GtkCListRow *r1 = (const GtkCListRow *) ptr1;
7019 const GtkCListRow *r2 = (const GtkCListRow *) ptr2;
7020 const SummaryView *sv = g_object_get_data(G_OBJECT(clist), "summaryview");
7023 g_return_val_if_fail(sv, -1);
7025 str1 = GTK_CELL_TEXT(r1->cell[sv->col_pos[S_COL_FROM]])->text;
7026 str2 = GTK_CELL_TEXT(r2->cell[sv->col_pos[S_COL_FROM]])->text;
7029 return str2 != NULL;
7034 res = g_utf8_collate(str1, str2);
7035 return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
7038 static gint summary_cmp_by_to(GtkCList *clist, gconstpointer ptr1,
7041 const gchar *str1, *str2;
7042 const GtkCListRow *r1 = (const GtkCListRow *) ptr1;
7043 const GtkCListRow *r2 = (const GtkCListRow *) ptr2;
7044 const SummaryView *sv = g_object_get_data(G_OBJECT(clist), "summaryview");
7046 g_return_val_if_fail(sv, -1);
7048 str1 = GTK_CELL_TEXT(r1->cell[sv->col_pos[S_COL_TO]])->text;
7049 str2 = GTK_CELL_TEXT(r2->cell[sv->col_pos[S_COL_TO]])->text;
7052 return str2 != NULL;
7057 res = g_utf8_collate(str1, str2);
7058 return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
7061 static gint summary_cmp_by_tags(GtkCList *clist, gconstpointer ptr1,
7064 const gchar *str1, *str2;
7065 const GtkCListRow *r1 = (const GtkCListRow *) ptr1;
7066 const GtkCListRow *r2 = (const GtkCListRow *) ptr2;
7067 const SummaryView *sv = g_object_get_data(G_OBJECT(clist), "summaryview");
7069 g_return_val_if_fail(sv, -1);
7071 str1 = GTK_CELL_TEXT(r1->cell[sv->col_pos[S_COL_TAGS]])->text;
7072 str2 = GTK_CELL_TEXT(r2->cell[sv->col_pos[S_COL_TAGS]])->text;
7075 return str2 != NULL;
7080 res = g_utf8_collate(str1, str2);
7081 return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
7084 static gint summary_cmp_by_simplified_subject
7085 (GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2)
7087 const FolderItemPrefs *prefs;
7088 const gchar *str1, *str2;
7089 const GtkCListRow *r1 = (const GtkCListRow *) ptr1;
7090 const GtkCListRow *r2 = (const GtkCListRow *) ptr2;
7091 const MsgInfo *msginfo1 = r1->data;
7092 const MsgInfo *msginfo2 = r2->data;
7093 const SummaryView *sv = g_object_get_data(G_OBJECT(clist), "summaryview");
7096 g_return_val_if_fail(sv, -1);
7097 g_return_val_if_fail(msginfo1 != NULL && msginfo2 != NULL, -1);
7099 str1 = GTK_CELL_TEXT(r1->cell[sv->col_pos[S_COL_SUBJECT]])->text;
7100 str2 = GTK_CELL_TEXT(r2->cell[sv->col_pos[S_COL_SUBJECT]])->text;
7103 return str2 != NULL;
7108 prefs = msginfo1->folder->prefs;
7110 prefs = msginfo2->folder->prefs;
7114 res = subject_compare_for_sort(str1, str2);
7115 return (res != 0)? res: summary_cmp_by_date(clist, ptr1, ptr2);
7118 static gint summary_cmp_by_score(GtkCList *clist,
7119 gconstpointer ptr1, gconstpointer ptr2)
7121 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
7122 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
7125 /* if score are equal, sort by date */
7127 diff = msginfo1->score - msginfo2->score;
7131 return summary_cmp_by_date(clist, ptr1, ptr2);
7134 static void summary_ignore_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7136 SummaryView *summaryview = (SummaryView *) data;
7139 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7140 g_return_if_fail(msginfo);
7142 summary_msginfo_unset_flags(msginfo, MSG_WATCH_THREAD, 0);
7143 summary_msginfo_change_flags(msginfo, MSG_IGNORE_THREAD, 0, MSG_NEW | MSG_UNREAD, 0);
7145 summary_set_row_marks(summaryview, row);
7146 debug_print("Message %d is marked as ignore thread\n",
7150 void summary_ignore_thread(SummaryView *summaryview)
7152 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7154 gboolean froze = FALSE;
7156 START_LONG_OPERATION(summaryview, FALSE);
7157 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7158 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7159 GTK_CTREE_FUNC(summary_ignore_thread_func),
7162 END_LONG_OPERATION(summaryview);
7164 summary_status_show(summaryview);
7167 static void summary_unignore_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7169 SummaryView *summaryview = (SummaryView *) data;
7172 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7173 g_return_if_fail(msginfo);
7175 summary_msginfo_unset_flags(msginfo, MSG_IGNORE_THREAD, 0);
7177 summary_set_row_marks(summaryview, row);
7178 debug_print("Message %d is marked as unignore thread\n",
7182 void summary_unignore_thread(SummaryView *summaryview)
7184 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7186 gboolean froze = FALSE;
7188 START_LONG_OPERATION(summaryview, FALSE);
7189 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7190 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7191 GTK_CTREE_FUNC(summary_unignore_thread_func),
7194 END_LONG_OPERATION(summaryview);
7196 summary_status_show(summaryview);
7199 static void summary_check_ignore_thread_func
7200 (GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7203 gint *found_ignore = (gint *) data;
7205 if (*found_ignore) return;
7207 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7208 *found_ignore = msginfo && MSG_IS_IGNORE_THREAD(msginfo->flags);
7212 void summary_toggle_ignore_thread(SummaryView *summaryview)
7214 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7216 gint found_ignore = 0;
7218 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7219 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7220 GTK_CTREE_FUNC(summary_check_ignore_thread_func),
7224 summary_unignore_thread(summaryview);
7226 summary_ignore_thread(summaryview);
7229 static void summary_watch_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7231 SummaryView *summaryview = (SummaryView *) data;
7234 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7235 g_return_if_fail(msginfo);
7237 summary_msginfo_change_flags(msginfo, MSG_WATCH_THREAD, 0, MSG_IGNORE_THREAD, 0);
7239 summary_set_row_marks(summaryview, row);
7240 debug_print("Message %d is marked as watch thread\n",
7244 void summary_watch_thread(SummaryView *summaryview)
7246 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7248 gboolean froze = FALSE;
7250 START_LONG_OPERATION(summaryview, FALSE);
7251 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7252 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7253 GTK_CTREE_FUNC(summary_watch_thread_func),
7256 END_LONG_OPERATION(summaryview);
7258 summary_status_show(summaryview);
7261 static void summary_unwatch_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7263 SummaryView *summaryview = (SummaryView *) data;
7266 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7267 g_return_if_fail(msginfo);
7269 summary_msginfo_unset_flags(msginfo, MSG_WATCH_THREAD, 0);
7271 summary_set_row_marks(summaryview, row);
7272 debug_print("Message %d is marked as unwatch thread\n",
7276 void summary_unwatch_thread(SummaryView *summaryview)
7278 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7280 gboolean froze = FALSE;
7282 START_LONG_OPERATION(summaryview, FALSE);
7283 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7284 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7285 GTK_CTREE_FUNC(summary_unwatch_thread_func),
7288 END_LONG_OPERATION(summaryview);
7290 summary_status_show(summaryview);
7293 static void summary_check_watch_thread_func
7294 (GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
7297 gint *found_watch = (gint *) data;
7299 if (*found_watch) return;
7301 msginfo = gtk_ctree_node_get_row_data(ctree, row);
7302 *found_watch = msginfo && MSG_IS_WATCH_THREAD(msginfo->flags);
7306 void summary_toggle_watch_thread(SummaryView *summaryview)
7308 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
7310 gint found_watch = 0;
7312 for (cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next)
7313 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data),
7314 GTK_CTREE_FUNC(summary_check_watch_thread_func),
7318 summary_unwatch_thread(summaryview);
7320 summary_watch_thread(summaryview);
7323 void summary_toggle_show_read_messages(SummaryView *summaryview)
7325 FolderItemUpdateData source;
7326 if (summaryview->folder_item->hide_read_msgs)
7327 summaryview->folder_item->hide_read_msgs = 0;
7329 summaryview->folder_item->hide_read_msgs = 1;
7331 source.item = summaryview->folder_item;
7332 source.update_flags = F_ITEM_UPDATE_NAME;
7334 hooks_invoke(FOLDER_ITEM_UPDATE_HOOKLIST, &source);
7335 summary_show(summaryview, summaryview->folder_item);
7338 static void summary_set_hide_read_msgs_menu (SummaryView *summaryview,
7343 widget = gtk_item_factory_get_item(gtk_item_factory_from_widget(summaryview->mainwin->menubar),
7344 "/View/Hide read messages");
7345 g_object_set_data(G_OBJECT(widget), "dont_toggle",
7346 GINT_TO_POINTER(1));
7347 gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM(widget), action);
7348 g_object_set_data(G_OBJECT(widget), "dont_toggle",
7349 GINT_TO_POINTER(0));
7352 void summary_reflect_prefs_pixmap_theme(SummaryView *summaryview)
7354 GtkWidget *ctree = summaryview->ctree;
7357 stock_pixmap_gdk(ctree, STOCK_PIXMAP_MARK, &markxpm, &markxpmmask);
7358 stock_pixmap_gdk(ctree, STOCK_PIXMAP_DELETED, &deletedxpm, &deletedxpmmask);
7359 stock_pixmap_gdk(ctree, STOCK_PIXMAP_NEW, &newxpm, &newxpmmask);
7360 stock_pixmap_gdk(ctree, STOCK_PIXMAP_UNREAD, &unreadxpm, &unreadxpmmask);
7361 stock_pixmap_gdk(ctree, STOCK_PIXMAP_REPLIED, &repliedxpm, &repliedxpmmask);
7362 stock_pixmap_gdk(ctree, STOCK_PIXMAP_FORWARDED, &forwardedxpm, &forwardedxpmmask);
7363 stock_pixmap_gdk(ctree, STOCK_PIXMAP_CLIP, &clipxpm, &clipxpmmask);
7364 stock_pixmap_gdk(ctree, STOCK_PIXMAP_LOCKED, &lockedxpm, &lockedxpmmask);
7365 stock_pixmap_gdk(ctree, STOCK_PIXMAP_IGNORETHREAD, &ignorethreadxpm, &ignorethreadxpmmask);
7366 stock_pixmap_gdk(ctree, STOCK_PIXMAP_WATCHTHREAD, &watchthreadxpm, &watchthreadxpmmask);
7367 stock_pixmap_gdk(ctree, STOCK_PIXMAP_CLIP_KEY, &clipkeyxpm, &clipkeyxpmmask);
7368 stock_pixmap_gdk(ctree, STOCK_PIXMAP_KEY, &keyxpm, &keyxpmmask);
7369 stock_pixmap_gdk(ctree, STOCK_PIXMAP_GPG_SIGNED, &gpgsignedxpm, &gpgsignedxpmmask);
7370 stock_pixmap_gdk(ctree, STOCK_PIXMAP_CLIP_GPG_SIGNED, &clipgpgsignedxpm, &clipgpgsignedxpmmask);
7371 stock_pixmap_gdk(ctree, STOCK_PIXMAP_SPAM, &spamxpm, &spamxpmmask);
7372 stock_pixmap_gdk(ctree, STOCK_PIXMAP_MOVED, &movedxpm, &movedxpmmask);
7373 stock_pixmap_gdk(ctree, STOCK_PIXMAP_COPIED, &copiedxpm, &copiedxpmmask);
7375 summary_set_folder_pixmap(summaryview, STOCK_PIXMAP_DIR_OPEN);
7377 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_QUICKSEARCH);
7378 gtk_container_remove (GTK_CONTAINER(summaryview->toggle_search),
7379 summaryview->quick_search_pixmap);
7380 gtk_container_add(GTK_CONTAINER(summaryview->toggle_search), pixmap);
7381 gtk_widget_show(pixmap);
7382 summaryview->quick_search_pixmap = pixmap;
7385 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_SELECTION);
7386 gtk_container_remove (GTK_CONTAINER(summaryview->multiple_sel_togbtn),
7387 summaryview->multiple_sel_image);
7388 gtk_container_add(GTK_CONTAINER(summaryview->multiple_sel_togbtn), pixmap);
7389 gtk_widget_show(pixmap);
7390 summaryview->multiple_sel_togbtn = pixmap;
7393 folderview_unselect(summaryview->folderview);
7394 folderview_select(summaryview->folderview, summaryview->folder_item);
7395 summary_set_column_titles(summaryview);
7398 void summary_reflect_prefs_custom_colors(SummaryView *summaryview)
7403 /* re-create colorlabel submenu */
7404 menu = GTK_MENU_SHELL(summaryview->colorlabel_menu);
7405 g_return_if_fail(menu != NULL);
7407 /* clear items. get item pointers. */
7408 for (cur = menu->children; cur != NULL && cur->data != NULL; cur = cur->next) {
7409 gtk_menu_item_remove_submenu(GTK_MENU_ITEM(cur->data));
7411 summary_colorlabel_menu_create(summaryview, TRUE);
7415 * Harvest addresses for selected messages in summary view.
7417 void summary_harvest_address(SummaryView *summaryview)
7419 GtkCTree *ctree = GTK_CTREE( summaryview->ctree );
7425 for( cur = GTK_CLIST(ctree)->selection; cur != NULL && cur->data != NULL; cur = cur->next ) {
7426 msginfo = gtk_ctree_node_get_row_data( ctree, GTK_CTREE_NODE(cur->data) );
7429 msgList = g_list_append( msgList, GUINT_TO_POINTER( msginfo->msgnum ) );
7431 addressbook_harvest( summaryview->folder_item, TRUE, msgList );
7432 g_list_free( msgList );
7435 static regex_t *summary_compile_simplify_regexp(gchar *simplify_subject_regexp)
7438 gchar buf[BUFFSIZE];
7439 regex_t *preg = NULL;
7441 preg = g_new0(regex_t, 1);
7443 err = string_match_precompile(simplify_subject_regexp,
7444 preg, REG_EXTENDED);
7446 regerror(err, preg, buf, BUFFSIZE);
7447 alertpanel_error(_("Regular expression (regexp) error:\n%s"), buf);
7455 void summary_set_prefs_from_folderitem(SummaryView *summaryview, FolderItem *item)
7457 FolderSortKey sort_key;
7458 FolderSortType sort_type;
7459 g_return_if_fail(summaryview != NULL);
7460 g_return_if_fail(item != NULL);
7462 /* Subject simplification */
7463 if(summaryview->simplify_subject_preg) {
7464 regfree(summaryview->simplify_subject_preg);
7465 g_free(summaryview->simplify_subject_preg);
7466 summaryview->simplify_subject_preg = NULL;
7468 if(item->prefs && item->prefs->simplify_subject_regexp &&
7469 item->prefs->simplify_subject_regexp[0] && item->prefs->enable_simplify_subject)
7470 summaryview->simplify_subject_preg = summary_compile_simplify_regexp(item->prefs->simplify_subject_regexp);
7473 sort_key = item->sort_key;
7474 sort_type = item->sort_type;
7476 folder_get_sort_type(item->folder, &sort_key, &sort_type);
7478 summaryview->sort_key = sort_key;
7479 summaryview->sort_type = sort_type;
7482 summaryview->threaded = item->threaded;
7483 summaryview->thread_collapsed = item->thread_collapsed;
7488 void summary_save_prefs_to_folderitem(SummaryView *summaryview, FolderItem *item)
7491 item->sort_key = summaryview->sort_key;
7492 item->sort_type = summaryview->sort_type;
7495 item->threaded = summaryview->threaded;
7496 item->thread_collapsed = summaryview->thread_collapsed;
7499 static gboolean summary_update_msg(gpointer source, gpointer data)
7501 MsgInfoUpdate *msginfo_update = (MsgInfoUpdate *) source;
7502 SummaryView *summaryview = (SummaryView *)data;
7505 g_return_val_if_fail(msginfo_update != NULL, TRUE);
7506 g_return_val_if_fail(summaryview != NULL, FALSE);
7508 if (msginfo_update->msginfo->folder != summaryview->folder_item)
7511 if (msginfo_update->flags & MSGINFO_UPDATE_FLAGS) {
7512 node = gtk_ctree_find_by_row_data(
7513 GTK_CTREE(summaryview->ctree), NULL,
7514 msginfo_update->msginfo);
7517 summary_set_row_marks(summaryview, node);
7523 void summary_update_unread(SummaryView *summaryview, FolderItem *removed_item)
7525 guint new, unread, unreadmarked, marked, total;
7526 guint replied, forwarded, locked, ignored, watched;
7527 static gboolean tips_initialized = FALSE;
7528 #if !(GTK_CHECK_VERSION(2,12,0))
7529 GtkTooltips *tips = summaryview->tips;
7532 if (prefs_common.layout_mode != SMALL_LAYOUT) {
7533 if (tips_initialized) {
7534 summary_set_folder_pixmap(summaryview, STOCK_PIXMAP_DIR_OPEN);
7535 CLAWS_SET_TIP(summaryview->folder_pixmap_eventbox,
7537 tips_initialized = FALSE;
7541 folder_count_total_msgs(&new, &unread, &unreadmarked, &marked, &total,
7542 &replied, &forwarded, &locked, &ignored,
7545 total -= removed_item->total_msgs;
7546 new -= removed_item->new_msgs;
7547 unread -= removed_item->unread_msgs;
7550 if (new > 0 || unread > 0) {
7551 tips_initialized = TRUE;
7552 summary_set_folder_pixmap(summaryview, STOCK_PIXMAP_DIR_OPEN_HRM);
7553 CLAWS_SET_TIP(summaryview->folder_pixmap_eventbox,
7554 _("Go back to the folder list (You have unread messages)"));
7556 tips_initialized = TRUE;
7557 summary_set_folder_pixmap(summaryview, STOCK_PIXMAP_DIR_OPEN);
7558 CLAWS_SET_TIP(summaryview->folder_pixmap_eventbox,
7559 _("Go back to the folder list"));
7563 static gboolean summary_update_folder_item_hook(gpointer source, gpointer data)
7565 FolderItemUpdateData *hookdata = (FolderItemUpdateData *)source;
7566 SummaryView *summaryview = (SummaryView *)data;
7568 g_return_val_if_fail(hookdata != NULL, FALSE);
7569 g_return_val_if_fail(hookdata->item != NULL, FALSE);
7570 g_return_val_if_fail(summaryview != NULL, FALSE);
7572 if (hookdata->update_flags & F_ITEM_UPDATE_NAME) {
7573 gchar *name = folder_item_get_name(hookdata->item);
7574 gtk_label_set_text(GTK_LABEL(summaryview->statlabel_folder), name);
7577 summary_update_unread(summaryview, NULL);
7582 static gboolean summary_update_folder_hook(gpointer source, gpointer data)
7584 FolderUpdateData *hookdata;
7585 SummaryView *summaryview = (SummaryView *)data;
7587 if (hookdata->update_flags & FOLDER_REMOVE_FOLDERITEM)
7588 summary_update_unread(summaryview, hookdata->item);
7590 summary_update_unread(summaryview, NULL);
7596 *\brief change summaryview to display your answer(s) to a message
7598 *\param summaryview The SummaryView ;)
7599 *\param msginfo The message for which answers are searched
7602 static void summary_find_answers (SummaryView *summaryview, MsgInfo *msg)
7604 FolderItem *sent_folder = NULL;
7605 PrefsAccount *account = NULL;
7606 GtkCTreeNode *node = NULL;
7608 if (msg == NULL || msg->msgid == NULL)
7611 account = account_get_reply_account(msg, prefs_common.reply_account_autosel);
7612 if (account == NULL)
7614 sent_folder = account_get_special_folder
7615 (account, F_OUTBOX);
7617 buf = g_strdup_printf("inreplyto matchcase \"%s\"", msg->msgid);
7619 if (sent_folder != summaryview->folder_item) {
7620 folderview_select(summaryview->mainwin->folderview, sent_folder);
7623 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(summaryview->toggle_search), TRUE);
7625 quicksearch_set(summaryview->quicksearch, QUICK_SEARCH_EXTENDED, buf);
7628 node = gtk_ctree_node_nth(GTK_CTREE(summaryview->ctree), 0);
7630 summary_select_node(summaryview, node, TRUE, TRUE);
7633 gint summaryview_export_mbox_list(SummaryView *summaryview)
7634 /* return values: -2 skipped, -1 error, 0 OK */
7636 GSList *list = summary_get_selected_msg_list(summaryview);
7637 gchar *mbox = filesel_select_file_save(_("Export to mbox file"), NULL);
7645 ret = export_list_to_mbox(list, mbox);
7653 void summaryview_lock(SummaryView *summaryview, FolderItem *item)
7655 if (!summaryview || !summaryview->folder_item || !item) {
7659 if (summaryview->folder_item->folder == item->folder) {
7660 gtk_widget_set_sensitive(summaryview->ctree, FALSE);
7663 void summaryview_unlock(SummaryView *summaryview, FolderItem *item)
7665 gtk_widget_set_sensitive(summaryview->ctree, TRUE);