2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2001 Hiroyuki Yamamoto
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 2 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, write to the Free Software
17 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23 #include <gdk/gdkkeysyms.h>
24 #include <gtk/gtkscrolledwindow.h>
25 #include <gtk/gtkwidget.h>
26 #include <gtk/gtkpixmap.h>
27 #include <gtk/gtkctree.h>
28 #include <gtk/gtkcontainer.h>
29 #include <gtk/gtksignal.h>
30 #include <gtk/gtktext.h>
31 #include <gtk/gtkmenu.h>
32 #include <gtk/gtkmenuitem.h>
33 #include <gtk/gtkitemfactory.h>
34 #include <gtk/gtkvbox.h>
35 #include <gtk/gtkhbox.h>
36 #include <gtk/gtkwindow.h>
37 #include <gtk/gtkstyle.h>
38 #include <gtk/gtkarrow.h>
39 #include <gtk/gtkeventbox.h>
40 #include <gtk/gtkstatusbar.h>
41 #include <gtk/gtkmenuitem.h>
53 #include "mainwindow.h"
54 #include "folderview.h"
55 #include "summaryview.h"
56 #include "messageview.h"
57 #include "foldersel.h"
59 #include "procheader.h"
60 #include "headerwindow.h"
61 #include "sourcewindow.h"
62 #include "prefs_common.h"
68 #include "manage_window.h"
69 #include "alertpanel.h"
70 #include "inputdialog.h"
71 #include "statusbar.h"
74 #include "colorlabel.h"
75 #include "addressbook.h"
76 #include "addr_compl.h"
78 #include "prefs_folder_item.h"
79 #include "filtering.h"
81 #include "pixmaps/dir-open.xpm"
82 #include "pixmaps/mark.xpm"
83 #include "pixmaps/deleted.xpm"
84 #include "pixmaps/new.xpm"
85 #include "pixmaps/unread.xpm"
86 #include "pixmaps/replied.xpm"
87 #include "pixmaps/forwarded.xpm"
88 #include "pixmaps/clip.xpm"
89 #include "pixmaps/ignorethread.xpm"
91 #define STATUSBAR_PUSH(mainwin, str) \
93 gtk_statusbar_push(GTK_STATUSBAR(mainwin->statusbar), \
94 mainwin->summaryview_cid, str); \
95 gtkut_widget_wait_for_draw(mainwin->hbox_stat); \
98 #define STATUSBAR_POP(mainwin) \
100 gtk_statusbar_pop(GTK_STATUSBAR(mainwin->statusbar), \
101 mainwin->summaryview_cid); \
104 #define SUMMARY_COL_MARK_WIDTH 10
105 #define SUMMARY_COL_UNREAD_WIDTH 13
106 #define SUMMARY_COL_MIME_WIDTH 10
108 static GdkFont *boldfont;
109 static GdkFont *smallfont;
111 static GtkStyle *bold_style;
112 static GtkStyle *bold_marked_style;
113 static GtkStyle *bold_deleted_style;
114 static GtkStyle *small_style;
115 static GtkStyle *small_marked_style;
116 static GtkStyle *small_deleted_style;
118 static GdkPixmap *folderxpm;
119 static GdkBitmap *folderxpmmask;
121 static GdkPixmap *markxpm;
122 static GdkBitmap *markxpmmask;
123 static GdkPixmap *deletedxpm;
124 static GdkBitmap *deletedxpmmask;
126 static GdkPixmap *newxpm;
127 static GdkBitmap *newxpmmask;
128 static GdkPixmap *unreadxpm;
129 static GdkBitmap *unreadxpmmask;
130 static GdkPixmap *repliedxpm;
131 static GdkBitmap *repliedxpmmask;
132 static GdkPixmap *forwardedxpm;
133 static GdkBitmap *forwardedxpmmask;
134 static GdkPixmap *ignorethreadxpm;
135 static GdkBitmap *ignorethreadxpmmask;
137 static GdkPixmap *clipxpm;
138 static GdkBitmap *clipxpmmask;
140 static void summary_free_msginfo_func (GtkCTree *ctree,
143 static void summary_set_marks_func (GtkCTree *ctree,
146 static void summary_write_cache_func (GtkCTree *ctree,
150 static void summary_set_menu_sensitive (SummaryView *summaryview);
151 static void summary_set_add_sender_menu (SummaryView *summaryview);
153 static void summary_select_node (SummaryView *summaryview,
155 gboolean display_msg);
157 static guint summary_get_msgnum (SummaryView *summaryview,
160 static GtkCTreeNode *summary_find_next_unread_msg
161 (SummaryView *summaryview,
162 GtkCTreeNode *current_node);
163 static GtkCTreeNode *summary_find_next_marked_msg
164 (SummaryView *summaryview,
165 GtkCTreeNode *current_node);
166 static GtkCTreeNode *summary_find_prev_marked_msg
167 (SummaryView *summaryview,
168 GtkCTreeNode *current_node);
169 static GtkCTreeNode *summary_find_next_labeled_msg
170 (SummaryView *summaryview,
171 GtkCTreeNode *current_node);
172 static GtkCTreeNode *summary_find_prev_labeled_msg
173 (SummaryView *summaryview,
174 GtkCTreeNode *current_node);
175 static GtkCTreeNode *summary_find_msg_by_msgnum
176 (SummaryView *summaryview,
179 static GtkCTreeNode *summary_find_prev_unread_msg
180 (SummaryView *summaryview,
181 GtkCTreeNode *current_node);
184 static void summary_update_status (SummaryView *summaryview);
186 /* display functions */
187 static void summary_status_show (SummaryView *summaryview);
188 static void summary_set_ctree_from_list (SummaryView *summaryview,
190 static void summary_set_header (gchar *text[],
192 static void summary_display_msg (SummaryView *summaryview,
194 gboolean new_window);
195 static void summary_toggle_view (SummaryView *summaryview);
196 static void summary_set_row_marks (SummaryView *summaryview,
199 /* message handling */
200 static void summary_mark_row (SummaryView *summaryview,
202 static void summary_mark_row_as_read (SummaryView *summaryview,
204 static void summary_mark_row_as_unread (SummaryView *summaryview,
206 static void summary_delete_row (SummaryView *summaryview,
208 static void summary_unmark_row (SummaryView *summaryview,
210 static void summary_move_row_to (SummaryView *summaryview,
212 FolderItem *to_folder);
213 static void summary_copy_row_to (SummaryView *summaryview,
215 FolderItem *to_folder);
217 static void summary_delete_duplicated_func
220 SummaryView *summaryview);
222 static void summary_execute_move (SummaryView *summaryview);
223 static void summary_execute_move_func (GtkCTree *ctree,
226 static void summary_execute_copy (SummaryView *summaryview);
227 static void summary_execute_copy_func (GtkCTree *ctree,
230 static void summary_execute_delete (SummaryView *summaryview);
231 static void summary_execute_delete_func (GtkCTree *ctree,
234 static void summary_ignore_thread(SummaryView *summaryview);
235 static void summary_unignore_thread(SummaryView *summaryview);
237 static void summary_unthread_for_exec (SummaryView *summaryview);
238 static void summary_unthread_for_exec_func (GtkCTree *ctree,
242 static void summary_filter_func (GtkCTree *ctree,
246 static void summary_colorlabel_menu_item_activate_cb
249 static void summary_colorlabel_menu_item_activate_item_cb
250 (GtkMenuItem *label_menu_item,
252 static void summary_colorlabel_menu_create(SummaryView *summaryview);
254 /* callback functions */
255 static void summary_toggle_pressed (GtkWidget *eventbox,
256 GdkEventButton *event,
257 SummaryView *summaryview);
258 static void summary_button_pressed (GtkWidget *ctree,
259 GdkEventButton *event,
260 SummaryView *summaryview);
261 static void summary_button_released (GtkWidget *ctree,
262 GdkEventButton *event,
263 SummaryView *summaryview);
264 static void summary_key_pressed (GtkWidget *ctree,
266 SummaryView *summaryview);
267 static void summary_open_row (GtkSCTree *sctree,
268 SummaryView *summaryview);
269 static void summary_tree_expanded (GtkCTree *ctree,
271 SummaryView *summaryview);
272 static void summary_tree_collapsed (GtkCTree *ctree,
274 SummaryView *summaryview);
275 static void summary_selected (GtkCTree *ctree,
278 SummaryView *summaryview);
279 static void summary_col_resized (GtkCList *clist,
282 SummaryView *summaryview);
283 static void summary_reply_cb (SummaryView *summaryview,
286 static void summary_show_all_header_cb (SummaryView *summaryview,
290 static void summary_num_clicked (GtkWidget *button,
291 SummaryView *summaryview);
292 static void summary_score_clicked (GtkWidget *button,
293 SummaryView *summaryview);
294 static void summary_size_clicked (GtkWidget *button,
295 SummaryView *summaryview);
296 static void summary_date_clicked (GtkWidget *button,
297 SummaryView *summaryview);
298 static void summary_from_clicked (GtkWidget *button,
299 SummaryView *summaryview);
300 static void summary_subject_clicked (GtkWidget *button,
301 SummaryView *summaryview);
302 static void summary_mark_clicked (GtkWidget *button,
303 SummaryView *summaryview);
305 static void summary_start_drag (GtkWidget *widget,
308 SummaryView *summaryview);
309 static void summary_drag_data_get (GtkWidget *widget,
310 GdkDragContext *drag_context,
311 GtkSelectionData *selection_data,
314 SummaryView *summaryview);
316 /* custom compare functions for sorting */
318 static gint summary_cmp_by_num (GtkCList *clist,
321 static gint summary_cmp_by_size (GtkCList *clist,
324 static gint summary_cmp_by_date (GtkCList *clist,
327 static gint summary_cmp_by_from (GtkCList *clist,
330 static gint summary_cmp_by_subject (GtkCList *clist,
333 static gint summary_cmp_by_score (GtkCList *clist,
336 static gint summary_cmp_by_label (GtkCList *clist,
341 static void summary_mark_all_read (SummaryView *summaryview);
344 GtkTargetEntry summary_drag_types[1] =
346 {"text/plain", GTK_TARGET_SAME_APP, TARGET_DUMMY}
349 static GtkItemFactoryEntry summary_popup_entries[] =
351 {N_("/M_ove..."), NULL, summary_move_to, 0, NULL},
352 {N_("/_Copy..."), NULL, summary_copy_to, 0, NULL},
353 {N_("/_Delete"), NULL, summary_delete, 0, NULL},
354 {N_("/E_xecute"), NULL, summary_execute, 0, NULL},
355 {N_("/_Mark"), NULL, NULL, 0, "<Branch>"},
356 {N_("/_Mark/_Mark"), NULL, summary_mark, 0, NULL},
357 {N_("/_Mark/_Unmark"), NULL, summary_unmark, 0, NULL},
358 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
359 {N_("/_Mark/Mark as unr_ead"), NULL, summary_mark_as_unread, 0, NULL},
360 {N_("/_Mark/Mark as rea_d"), NULL, summary_mark_as_read, 0, NULL},
362 {N_("/_Mark/Mark all read"), NULL, summary_mark_all_read, 0, NULL},
364 {N_("/_Mark/Ignore thread"), NULL, summary_ignore_thread, 0, NULL},
365 {N_("/_Mark/Unignore thread"), NULL, summary_unignore_thread, 0, NULL},
366 {N_("/Color la_bel"), NULL, NULL, 0, NULL},
368 {N_("/---"), NULL, NULL, 0, "<Separator>"},
369 {N_("/_Reply"), NULL, summary_reply_cb, COMPOSE_REPLY, NULL},
370 {N_("/Repl_y to sender"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_SENDER, NULL},
371 {N_("/Follow-up and reply to"), NULL, summary_reply_cb, COMPOSE_FOLLOWUP_AND_REPLY_TO, NULL},
372 {N_("/Reply to a_ll"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_ALL, NULL},
373 {N_("/_Forward"), NULL, summary_reply_cb, COMPOSE_FORWARD, NULL},
374 {N_("/Forward as a_ttachment"),
375 NULL, summary_reply_cb, COMPOSE_FORWARD_AS_ATTACH, NULL},
376 {N_("/---"), NULL, NULL, 0, "<Separator>"},
377 {N_("/Add sender to address _book"),
378 NULL, NULL, 0, NULL},
379 {N_("/---"), NULL, NULL, 0, "<Separator>"},
380 {N_("/Open in new _window"), NULL, summary_open_msg, 0, NULL},
381 {N_("/View so_urce"), NULL, summary_view_source, 0, NULL},
382 {N_("/Show all _header"), NULL, summary_show_all_header_cb, 0, NULL},
383 {N_("/Re-_edit"), NULL, summary_reedit, 0, NULL},
384 {N_("/---"), NULL, NULL, 0, "<Separator>"},
385 {N_("/_Save as..."), NULL, summary_save_as, 0, NULL},
386 {N_("/_Print..."), NULL, summary_print, 0, NULL},
387 {N_("/---"), NULL, NULL, 0, "<Separator>"},
388 {N_("/Select _all"), NULL, summary_select_all, 0, NULL}
391 SummaryView *summary_create(void)
393 SummaryView *summaryview;
394 gchar *titles[N_SUMMARY_COLS] = {_("M"), _("U")};
396 GtkWidget *scrolledwin;
399 GtkWidget *statlabel_folder;
400 GtkWidget *statlabel_select;
401 GtkWidget *statlabel_msgs;
402 GtkWidget *toggle_eventbox;
403 GtkWidget *toggle_arrow;
404 GtkWidget *popupmenu;
405 GtkItemFactory *popupfactory;
409 debug_print(_("Creating summary view...\n"));
410 summaryview = g_new0(SummaryView, 1);
412 vbox = gtk_vbox_new(FALSE, 2);
414 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
415 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
416 GTK_POLICY_AUTOMATIC,
418 gtk_box_pack_start(GTK_BOX(vbox), scrolledwin, TRUE, TRUE, 0);
419 gtk_widget_set_usize(vbox,
420 prefs_common.summaryview_width,
421 prefs_common.summaryview_height);
423 if (prefs_common.trans_hdr) {
424 titles[S_COL_NUMBER] = _("No.");
425 titles[S_COL_DATE] = _("Date");
426 titles[S_COL_FROM] = _("From");
427 titles[S_COL_SUBJECT] = _("Subject");
429 titles[S_COL_NUMBER] = "No.";
430 titles[S_COL_DATE] = "Date";
431 titles[S_COL_FROM] = "From";
432 titles[S_COL_SUBJECT] = "Subject";
434 titles[S_COL_SIZE] = _("Size");
435 titles[S_COL_SCORE] = _("Score");
437 ctree = gtk_sctree_new_with_titles(N_SUMMARY_COLS, S_COL_SUBJECT, titles);
438 gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
439 GTK_CLIST(ctree)->hadjustment);
440 gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
441 GTK_CLIST(ctree)->vadjustment);
442 gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
443 gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_EXTENDED);
444 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_MARK,
446 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_UNREAD,
448 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_MIME,
450 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_NUMBER,
452 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_SCORE,
454 gtk_clist_set_column_justification(GTK_CLIST(ctree), S_COL_SIZE,
456 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_MARK,
457 SUMMARY_COL_MARK_WIDTH);
458 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_UNREAD,
459 SUMMARY_COL_UNREAD_WIDTH);
460 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_MIME,
461 SUMMARY_COL_MIME_WIDTH);
462 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_NUMBER,
463 prefs_common.summary_col_number);
464 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_SCORE,
465 prefs_common.summary_col_score);
466 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_SIZE,
467 prefs_common.summary_col_size);
468 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_DATE,
469 prefs_common.summary_col_date);
470 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_FROM,
471 prefs_common.summary_col_from);
472 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_SUBJECT,
473 prefs_common.summary_col_subject);
474 gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
475 gtk_ctree_set_expander_style(GTK_CTREE(ctree),
476 GTK_CTREE_EXPANDER_SQUARE);
478 gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_NONE);
479 gtk_ctree_set_expander_style(GTK_CTREE(ctree),
480 GTK_CTREE_EXPANDER_TRIANGLE);
482 gtk_ctree_set_indent(GTK_CTREE(ctree), 18);
483 gtk_object_set_user_data(GTK_OBJECT(ctree), summaryview);
485 /* don't let title buttons take key focus */
486 for (i = 0; i < N_SUMMARY_COLS; i++)
487 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(ctree)->column[i].button,
490 /* connect signal to the buttons for sorting */
492 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_NUMBER].button),
494 GTK_SIGNAL_FUNC(summary_num_clicked),
497 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_SCORE].button),
499 GTK_SIGNAL_FUNC(summary_score_clicked),
502 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_SIZE].button),
504 GTK_SIGNAL_FUNC(summary_size_clicked),
507 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_DATE].button),
509 GTK_SIGNAL_FUNC(summary_date_clicked),
512 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_FROM].button),
514 GTK_SIGNAL_FUNC(summary_from_clicked),
517 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_SUBJECT].button),
519 GTK_SIGNAL_FUNC(summary_subject_clicked),
522 (GTK_OBJECT(GTK_CLIST(ctree)->column[S_COL_MARK].button),
524 GTK_SIGNAL_FUNC(summary_mark_clicked),
527 /* create status label */
528 hbox = gtk_hbox_new(FALSE, 0);
529 gtk_box_pack_end(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
531 statlabel_folder = gtk_label_new("");
532 gtk_box_pack_start(GTK_BOX(hbox), statlabel_folder, FALSE, FALSE, 2);
533 statlabel_select = gtk_label_new("");
534 gtk_box_pack_start(GTK_BOX(hbox), statlabel_select, FALSE, FALSE, 16);
536 /* toggle view button */
537 toggle_eventbox = gtk_event_box_new();
538 gtk_box_pack_end(GTK_BOX(hbox), toggle_eventbox, FALSE, FALSE, 4);
539 toggle_arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
540 gtk_container_add(GTK_CONTAINER(toggle_eventbox), toggle_arrow);
542 statlabel_msgs = gtk_label_new("");
543 gtk_box_pack_end(GTK_BOX(hbox), statlabel_msgs, FALSE, FALSE, 4);
545 /* create popup menu */
546 n_entries = sizeof(summary_popup_entries) /
547 sizeof(summary_popup_entries[0]);
548 popupmenu = menu_create_items(summary_popup_entries, n_entries,
549 "<SummaryView>", &popupfactory,
552 /* connect signals */
553 gtk_signal_connect(GTK_OBJECT(ctree), "tree_select_row",
554 GTK_SIGNAL_FUNC(summary_selected), summaryview);
555 gtk_signal_connect(GTK_OBJECT(ctree), "button_press_event",
556 GTK_SIGNAL_FUNC(summary_button_pressed),
558 gtk_signal_connect(GTK_OBJECT(ctree), "button_release_event",
559 GTK_SIGNAL_FUNC(summary_button_released),
561 gtk_signal_connect(GTK_OBJECT(ctree), "key_press_event",
562 GTK_SIGNAL_FUNC(summary_key_pressed), summaryview);
563 gtk_signal_connect(GTK_OBJECT(ctree), "resize_column",
564 GTK_SIGNAL_FUNC(summary_col_resized), summaryview);
565 gtk_signal_connect(GTK_OBJECT(ctree), "open_row",
566 GTK_SIGNAL_FUNC(summary_open_row), summaryview);
568 gtk_signal_connect_after(GTK_OBJECT(ctree), "tree_expand",
569 GTK_SIGNAL_FUNC(summary_tree_expanded),
571 gtk_signal_connect_after(GTK_OBJECT(ctree), "tree_collapse",
572 GTK_SIGNAL_FUNC(summary_tree_collapsed),
575 gtk_signal_connect(GTK_OBJECT(ctree), "start_drag",
576 GTK_SIGNAL_FUNC(summary_start_drag),
578 gtk_signal_connect(GTK_OBJECT(ctree), "drag_data_get",
579 GTK_SIGNAL_FUNC(summary_drag_data_get),
582 gtk_signal_connect(GTK_OBJECT(toggle_eventbox), "button_press_event",
583 GTK_SIGNAL_FUNC(summary_toggle_pressed),
586 summaryview->vbox = vbox;
587 summaryview->scrolledwin = scrolledwin;
588 summaryview->ctree = ctree;
589 summaryview->hbox = hbox;
590 summaryview->statlabel_folder = statlabel_folder;
591 summaryview->statlabel_select = statlabel_select;
592 summaryview->statlabel_msgs = statlabel_msgs;
593 summaryview->toggle_eventbox = toggle_eventbox;
594 summaryview->toggle_arrow = toggle_arrow;
595 summaryview->popupmenu = popupmenu;
596 summaryview->popupfactory = popupfactory;
597 summaryview->msg_is_toggled_on = TRUE;
598 summaryview->sort_mode = SORT_BY_NONE;
599 summaryview->sort_type = GTK_SORT_ASCENDING;
601 summary_change_display_item(summaryview);
603 gtk_widget_show_all(vbox);
608 void summary_init(SummaryView *summaryview)
613 PIXMAP_CREATE(summaryview->ctree, markxpm, markxpmmask, mark_xpm);
614 PIXMAP_CREATE(summaryview->ctree, deletedxpm, deletedxpmmask,
616 PIXMAP_CREATE(summaryview->ctree, newxpm, newxpmmask, new_xpm);
617 PIXMAP_CREATE(summaryview->ctree, unreadxpm, unreadxpmmask, unread_xpm);
618 PIXMAP_CREATE(summaryview->ctree, repliedxpm, repliedxpmmask,
620 PIXMAP_CREATE(summaryview->ctree, forwardedxpm, forwardedxpmmask,
622 PIXMAP_CREATE(summaryview->ctree, ignorethreadxpm, ignorethreadxpmmask,
624 PIXMAP_CREATE(summaryview->ctree, clipxpm, clipxpmmask, clip_xpm);
625 PIXMAP_CREATE(summaryview->hbox, folderxpm, folderxpmmask,
628 pixmap = gtk_pixmap_new(clipxpm, clipxpmmask);
629 gtk_clist_set_column_widget(GTK_CLIST(summaryview->ctree),
631 gtk_widget_show(pixmap);
634 small_style = gtk_style_copy
635 (gtk_widget_get_style(summaryview->ctree));
637 smallfont = gdk_fontset_load(SMALL_FONT);
638 small_style->font = smallfont;
639 small_marked_style = gtk_style_copy(small_style);
640 small_marked_style->fg[GTK_STATE_NORMAL] =
641 summaryview->color_marked;
642 small_deleted_style = gtk_style_copy(small_style);
643 small_deleted_style->fg[GTK_STATE_NORMAL] =
644 summaryview->color_dim;
647 bold_style = gtk_style_copy
648 (gtk_widget_get_style(summaryview->ctree));
650 boldfont = gdk_fontset_load(BOLD_FONT);
651 bold_style->font = boldfont;
652 bold_marked_style = gtk_style_copy(bold_style);
653 bold_marked_style->fg[GTK_STATE_NORMAL] =
654 summaryview->color_marked;
655 bold_deleted_style = gtk_style_copy(bold_style);
656 bold_deleted_style->fg[GTK_STATE_NORMAL] =
657 summaryview->color_dim;
660 style = gtk_style_copy(gtk_widget_get_style
661 (summaryview->statlabel_folder));
663 gtk_widget_set_style(summaryview->statlabel_folder, style);
664 gtk_widget_set_style(summaryview->statlabel_select, style);
665 gtk_widget_set_style(summaryview->statlabel_msgs, style);
667 pixmap = gtk_pixmap_new(folderxpm, folderxpmmask);
668 gtk_box_pack_start(GTK_BOX(summaryview->hbox), pixmap, FALSE, FALSE, 4);
669 gtk_box_reorder_child(GTK_BOX(summaryview->hbox), pixmap, 0);
670 gtk_widget_show(pixmap);
672 summary_clear_list(summaryview);
673 summary_colorlabel_menu_create(summaryview);
674 summary_set_menu_sensitive(summaryview);
678 GtkCTreeNode * summary_find_next_important_score(SummaryView *summaryview,
679 GtkCTreeNode *current_node)
681 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
684 gint best_score = MIN_SCORE;
685 GtkCTreeNode *best_node = NULL;
688 /*node = current_node;*/
689 node = GTK_CTREE_NODE_NEXT(current_node);
691 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
693 for (; node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
694 msginfo = gtk_ctree_node_get_row_data(ctree, node);
695 if (msginfo->score >= summaryview->important_score)
697 if (msginfo->score > best_score) {
698 best_score = msginfo->score;
709 GtkCTreeNode * summary_find_prev_important_score(SummaryView *summaryview,
710 GtkCTreeNode *current_node)
712 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
715 gint best_score = MIN_SCORE;
716 GtkCTreeNode *best_node = NULL;
719 /*node = current_node;*/
720 node = GTK_CTREE_NODE_PREV(current_node);
722 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
724 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
725 msginfo = gtk_ctree_node_get_row_data(ctree, node);
726 if (msginfo->score >= summaryview->important_score)
728 if (msginfo->score > best_score) {
729 best_score = msginfo->score;
740 gboolean summary_show(SummaryView *summaryview, FolderItem *item,
741 gboolean update_cache)
743 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
745 GSList *mlist = NULL;
748 guint selected_msgnum = 0;
749 guint displayed_msgnum = 0;
750 GtkCTreeNode *selected_node = summaryview->folderview->selected;
754 static gboolean locked = FALSE;
760 STATUSBAR_POP(summaryview->mainwin);
762 is_refresh = (!prefs_common.open_inbox_on_inc &&
763 item == summaryview->folder_item &&
764 update_cache == FALSE) ? TRUE : FALSE;
766 selected_msgnum = summary_get_msgnum(summaryview,
767 summaryview->selected);
768 displayed_msgnum = summary_get_msgnum(summaryview,
769 summaryview->displayed);
772 /* process the marks if any */
773 if (summaryview->mainwin->lock_count == 0 &&
774 (summaryview->moved > 0 || summaryview->copied > 0)) {
777 val = alertpanel(_("Process mark"),
778 _("Some marks are left. Process it?"),
779 _("Yes"), _("No"), _("Cancel"));
780 if (G_ALERTDEFAULT == val)
781 summary_execute(summaryview);
782 else if (G_ALERTALTERNATE == val)
783 summary_write_cache(summaryview);
788 folder_update_op_count();
790 else if (!summaryview->filtering_happened) {
791 summary_write_cache(summaryview);
794 summaryview->filtering_happened = FALSE;
796 summaryview->folderview->opened = selected_node;
798 gtk_clist_freeze(GTK_CLIST(ctree));
800 summary_clear_list(summaryview);
801 summary_set_menu_sensitive(summaryview);
803 messageview_clear(summaryview->messageview);
806 if (!item || !item->path || !item->parent || item->no_select ||
807 (item->folder->type == F_MH &&
808 ((buf = folder_item_get_path(item)) == NULL ||
809 change_dir(buf) < 0))) {
811 debug_print(_("empty folder\n\n"));
813 messageview_clear(summaryview->messageview);
814 summary_clear_all(summaryview);
815 summaryview->folder_item = item;
816 gtk_clist_thaw(GTK_CLIST(ctree));
822 summaryview->folder_item = item;
824 gtk_signal_handler_block_by_data(GTK_OBJECT(ctree), summaryview);
826 buf = g_strdup_printf(_("Scanning folder (%s)..."), item->path);
827 debug_print("%s\n", buf);
828 STATUSBAR_PUSH(summaryview->mainwin, buf);
831 main_window_cursor_wait(summaryview->mainwin);
833 mlist = item->folder->get_msg_list(item->folder, item, !update_cache);
835 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
836 MsgInfo * msginfo = (MsgInfo *) cur->data;
838 msginfo->score = score_message(global_scoring, msginfo);
839 if (msginfo->score != MAX_SCORE &&
840 msginfo->score != MIN_SCORE) {
841 msginfo->score += score_message(item->prefs->scoring,
846 summaryview->killed_messages = NULL;
847 if ((global_scoring || item->prefs->scoring) &&
848 (item->folder->type == F_NEWS)) {
853 kill_score = prefs_common.kill_score;
854 if (item->prefs->kill_score > kill_score)
855 kill_score = item->prefs->kill_score;
856 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
857 MsgInfo * msginfo = (MsgInfo *) cur->data;
859 if (MSG_IS_NEWS(msginfo->flags) &&
860 (msginfo->score <= kill_score))
861 summaryview->killed_messages = g_slist_append(summaryview->killed_messages, msginfo);
863 not_killed = g_slist_append(not_killed,
870 STATUSBAR_POP(summaryview->mainwin);
872 /* set ctree and hash table from the msginfo list
873 creating thread, and count the number of messages */
874 summary_set_ctree_from_list(summaryview, mlist);
878 summary_write_cache(summaryview);
880 gtk_signal_handler_unblock_by_data(GTK_OBJECT(ctree), summaryview);
882 gtk_clist_thaw(GTK_CLIST(ctree));
885 sort_mode = prefs_folder_item_get_sort_mode(item);
886 sort_type = prefs_folder_item_get_sort_type(item);
888 if (sort_mode != SORT_BY_NONE) {
889 summaryview->sort_mode = sort_mode;
890 if (sort_type == GTK_SORT_DESCENDING)
891 summaryview->sort_type = GTK_SORT_ASCENDING;
893 summaryview->sort_type = GTK_SORT_DESCENDING;
895 summary_sort(summaryview, sort_mode);
899 summaryview->displayed =
900 summary_find_msg_by_msgnum(summaryview,
902 if (!summaryview->displayed)
903 messageview_clear(summaryview->messageview);
904 summary_select_by_msgnum(summaryview, selected_msgnum);
905 if (!summaryview->selected) {
906 /* no selected message - select first unread
907 message, but do not display it */
908 node = summary_find_next_unread_msg(summaryview, NULL);
909 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL)
910 node = GTK_CTREE_NODE
911 (GTK_CLIST(ctree)->row_list_end);
912 summary_select_node(summaryview, node, FALSE);
915 /* select first unread message */
916 if (sort_mode == SORT_BY_SCORE)
917 node = summary_find_next_important_score(summaryview,
920 node = summary_find_next_unread_msg(summaryview, NULL);
922 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL)
923 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list_end);
924 summary_select_node(summaryview, node,
925 prefs_common.open_unread_on_enter);
928 summary_status_show(summaryview);
930 summary_set_menu_sensitive(summaryview);
932 main_window_set_toolbar_sensitive(summaryview->mainwin);
935 STATUSBAR_PUSH(summaryview->mainwin, _("Done."));
937 main_window_cursor_normal(summaryview->mainwin);
943 void summary_clear_list(SummaryView *summaryview)
945 GtkCList *clist = GTK_CLIST(summaryview->ctree);
949 gtk_clist_freeze(clist);
951 for(cur = summaryview->killed_messages ; cur != NULL ;
952 cur = g_slist_next(cur)) {
953 MsgInfo * msginfo = (MsgInfo *) cur->data;
955 procmsg_msginfo_free(msginfo);
957 g_slist_free(summaryview->killed_messages);
958 summaryview->killed_messages = NULL;
960 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree),
961 NULL, summary_free_msginfo_func, NULL);
963 summaryview->folder_item = NULL;
965 summaryview->display_msg = FALSE;
967 summaryview->selected = NULL;
968 summaryview->displayed = NULL;
969 summaryview->newmsgs = summaryview->unread = 0;
970 summaryview->messages = summaryview->total_size = 0;
971 summaryview->deleted = summaryview->moved = 0;
972 summaryview->copied = 0;
973 if (summaryview->msgid_table) {
974 g_hash_table_destroy(summaryview->msgid_table);
975 summaryview->msgid_table = NULL;
977 if (summaryview->subject_table) {
978 g_hash_table_destroy(summaryview->subject_table);
979 summaryview->subject_table = NULL;
981 summaryview->mlist = NULL;
982 if (summaryview->folder_table) {
983 g_hash_table_destroy(summaryview->folder_table);
984 summaryview->folder_table = NULL;
986 summaryview->sort_mode = SORT_BY_NONE;
987 summaryview->sort_type = GTK_SORT_ASCENDING;
989 gtk_clist_clear(clist);
990 optimal_width = gtk_clist_optimal_column_width(clist, S_COL_SUBJECT);
991 gtk_clist_set_column_width(clist, S_COL_SUBJECT, optimal_width);
993 gtk_clist_thaw(clist);
996 void summary_clear_all(SummaryView *summaryview)
998 summary_clear_list(summaryview);
999 summary_set_menu_sensitive(summaryview);
1000 main_window_set_toolbar_sensitive(summaryview->mainwin);
1001 summary_status_show(summaryview);
1004 SummarySelection summary_get_selection_type(SummaryView *summaryview)
1006 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1007 SummarySelection selection;
1009 if (!clist->row_list)
1010 selection = SUMMARY_NONE;
1011 else if (!clist->selection)
1012 selection = SUMMARY_SELECTED_NONE;
1013 else if (!clist->selection->next)
1014 selection = SUMMARY_SELECTED_SINGLE;
1016 selection = SUMMARY_SELECTED_MULTIPLE;
1021 static void summary_set_menu_sensitive(SummaryView *summaryview)
1023 GtkItemFactory *ifactory = summaryview->popupfactory;
1024 SummarySelection selection;
1027 selection = summary_get_selection_type(summaryview);
1028 main_window_set_menu_sensitive(summaryview->mainwin);
1030 if (selection == SUMMARY_NONE) {
1033 submenu = gtk_item_factory_get_widget
1034 (summaryview->popupfactory, "/Mark");
1035 menu_set_insensitive_all(GTK_MENU_SHELL(submenu));
1036 menu_set_insensitive_all
1037 (GTK_MENU_SHELL(summaryview->popupmenu));
1041 if (summaryview->folder_item->folder->type != F_NEWS) {
1042 menu_set_sensitive(ifactory, "/Move...", TRUE);
1043 menu_set_sensitive(ifactory, "/Copy...", TRUE);
1044 if (summaryview->folder_item->stype != F_TRASH)
1045 menu_set_sensitive(ifactory, "/Delete", TRUE);
1048 menu_set_sensitive(ifactory, "/Execute", TRUE);
1050 menu_set_sensitive(ifactory, "/Mark", TRUE);
1051 menu_set_sensitive(ifactory, "/Mark/Mark", TRUE);
1052 menu_set_sensitive(ifactory, "/Mark/Unmark", TRUE);
1054 menu_set_sensitive(ifactory, "/Mark/Mark as unread", TRUE);
1055 menu_set_sensitive(ifactory, "/Mark/Mark as read", TRUE);
1057 menu_set_sensitive(ifactory, "/Mark/Mark all read", TRUE);
1059 menu_set_sensitive(ifactory, "/Mark/Ignore thread", TRUE);
1060 menu_set_sensitive(ifactory, "/Mark/Unignore thread", TRUE);
1062 menu_set_sensitive(ifactory, "/Color label", TRUE);
1064 sens = (selection == SUMMARY_SELECTED_MULTIPLE) ? FALSE : TRUE;
1065 menu_set_sensitive(ifactory, "/Reply", sens);
1066 menu_set_sensitive(ifactory, "/Reply to sender", sens);
1067 menu_set_sensitive(ifactory, "/Reply to all", sens);
1068 menu_set_sensitive(ifactory, "/Forward", TRUE);
1069 menu_set_sensitive(ifactory, "/Forward as attachment", TRUE);
1071 menu_set_sensitive(ifactory, "/Open in new window", sens);
1072 menu_set_sensitive(ifactory, "/View source", sens);
1073 menu_set_sensitive(ifactory, "/Show all header", sens);
1074 if ((summaryview->folder_item->stype == F_DRAFT) ||
1075 (summaryview->folder_item->stype == F_OUTBOX) ||
1076 (summaryview->folder_item->stype == F_QUEUE))
1077 menu_set_sensitive(ifactory, "/Re-edit", sens);
1079 menu_set_sensitive(ifactory, "/Save as...", sens);
1080 menu_set_sensitive(ifactory, "/Print...", TRUE);
1082 menu_set_sensitive(ifactory, "/Select all", TRUE);
1084 if (summaryview->folder_item->folder->account)
1085 sens = summaryview->folder_item->folder->account->protocol
1089 menu_set_sensitive(ifactory, "/Follow-up and reply to", sens);
1092 static void summary_set_add_sender_menu(SummaryView *summaryview)
1099 menu = gtk_item_factory_get_item(summaryview->popupfactory,
1100 "/Add sender to address book");
1101 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
1102 summaryview->selected);
1103 if (!msginfo || !msginfo->from) {
1104 gtk_widget_set_sensitive(menu, FALSE);
1105 submenu = gtk_menu_new();
1106 gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu), submenu);
1110 gtk_widget_set_sensitive(menu, TRUE);
1111 Xstrdup_a(from, msginfo->from, return);
1112 eliminate_address_comment(from);
1113 extract_address(from);
1114 addressbook_add_submenu(menu, msginfo->fromname, from, NULL);
1118 void summary_select_next_unread(SummaryView *summaryview)
1121 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1123 node = summary_find_next_unread_msg(summaryview,
1124 summaryview->selected);
1127 gtkut_ctree_expand_parent_all(ctree, node);
1128 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1129 gtk_sctree_select(GTK_SCTREE(ctree), node);
1131 /* BUGFIX: select next unread message
1132 * REVERT: causes incorrect folder stats
1133 * gtk_ctree_node_moveto(ctree, node, -1, 0.5, 0.0);
1136 if (summaryview->displayed == node)
1137 summaryview->displayed = NULL;
1138 summary_display_msg(summaryview, node, FALSE);
1142 val = alertpanel(_("No unread message"),
1143 _("No unread message found. Go to next folder?"),
1144 _("Yes"), _("No"), NULL);
1145 if (val == G_ALERTDEFAULT) {
1146 if (gtk_signal_n_emissions_by_name
1147 (GTK_OBJECT(ctree), "key_press_event") > 0)
1148 gtk_signal_emit_stop_by_name
1151 folderview_select_next_unread(summaryview->folderview);
1156 void summary_select_next_marked(SummaryView *summaryview)
1159 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1161 node = summary_find_next_marked_msg(summaryview,
1162 summaryview->selected);
1167 val = alertpanel(_("No more marked messages"),
1168 _("No marked message found. "
1169 "Search from the beginning?"),
1170 _("Yes"), _("No"), NULL);
1171 if (val != G_ALERTDEFAULT) return;
1172 node = summary_find_next_marked_msg(summaryview,
1176 alertpanel_notice(_("No marked messages."));
1178 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1179 gtk_sctree_select(GTK_SCTREE(ctree), node);
1180 if (summaryview->displayed == node)
1181 summaryview->displayed = NULL;
1182 summary_display_msg(summaryview, node, FALSE);
1186 void summary_select_prev_marked(SummaryView *summaryview)
1189 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1191 node = summary_find_prev_marked_msg(summaryview,
1192 summaryview->selected);
1197 val = alertpanel(_("No more marked messages"),
1198 _("No marked message found. "
1199 "Search from the end?"),
1200 _("Yes"), _("No"), NULL);
1201 if (val != G_ALERTDEFAULT) return;
1202 node = summary_find_prev_marked_msg(summaryview,
1206 alertpanel_notice(_("No marked messages."));
1208 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1209 gtk_sctree_select(GTK_SCTREE(ctree), node);
1210 if (summaryview->displayed == node)
1211 summaryview->displayed = NULL;
1212 summary_display_msg(summaryview, node, FALSE);
1216 void summary_select_next_labeled(SummaryView *summaryview)
1219 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1221 node = summary_find_next_labeled_msg(summaryview,
1222 summaryview->selected);
1227 val = alertpanel(_("No more labeled messages"),
1228 _("No labeled message found. "
1229 "Search from the beginning?"),
1230 _("Yes"), _("No"), NULL);
1231 if (val != G_ALERTDEFAULT) return;
1232 node = summary_find_next_labeled_msg(summaryview,
1236 alertpanel_notice(_("No labeled messages."));
1238 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1239 gtk_sctree_select(GTK_SCTREE(ctree), node);
1240 if (summaryview->displayed == node)
1241 summaryview->displayed = NULL;
1242 summary_display_msg(summaryview, node, FALSE);
1246 void summary_select_prev_labeled(SummaryView *summaryview)
1249 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1251 node = summary_find_prev_labeled_msg(summaryview,
1252 summaryview->selected);
1257 val = alertpanel(_("No more labeled messages"),
1258 _("No labeled message found. "
1259 "Search from the end?"),
1260 _("Yes"), _("No"), NULL);
1261 if (val != G_ALERTDEFAULT) return;
1262 node = summary_find_prev_labeled_msg(summaryview,
1266 alertpanel_notice(_("No labeled messages."));
1268 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1269 gtk_sctree_select(GTK_SCTREE(ctree), node);
1270 if (summaryview->displayed == node)
1271 summaryview->displayed = NULL;
1272 summary_display_msg(summaryview, node, FALSE);
1276 void summary_select_by_msgnum(SummaryView *summaryview, guint msgnum)
1280 node = summary_find_msg_by_msgnum(summaryview, msgnum);
1281 summary_select_node(summaryview, node, FALSE);
1285 * summary_select_node:
1286 * @summaryview: Summary view.
1287 * @node: Summary tree node.
1288 * @display_msg: TRUE to display the selected message.
1290 * Select @node (bringing it into view by scrolling and expanding its
1291 * thread, if necessary) and unselect all others. If @display_msg is
1292 * TRUE, display the corresponding message in the message view.
1294 static void summary_select_node(SummaryView *summaryview, GtkCTreeNode *node,
1295 gboolean display_msg)
1297 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1301 gtkut_ctree_expand_parent_all(ctree, node);
1302 gtk_ctree_node_moveto(ctree, node, -1, 0.5, 0);
1303 gtk_widget_grab_focus(GTK_WIDGET(ctree));
1304 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1305 summaryview->display_msg = display_msg;
1306 gtk_sctree_select(GTK_SCTREE(ctree), node);
1310 static guint summary_get_msgnum(SummaryView *summaryview, GtkCTreeNode *node)
1312 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1317 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1318 return msginfo->msgnum;
1321 static GtkCTreeNode *summary_find_next_unread_msg(SummaryView *summaryview,
1322 GtkCTreeNode *current_node)
1324 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1329 node = current_node;
1331 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1333 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1334 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1335 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)) break;
1341 static GtkCTreeNode *summary_find_next_marked_msg(SummaryView *summaryview,
1342 GtkCTreeNode *current_node)
1344 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1349 node = GTK_CTREE_NODE_NEXT(current_node);
1351 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1353 for (; node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
1354 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1355 if (MSG_IS_MARKED(msginfo->flags)) break;
1361 static GtkCTreeNode *summary_find_prev_marked_msg(SummaryView *summaryview,
1362 GtkCTreeNode *current_node)
1364 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1369 node = GTK_CTREE_NODE_PREV(current_node);
1371 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list_end);
1373 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1374 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1375 if (MSG_IS_MARKED(msginfo->flags)) break;
1381 static GtkCTreeNode *summary_find_next_labeled_msg(SummaryView *summaryview,
1382 GtkCTreeNode *current_node)
1384 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1389 node = GTK_CTREE_NODE_NEXT(current_node);
1391 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1393 for (; node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
1394 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1395 if (MSG_GET_COLORLABEL_VALUE(msginfo->flags) > 0) break;
1401 static GtkCTreeNode *summary_find_prev_labeled_msg(SummaryView *summaryview,
1402 GtkCTreeNode *current_node)
1404 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1409 node = GTK_CTREE_NODE_PREV(current_node);
1411 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list_end);
1413 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1414 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1415 if (MSG_GET_COLORLABEL_VALUE(msginfo->flags) > 0) break;
1421 static GtkCTreeNode *summary_find_msg_by_msgnum(SummaryView *summaryview,
1424 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1428 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1430 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1431 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1432 if (msginfo->msgnum == msgnum) break;
1439 static GtkCTreeNode *summary_find_prev_unread_msg(SummaryView *summaryview,
1440 GtkCTreeNode *current_node)
1442 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1447 node = current_node;
1448 /*node = GTK_CTREE_NODE_PREV(current_node);*/
1450 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list_end);
1452 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1453 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1454 if (MSG_IS_UNREAD(msginfo->flags)) break;
1461 static guint attract_hash_func(gconstpointer key)
1467 Xstrdup_a(str, (const gchar *)key, return 0);
1474 for (p += 1; *p != '\0'; p++)
1475 h = (h << 5) - h + *p;
1481 static gint attract_compare_func(gconstpointer a, gconstpointer b)
1483 return subject_compare((const gchar *)a, (const gchar *)b) == 0;
1486 void summary_attract_by_subject(SummaryView *summaryview)
1488 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1489 GtkCList *clist = GTK_CLIST(ctree);
1490 GtkCTreeNode *src_node;
1491 GtkCTreeNode *dst_node, *sibling;
1493 MsgInfo *src_msginfo, *dst_msginfo;
1494 GHashTable *subject_table;
1496 debug_print(_("Attracting messages by subject..."));
1497 STATUSBAR_PUSH(summaryview->mainwin,
1498 _("Attracting messages by subject..."));
1500 main_window_cursor_wait(summaryview->mainwin);
1501 gtk_clist_freeze(clist);
1503 subject_table = g_hash_table_new(attract_hash_func,
1504 attract_compare_func);
1506 for (src_node = GTK_CTREE_NODE(clist->row_list);
1509 tmp = GTK_CTREE_ROW(src_node)->sibling;
1510 src_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(src_node);
1511 if (!src_msginfo) continue;
1512 if (!src_msginfo->subject) continue;
1514 /* find attracting node */
1515 dst_node = g_hash_table_lookup(subject_table,
1516 src_msginfo->subject);
1519 dst_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(dst_node);
1521 /* if the time difference is more than 20 days,
1523 if (ABS(src_msginfo->date_t - dst_msginfo->date_t)
1524 > 60 * 60 * 24 * 20)
1527 sibling = GTK_CTREE_ROW(dst_node)->sibling;
1528 if (src_node != sibling)
1529 gtk_ctree_move(ctree, src_node, NULL, sibling);
1532 g_hash_table_insert(subject_table,
1533 src_msginfo->subject, src_node);
1536 g_hash_table_destroy(subject_table);
1538 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
1540 gtk_clist_thaw(clist);
1542 debug_print(_("done.\n"));
1543 STATUSBAR_POP(summaryview->mainwin);
1545 main_window_cursor_normal(summaryview->mainwin);
1548 static void summary_free_msginfo_func(GtkCTree *ctree, GtkCTreeNode *node,
1551 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, node);
1554 procmsg_msginfo_free(msginfo);
1557 static void summary_set_marks_func(GtkCTree *ctree, GtkCTreeNode *node,
1560 SummaryView *summaryview = data;
1563 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1565 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1566 summaryview->newmsgs++;
1567 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1568 summaryview->unread++;
1569 if (MSG_IS_DELETED(msginfo->flags))
1570 summaryview->deleted++;
1572 summaryview->messages++;
1573 summaryview->total_size += msginfo->size;
1575 summary_set_row_marks(summaryview, node);
1578 static void summary_update_status(SummaryView *summaryview)
1580 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1584 summaryview->newmsgs = summaryview->unread =
1585 summaryview->messages = summaryview->total_size =
1586 summaryview->deleted = summaryview->moved = summaryview->copied = 0;
1588 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1589 node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1590 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
1592 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1593 summaryview->newmsgs++;
1594 if (MSG_IS_UNREAD(msginfo->flags)&& !MSG_IS_IGNORE_THREAD(msginfo->flags))
1595 summaryview->unread++;
1596 if (MSG_IS_DELETED(msginfo->flags))
1597 summaryview->deleted++;
1598 if (MSG_IS_MOVE(msginfo->flags))
1599 summaryview->moved++;
1600 if (MSG_IS_COPY(msginfo->flags))
1601 summaryview->copied++;
1602 summaryview->messages++;
1603 summaryview->total_size += msginfo->size;
1607 static void summary_status_show(SummaryView *summaryview)
1610 gchar *del, *mv, *cp;
1613 GList *rowlist, *cur;
1614 guint n_selected = 0;
1618 if (!summaryview->folder_item) {
1619 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder), "");
1620 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), "");
1621 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), "");
1625 rowlist = GTK_CLIST(summaryview->ctree)->selection;
1626 for (cur = rowlist; cur != NULL; cur = cur->next) {
1627 msginfo = gtk_ctree_node_get_row_data
1628 (GTK_CTREE(summaryview->ctree),
1629 GTK_CTREE_NODE(cur->data));
1630 sel_size += msginfo->size;
1634 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder),
1635 summaryview->folder_item &&
1636 summaryview->folder_item->folder->type == F_NEWS
1637 ? g_basename(summaryview->folder_item->path)
1638 : summaryview->folder_item->path);
1640 if (summaryview->deleted)
1641 del = g_strdup_printf(_("%d deleted"), summaryview->deleted);
1644 if (summaryview->moved)
1645 mv = g_strdup_printf(_("%s%d moved"),
1646 summaryview->deleted ? _(", ") : "",
1647 summaryview->moved);
1650 if (summaryview->copied)
1651 cp = g_strdup_printf(_("%s%d copied"),
1652 summaryview->deleted ||
1653 summaryview->moved ? _(", ") : "",
1654 summaryview->copied);
1658 if (summaryview->deleted || summaryview->moved || summaryview->copied)
1664 sel = g_strdup_printf(" (%s)", to_human_readable(sel_size));
1667 str = g_strconcat(n_selected ? itos(n_selected) : "",
1668 n_selected ? _(" item(s) selected") : "",
1669 sel, spc, del, mv, cp, NULL);
1670 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), str);
1677 if (summaryview->folder_item &&
1678 FOLDER_IS_LOCAL(summaryview->folder_item->folder)) {
1679 str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
1680 summaryview->newmsgs,
1681 summaryview->unread,
1682 summaryview->messages,
1683 to_human_readable(summaryview->total_size));
1685 str = g_strdup_printf(_("%d new, %d unread, %d total"),
1686 summaryview->newmsgs,
1687 summaryview->unread,
1688 summaryview->messages);
1690 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), str);
1693 folderview_update_msg_num(summaryview->folderview,
1694 summaryview->folderview->opened,
1695 summaryview->newmsgs,
1696 summaryview->unread,
1697 summaryview->messages);
1700 void summary_sort(SummaryView *summaryview, SummarySortType type)
1702 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1703 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1704 GtkCListCompareFunc cmp_func;
1706 if (!summaryview->folder_item)
1710 case SORT_BY_NUMBER:
1711 cmp_func = (GtkCListCompareFunc)summary_cmp_by_num;
1714 cmp_func = (GtkCListCompareFunc)summary_cmp_by_size;
1717 cmp_func = (GtkCListCompareFunc)summary_cmp_by_date;
1720 cmp_func = (GtkCListCompareFunc)summary_cmp_by_from;
1722 case SORT_BY_SUBJECT:
1723 cmp_func = (GtkCListCompareFunc)summary_cmp_by_subject;
1726 cmp_func = (GtkCListCompareFunc)summary_cmp_by_score;
1729 cmp_func = (GtkCListCompareFunc)summary_cmp_by_label;
1735 debug_print(_("Sorting summary..."));
1736 STATUSBAR_PUSH(summaryview->mainwin, _("Sorting summary..."));
1738 main_window_cursor_wait(summaryview->mainwin);
1740 gtk_clist_set_compare_func(clist, cmp_func);
1742 /* toggle sort type if the same column is selected */
1743 if (summaryview->sort_mode == type)
1744 summaryview->sort_type =
1745 summaryview->sort_type == GTK_SORT_ASCENDING
1746 ? GTK_SORT_DESCENDING : GTK_SORT_ASCENDING;
1748 summaryview->sort_type = GTK_SORT_ASCENDING;
1749 gtk_clist_set_sort_type(clist, summaryview->sort_type);
1750 summaryview->sort_mode = type;
1752 gtk_ctree_sort_node(ctree, NULL);
1754 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
1755 /*gtkut_ctree_set_focus_row(ctree, summaryview->selected);*/
1757 prefs_folder_item_set_config(summaryview->folder_item,
1758 summaryview->sort_type,
1759 summaryview->sort_mode);
1760 prefs_folder_item_save_config(summaryview->folder_item);
1762 debug_print(_("done.\n"));
1763 STATUSBAR_POP(summaryview->mainwin);
1765 main_window_cursor_normal(summaryview->mainwin);
1768 static GtkCTreeNode * subject_table_lookup(GHashTable *subject_table,
1771 if (g_strncasecmp(subject, "Re: ", 4) == 0)
1772 return g_hash_table_lookup(subject_table, subject + 4);
1774 return g_hash_table_lookup(subject_table, subject);
1777 static void subject_table_insert(GHashTable *subject_table, gchar * subject,
1778 GtkCTreeNode * node)
1780 if (g_strncasecmp(subject, "Re: ", 4) == 0)
1781 g_hash_table_insert(subject_table, subject + 4, node);
1783 g_hash_table_insert(subject_table, subject, node);
1786 static void summary_set_ctree_from_list(SummaryView *summaryview,
1789 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1791 MsgInfo *parentinfo;
1792 MsgInfo *cur_msginfo;
1793 GtkCTreeNode *node, *parent;
1794 gchar *text[N_SUMMARY_COLS];
1795 GHashTable *msgid_table;
1796 GHashTable *subject_table;
1798 GtkCTreeNode *cur_parent;
1802 debug_print(_("\tSetting summary from message data..."));
1803 STATUSBAR_PUSH(summaryview->mainwin,
1804 _("Setting summary from message data..."));
1807 msgid_table = g_hash_table_new(g_str_hash, g_str_equal);
1808 summaryview->msgid_table = msgid_table;
1809 subject_table = g_hash_table_new(g_str_hash, g_str_equal);
1810 summaryview->subject_table = subject_table;
1812 if (prefs_common.use_addr_book)
1813 start_address_completion();
1815 /*main_window_set_thread_option(summaryview->mainwin)*/;
1817 for (cur = mlist ; cur != NULL; cur = cur->next) {
1818 msginfo = (MsgInfo *)cur->data;
1819 msginfo->threadscore = msginfo->score;
1822 if (global_scoring || summaryview->folder_item->prefs->scoring) {
1823 summaryview->important_score = prefs_common.important_score;
1824 if (summaryview->folder_item->prefs->important_score >
1825 summaryview->important_score)
1826 summaryview->important_score =
1827 summaryview->folder_item->prefs->important_score;
1830 if (prefs_common.enable_thread) {
1831 /*if (summaryview->folder_item->prefs->enable_thread) { */
1832 for (; mlist != NULL; mlist = mlist->next) {
1833 msginfo = (MsgInfo *)mlist->data;
1836 summary_set_header(text, msginfo);
1838 /* search parent node for threading */
1839 if (msginfo->inreplyto && *msginfo->inreplyto) {
1840 parent = g_hash_table_lookup
1841 (msgid_table, msginfo->inreplyto);
1843 if (parent == NULL && msginfo->subject &&
1844 g_strncasecmp(msginfo->subject, "Re: ", 4) == 0) {
1845 parent = subject_table_lookup
1846 (subject_table, msginfo->subject);
1849 parentinfo = gtk_ctree_node_get_row_data(ctree, parent);
1850 if(!MSG_IS_IGNORE_THREAD(msginfo->flags) && parentinfo && MSG_IS_IGNORE_THREAD(parentinfo->flags)) {
1851 MSG_SET_PERM_FLAGS(msginfo->flags, MSG_IGNORE_THREAD);
1855 node = gtk_ctree_insert_node
1856 (ctree, parent, NULL, text, 2,
1857 NULL, NULL, NULL, NULL, FALSE,
1858 parent ? TRUE : FALSE);
1859 GTKUT_CTREE_NODE_SET_ROW_DATA(node, msginfo);
1860 summary_set_marks_func(ctree, node, summaryview);
1862 if (MSG_GET_COLORLABEL(msginfo->flags))
1863 summary_set_colorlabel_color(ctree, node, MSG_GET_COLORLABEL_VALUE(msginfo->flags));
1865 /* preserve previous node if the message is
1867 if (msginfo->msgid && *msginfo->msgid &&
1868 g_hash_table_lookup(msgid_table, msginfo->msgid)
1870 g_hash_table_insert(msgid_table,
1871 msginfo->msgid, node);
1872 if (msginfo->subject &&
1873 subject_table_lookup(subject_table,
1874 msginfo->subject) == NULL) {
1875 subject_table_insert(subject_table,
1876 msginfo->subject, node);
1879 cur_parent = parent;
1880 cur_msginfo = msginfo;
1881 while (cur_parent != NULL) {
1882 parentinfo = gtk_ctree_node_get_row_data(ctree, cur_parent);
1887 if (parentinfo->threadscore <
1888 cur_msginfo->threadscore) {
1890 parentinfo->threadscore =
1891 cur_msginfo->threadscore;
1893 s = itos(parentinfo->threadscore);
1894 gtk_ctree_node_set_text(ctree, cur_parent, S_COL_SCORE, s);
1899 cur_msginfo = parentinfo;
1900 if (cur_msginfo->inreplyto &&
1901 *cur_msginfo->inreplyto) {
1902 cur_parent = g_hash_table_lookup(msgid_table, cur_msginfo->inreplyto);
1904 if (cur_parent == NULL &&
1905 cur_msginfo->subject &&
1906 g_strncasecmp(cur_msginfo->subject,
1908 cur_parent = subject_table_lookup(subject_table, cur_msginfo->subject);
1913 /* complete the thread */
1914 summary_thread_build(summaryview, TRUE);
1916 for (; mlist != NULL; mlist = mlist->next) {
1917 msginfo = (MsgInfo *)mlist->data;
1919 summary_set_header(text, msginfo);
1921 node = gtk_ctree_insert_node
1922 (ctree, NULL, NULL, text, 2,
1923 NULL, NULL, NULL, NULL, FALSE, FALSE);
1924 GTKUT_CTREE_NODE_SET_ROW_DATA(node, msginfo);
1925 summary_set_marks_func(ctree, node, summaryview);
1927 if ( MSG_GET_COLORLABEL(msginfo->flags) )
1928 summary_set_colorlabel_color(ctree, node, MSG_GET_COLORLABEL_VALUE(msginfo->flags));
1930 if (msginfo->msgid && *msginfo->msgid &&
1931 g_hash_table_lookup(msgid_table, msginfo->msgid)
1933 g_hash_table_insert(msgid_table,
1934 msginfo->msgid, node);
1936 if (msginfo->subject &&
1937 subject_table_lookup(subject_table,
1938 msginfo->subject) == NULL)
1939 subject_table_insert(subject_table,
1940 msginfo->subject, node);
1944 if (prefs_common.enable_hscrollbar) {
1947 optimal_width = gtk_clist_optimal_column_width
1948 (GTK_CLIST(ctree), S_COL_SUBJECT);
1949 gtk_clist_set_column_width(GTK_CLIST(ctree), S_COL_SUBJECT,
1953 if (prefs_common.use_addr_book)
1954 end_address_completion();
1956 debug_print(_("done.\n"));
1957 STATUSBAR_POP(summaryview->mainwin);
1959 debug_print("\tmsgid hash table size = %d\n",
1960 g_hash_table_size(msgid_table));
1961 debug_print("\tsubject hash table size = %d\n",
1962 g_hash_table_size(subject_table));
1972 gint summary_write_cache(SummaryView *summaryview)
1974 struct wcachefp fps;
1975 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1976 gint ver = CACHE_VERSION;
1978 gchar *cachefile, *markfile;
1981 PrefsFolderItem *prefs;
1983 if (!summaryview->folder_item || !summaryview->folder_item->path)
1986 if (summaryview->folder_item->folder->update_mark != NULL)
1987 summaryview->folder_item->folder->update_mark(summaryview->folder_item->folder, summaryview->folder_item);
1989 cachefile = folder_item_get_cache_file(summaryview->folder_item);
1990 g_return_val_if_fail(cachefile != NULL, -1);
1991 if ((fps.cache_fp = fopen(cachefile, "w")) == NULL) {
1992 FILE_OP_ERROR(cachefile, "fopen");
1996 if (change_file_mode_rw(fps.cache_fp, cachefile) < 0)
1997 FILE_OP_ERROR(cachefile, "chmod");
1999 prefs = summaryview->folder_item->prefs;
2000 if (prefs && prefs->enable_folder_chmod && prefs->folder_chmod) {
2001 /* for cache file */
2002 filemode = prefs->folder_chmod;
2003 if (filemode & S_IRGRP) filemode |= S_IWGRP;
2004 if (filemode & S_IROTH) filemode |= S_IWOTH;
2006 fchmod(fileno(fps.cache_fp), filemode);
2008 chmod(cachefile, filemode);
2014 markfile = folder_item_get_mark_file(summaryview->folder_item);
2015 if ((fps.mark_fp = fopen(markfile, "w")) == NULL) {
2016 FILE_OP_ERROR(markfile, "fopen");
2017 fclose(fps.cache_fp);
2021 if (change_file_mode_rw(fps.mark_fp, markfile) < 0)
2022 FILE_OP_ERROR(markfile, "chmod");
2023 if (prefs && prefs->enable_folder_chmod && prefs->folder_chmod) {
2025 fchmod(fileno(fps.mark_fp), filemode);
2027 chmod(markfile, filemode);
2033 buf = g_strdup_printf(_("Writing summary cache (%s)..."),
2034 summaryview->folder_item->path);
2036 STATUSBAR_PUSH(summaryview->mainwin, buf);
2039 WRITE_CACHE_DATA_INT(ver, fps.cache_fp);
2041 WRITE_CACHE_DATA_INT(ver, fps.mark_fp);
2043 gtk_ctree_pre_recursive(ctree, NULL, summary_write_cache_func, &fps);
2045 for(cur = summaryview->killed_messages ; cur != NULL ;
2046 cur = g_slist_next(cur)) {
2047 MsgInfo *msginfo = (MsgInfo *) cur->data;
2049 procmsg_write_cache(msginfo, fps.cache_fp);
2050 procmsg_write_flags(msginfo, fps.mark_fp);
2053 fclose(fps.cache_fp);
2054 fclose(fps.mark_fp);
2056 debug_print(_("done.\n"));
2057 STATUSBAR_POP(summaryview->mainwin);
2062 static void summary_write_cache_func(GtkCTree *ctree, GtkCTreeNode *node,
2065 struct wcachefp *fps = data;
2066 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, node);
2068 if (msginfo == NULL) return;
2070 procmsg_write_cache(msginfo, fps->cache_fp);
2071 procmsg_write_flags(msginfo, fps->mark_fp);
2074 static void summary_set_header(gchar *text[], MsgInfo *msginfo)
2076 static gchar date_modified[80];
2077 static gchar *to = NULL;
2078 static gchar *from_name = NULL;
2079 static gchar col_number[11];
2080 static gchar col_score[11];
2082 text[S_COL_MARK] = NULL;
2083 text[S_COL_UNREAD] = NULL;
2084 text[S_COL_MIME] = NULL;
2085 text[S_COL_NUMBER] = itos_buf(col_number, msginfo->msgnum);
2086 text[S_COL_SIZE] = to_human_readable(msginfo->size);
2088 text[S_COL_SCORE] = itos_buf(col_score, msginfo->threadscore);
2090 text[S_COL_SCORE] = itos_buf(col_score, msginfo->score);
2093 if (msginfo->date_t) {
2094 procheader_date_get_localtime(date_modified,
2095 sizeof(date_modified),
2097 text[S_COL_DATE] = date_modified;
2098 } else if (msginfo->date)
2099 text[S_COL_DATE] = msginfo->date;
2101 text[S_COL_DATE] = _("(No Date)");
2103 text[S_COL_FROM] = msginfo->fromname ? msginfo->fromname :
2105 if (prefs_common.swap_from && msginfo->from && msginfo->to &&
2106 !MSG_IS_NEWS(msginfo->flags)) {
2109 Xalloca(from, strlen(msginfo->from) + 1, return);
2110 strcpy(from, msginfo->from);
2111 extract_address(from);
2112 if (account_find_from_address(from)) {
2114 to = g_strconcat("-->", msginfo->to, NULL);
2115 text[S_COL_FROM] = to;
2119 if ((text[S_COL_FROM] != to) && prefs_common.use_addr_book &&
2124 Xalloca(from, strlen(msginfo->from) + 1, return);
2125 strcpy(from, msginfo->from);
2126 extract_address(from);
2128 count = complete_address(from);
2131 from = get_complete_address(1);
2132 from_name = procheader_get_fromname(from);
2134 text[S_COL_FROM] = from_name;
2139 text[S_COL_SUBJECT] = msginfo->subject ? msginfo->subject :
2143 #define CHANGE_FLAGS(msginfo) \
2145 if (msginfo->folder->folder->change_flags != NULL) \
2146 msginfo->folder->folder->change_flags(msginfo->folder->folder, \
2151 static void summary_display_msg(SummaryView *summaryview, GtkCTreeNode *row,
2152 gboolean new_window)
2154 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2157 static gboolean lock = FALSE;
2159 if (!new_window && summaryview->displayed == row) return;
2160 g_return_if_fail(row != NULL);
2165 STATUSBAR_POP(summaryview->mainwin);
2168 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2170 filename = procmsg_get_message_file(msginfo);
2177 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2178 summaryview->newmsgs--;
2179 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2180 summaryview->unread--;
2181 if (MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)) {
2182 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_NEW | MSG_UNREAD);
2183 CHANGE_FLAGS(msginfo);
2184 summary_set_row_marks(summaryview, row);
2185 gtk_clist_thaw(GTK_CLIST(ctree));
2186 summary_status_show(summaryview);
2190 MessageView *msgview;
2192 msgview = messageview_create_with_new_window();
2193 messageview_show(msgview, msginfo);
2195 MessageView *msgview;
2197 msgview = summaryview->messageview;
2199 summaryview->displayed = row;
2200 if (!summaryview->msg_is_toggled_on)
2201 summary_toggle_view(summaryview);
2202 messageview_show(msgview, msginfo);
2203 if (msgview->type == MVIEW_TEXT ||
2204 (msgview->type == MVIEW_MIME &&
2205 GTK_CLIST(msgview->mimeview->ctree)->row_list == NULL))
2206 gtk_widget_grab_focus(summaryview->ctree);
2208 gtkut_ctree_node_move_if_on_the_edge(ctree, row);
2211 if (GTK_WIDGET_VISIBLE(summaryview->headerwin->window))
2212 header_window_show(summaryview->headerwin, msginfo);
2217 void summary_redisplay_msg(SummaryView *summaryview)
2221 if (summaryview->displayed) {
2222 node = summaryview->displayed;
2223 summaryview->displayed = NULL;
2224 summary_display_msg(summaryview, node, FALSE);
2228 void summary_open_msg(SummaryView *summaryview)
2230 if (!summaryview->selected) return;
2232 summary_display_msg(summaryview, summaryview->selected, TRUE);
2235 void summary_view_source(SummaryView * summaryview)
2237 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2239 SourceWindow *srcwin;
2241 if (!summaryview->selected) return;
2243 srcwin = source_window_create();
2244 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
2245 source_window_show_msg(srcwin, msginfo);
2246 source_window_show(srcwin);
2249 void summary_reedit(SummaryView *summaryview)
2253 if (!summaryview->selected) return;
2254 if (!summaryview->folder_item ||
2255 (summaryview->folder_item->stype != F_DRAFT &&
2256 summaryview->folder_item->stype != F_OUTBOX &&
2257 summaryview->folder_item->stype != F_QUEUE)) return;
2259 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
2260 summaryview->selected);
2261 if (!msginfo) return;
2263 compose_reedit(msginfo);
2266 void summary_step(SummaryView *summaryview, GtkScrollType type)
2268 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2270 if (type == GTK_SCROLL_STEP_FORWARD) {
2272 node = gtkut_ctree_node_next(ctree, summaryview->selected);
2274 gtkut_ctree_expand_parent_all(ctree, node);
2277 gtk_signal_emit_by_name(GTK_OBJECT(ctree), "scroll_vertical",
2280 if (summaryview->msg_is_toggled_on)
2281 summary_display_msg(summaryview, summaryview->selected, FALSE);
2284 static void summary_toggle_view(SummaryView *summaryview)
2286 MainWindow *mainwin = summaryview->mainwin;
2287 union CompositeWin *cwin = &mainwin->win;
2288 GtkWidget *vpaned = NULL;
2289 GtkWidget *container = NULL;
2291 switch (mainwin->type) {
2293 vpaned = cwin->sep_none.vpaned;
2294 container = cwin->sep_none.hpaned;
2296 case SEPARATE_FOLDER:
2297 vpaned = cwin->sep_folder.vpaned;
2298 container = mainwin->vbox_body;
2300 case SEPARATE_MESSAGE:
2305 if (vpaned->parent != NULL) {
2306 summaryview->msg_is_toggled_on = FALSE;
2307 summaryview->displayed = NULL;
2308 gtk_widget_ref(vpaned);
2309 gtk_container_remove(GTK_CONTAINER(container), vpaned);
2310 gtk_widget_reparent(GTK_WIDGET_PTR(summaryview), container);
2311 gtk_arrow_set(GTK_ARROW(summaryview->toggle_arrow),
2312 GTK_ARROW_UP, GTK_SHADOW_OUT);
2314 summaryview->msg_is_toggled_on = TRUE;
2315 gtk_widget_reparent(GTK_WIDGET_PTR(summaryview), vpaned);
2316 gtk_container_add(GTK_CONTAINER(container), vpaned);
2317 gtk_widget_unref(vpaned);
2318 gtk_arrow_set(GTK_ARROW(summaryview->toggle_arrow),
2319 GTK_ARROW_DOWN, GTK_SHADOW_OUT);
2322 gtk_widget_grab_focus(summaryview->ctree);
2325 static gboolean summary_search_unread_recursive(GtkCTree *ctree,
2331 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2332 if (msginfo && MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2334 node = GTK_CTREE_ROW(node)->children;
2336 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2339 if (summary_search_unread_recursive(ctree, node) == TRUE)
2341 node = GTK_CTREE_ROW(node)->sibling;
2347 static gboolean summary_have_unread_children(SummaryView *summaryview,
2350 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2352 if (!node) return FALSE;
2354 node = GTK_CTREE_ROW(node)->children;
2357 if (summary_search_unread_recursive(ctree, node) == TRUE)
2359 node = GTK_CTREE_ROW(node)->sibling;
2365 static void summary_set_row_marks(SummaryView *summaryview, GtkCTreeNode *row)
2367 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2368 GtkStyle *style = NULL;
2372 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2373 if (!msginfo) return;
2375 flags = msginfo->flags;
2377 gtk_ctree_node_set_foreground(ctree, row, NULL);
2379 /* set new/unread column */
2380 if (MSG_IS_IGNORE_THREAD(flags)) {
2381 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2382 ignorethreadxpm, ignorethreadxpmmask);
2383 } else if (MSG_IS_NEW(flags)) {
2384 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2385 newxpm, newxpmmask);
2386 } else if (MSG_IS_UNREAD(flags)) {
2387 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2388 unreadxpm, unreadxpmmask);
2389 } else if (MSG_IS_REPLIED(flags)) {
2390 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2391 repliedxpm, repliedxpmmask);
2392 } else if (MSG_IS_FORWARDED(flags)) {
2393 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2394 forwardedxpm, forwardedxpmmask);
2396 gtk_ctree_node_set_text(ctree, row, S_COL_UNREAD, NULL);
2399 if (prefs_common.bold_unread &&
2400 ((MSG_IS_UNREAD(flags) && !MSG_IS_IGNORE_THREAD(flags)) ||
2401 (!GTK_CTREE_ROW(row)->expanded &&
2402 GTK_CTREE_ROW(row)->children &&
2403 summary_have_unread_children(summaryview, row))))
2406 /* set mark column */
2407 if (MSG_IS_DELETED(flags)) {
2408 gtk_ctree_node_set_pixmap(ctree, row, S_COL_MARK,
2409 deletedxpm, deletedxpmmask);
2411 style = bold_deleted_style;
2413 style = small_deleted_style;
2415 gtk_ctree_node_set_foreground
2416 (ctree, row, &summaryview->color_dim);
2417 } else if (MSG_IS_MARKED(flags)) {
2418 gtk_ctree_node_set_pixmap(ctree, row, S_COL_MARK,
2419 markxpm, markxpmmask);
2420 } else if (MSG_IS_MOVE(flags)) {
2421 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, "o");
2423 style = bold_marked_style;
2425 style = small_marked_style;
2427 gtk_ctree_node_set_foreground
2428 (ctree, row, &summaryview->color_marked);
2429 } else if (MSG_IS_COPY(flags)) {
2430 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, "O");
2432 style = bold_marked_style;
2434 style = small_marked_style;
2436 gtk_ctree_node_set_foreground
2437 (ctree, row, &summaryview->color_marked);
2439 else if ((global_scoring ||
2440 summaryview->folder_item->prefs->scoring) &&
2441 (msginfo->score >= summaryview->important_score) &&
2442 (MSG_IS_MARKED(msginfo->flags) || MSG_IS_MOVE(msginfo->flags) || MSG_IS_COPY(msginfo->flags))) {
2443 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, "!");
2444 gtk_ctree_node_set_foreground(ctree, row,
2445 &summaryview->color_important);
2447 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, NULL);
2450 if (MSG_IS_MIME(flags)) {
2451 gtk_ctree_node_set_pixmap(ctree, row, S_COL_MIME,
2452 clipxpm, clipxpmmask);
2454 gtk_ctree_node_set_text(ctree, row, S_COL_MIME, NULL);
2457 style = small_style;
2459 gtk_ctree_node_set_row_style(ctree, row, style);
2461 if (MSG_GET_COLORLABEL(flags))
2462 summary_set_colorlabel_color(ctree, row,
2463 MSG_GET_COLORLABEL_VALUE(flags));
2466 void summary_set_marks_selected(SummaryView *summaryview)
2468 summary_set_row_marks(summaryview, summaryview->selected);
2471 static void summary_mark_row(SummaryView *summaryview, GtkCTreeNode *row)
2473 gboolean changed = FALSE;
2474 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2477 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2478 msginfo->to_folder = NULL;
2479 if (MSG_IS_DELETED(msginfo->flags))
2480 summaryview->deleted--;
2481 if (MSG_IS_MOVE(msginfo->flags)) {
2482 summaryview->moved--;
2485 if (MSG_IS_COPY(msginfo->flags)) {
2486 summaryview->copied--;
2489 if (changed && !prefs_common.immediate_exec) {
2490 msginfo->to_folder->op_count--;
2491 if (msginfo->to_folder->op_count == 0)
2492 folderview_update_item(msginfo->to_folder, 0);
2494 msginfo->to_folder = NULL;
2495 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_DELETED);
2496 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_MOVE | MSG_COPY);
2497 MSG_SET_PERM_FLAGS(msginfo->flags, MSG_MARKED);
2498 CHANGE_FLAGS(msginfo);
2499 summary_set_row_marks(summaryview, row);
2500 debug_print(_("Message %d is marked\n"), msginfo->msgnum);
2503 void summary_mark(SummaryView *summaryview)
2505 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2508 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2509 summary_mark_row(summaryview, GTK_CTREE_NODE(cur->data));
2511 /* summary_step(summaryview, GTK_SCROLL_STEP_FORWARD); */
2512 summary_status_show(summaryview);
2515 static void summary_mark_row_as_read(SummaryView *summaryview,
2518 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2521 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2522 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2523 summaryview->newmsgs--;
2524 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2525 summaryview->unread--;
2526 if (MSG_IS_NEW(msginfo->flags) ||
2527 MSG_IS_UNREAD(msginfo->flags)) {
2528 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_NEW | MSG_UNREAD);
2529 CHANGE_FLAGS(msginfo);
2530 summary_set_row_marks(summaryview, row);
2531 debug_print(_("Message %d is marked as read\n"),
2536 void summary_mark_as_read(SummaryView *summaryview)
2538 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2541 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2542 summary_mark_row_as_read(summaryview,
2543 GTK_CTREE_NODE(cur->data));
2545 summary_status_show(summaryview);
2549 static void summary_mark_all_read(SummaryView *summaryview)
2551 summary_select_all(summaryview);
2552 summary_mark_as_read(summaryview);
2553 summary_unselect_all(summaryview);
2557 static void summary_mark_row_as_unread(SummaryView *summaryview,
2560 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2563 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2564 if (MSG_IS_DELETED(msginfo->flags)) {
2565 msginfo->to_folder = NULL;
2566 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_DELETED);
2567 summaryview->deleted--;
2569 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_REPLIED | MSG_FORWARDED);
2570 if (!MSG_IS_UNREAD(msginfo->flags)) {
2571 MSG_SET_PERM_FLAGS(msginfo->flags, MSG_UNREAD);
2572 gtk_ctree_node_set_pixmap(ctree, row, S_COL_UNREAD,
2573 unreadxpm, unreadxpmmask);
2574 summaryview->unread++;
2575 debug_print(_("Message %d is marked as unread\n"),
2579 CHANGE_FLAGS(msginfo);
2581 summary_set_row_marks(summaryview, row);
2584 void summary_mark_as_unread(SummaryView *summaryview)
2586 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2589 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2590 summary_mark_row_as_unread(summaryview,
2591 GTK_CTREE_NODE(cur->data));
2593 summary_status_show(summaryview);
2596 static void summary_delete_row(SummaryView *summaryview, GtkCTreeNode *row)
2598 gboolean changed = FALSE;
2599 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2602 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2604 if (MSG_IS_DELETED(msginfo->flags)) return;
2606 if (MSG_IS_MOVE(msginfo->flags)) {
2607 summaryview->moved--;
2610 if (MSG_IS_COPY(msginfo->flags)) {
2611 summaryview->copied--;
2614 if (changed && !prefs_common.immediate_exec) {
2615 msginfo->to_folder->op_count--;
2616 if (msginfo->to_folder->op_count == 0)
2617 folderview_update_item(msginfo->to_folder, 0);
2619 msginfo->to_folder = NULL;
2620 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_MARKED);
2621 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_MOVE | MSG_COPY);
2622 MSG_SET_PERM_FLAGS(msginfo->flags, MSG_DELETED);
2623 CHANGE_FLAGS(msginfo);
2624 summaryview->deleted++;
2626 if (!prefs_common.immediate_exec)
2627 summary_set_row_marks(summaryview, row);
2629 debug_print(_("Message %s/%d is set to delete\n"),
2630 msginfo->folder->path, msginfo->msgnum);
2633 void summary_delete(SummaryView *summaryview)
2635 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2638 if (!summaryview->folder_item ||
2639 summaryview->folder_item->folder->type == F_NEWS) return;
2641 /* if current folder is trash, don't delete */
2642 if (summaryview->folder_item->stype == F_TRASH) {
2643 alertpanel_notice(_("Current folder is Trash."));
2647 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2648 summary_delete_row(summaryview, GTK_CTREE_NODE(cur->data));
2650 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
2652 if (prefs_common.immediate_exec)
2653 summary_execute(summaryview);
2655 summary_status_show(summaryview);
2658 void summary_delete_duplicated(SummaryView *summaryview)
2660 if (!summaryview->folder_item ||
2661 summaryview->folder_item->folder->type == F_NEWS) return;
2662 if (summaryview->folder_item->stype == F_TRASH) return;
2664 main_window_cursor_wait(summaryview->mainwin);
2665 debug_print(_("Deleting duplicated messages..."));
2666 STATUSBAR_PUSH(summaryview->mainwin,
2667 _("Deleting duplicated messages..."));
2669 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree), NULL,
2670 GTK_CTREE_FUNC(summary_delete_duplicated_func),
2673 if (prefs_common.immediate_exec)
2674 summary_execute(summaryview);
2676 summary_status_show(summaryview);
2678 debug_print(_("done.\n"));
2679 STATUSBAR_POP(summaryview->mainwin);
2680 main_window_cursor_normal(summaryview->mainwin);
2683 static void summary_delete_duplicated_func(GtkCTree *ctree, GtkCTreeNode *node,
2684 SummaryView *summaryview)
2686 GtkCTreeNode *found;
2687 MsgInfo *msginfo = GTK_CTREE_ROW(node)->row.data;
2689 if (!msginfo->msgid || !*msginfo->msgid) return;
2691 found = g_hash_table_lookup(summaryview->msgid_table, msginfo->msgid);
2693 if (found && found != node)
2694 summary_delete_row(summaryview, node);
2697 static void summary_unmark_row(SummaryView *summaryview, GtkCTreeNode *row)
2699 gboolean changed = FALSE;
2700 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2703 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2704 msginfo->to_folder = NULL;
2705 if (MSG_IS_DELETED(msginfo->flags))
2706 summaryview->deleted--;
2707 if (MSG_IS_MOVE(msginfo->flags)) {
2708 summaryview->moved--;
2711 if (MSG_IS_COPY(msginfo->flags)) {
2712 summaryview->copied--;
2715 if (changed && !prefs_common.immediate_exec) {
2716 msginfo->to_folder->op_count--;
2717 if (msginfo->to_folder->op_count == 0)
2718 folderview_update_item(msginfo->to_folder, 0);
2720 msginfo->to_folder = NULL;
2721 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_MARKED | MSG_DELETED);
2722 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_MOVE | MSG_COPY);
2723 CHANGE_FLAGS(msginfo);
2724 summary_set_row_marks(summaryview, row);
2726 debug_print(_("Message %s/%d is unmarked\n"),
2727 msginfo->folder->path, msginfo->msgnum);
2730 void summary_unmark(SummaryView *summaryview)
2732 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2735 for (cur = GTK_CLIST(ctree)->selection; cur != NULL;
2737 summary_unmark_row(summaryview, GTK_CTREE_NODE(cur->data));
2739 summary_status_show(summaryview);
2742 static void summary_move_row_to(SummaryView *summaryview, GtkCTreeNode *row,
2743 FolderItem *to_folder)
2745 gboolean changed = FALSE;
2746 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2749 g_return_if_fail(to_folder != NULL);
2751 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2752 if (MSG_IS_MOVE(msginfo->flags)) {
2753 if (!prefs_common.immediate_exec) {
2754 msginfo->to_folder->op_count--;
2755 if (msginfo->to_folder->op_count == 0) {
2756 folderview_update_item(msginfo->to_folder, 0);
2761 msginfo->to_folder = to_folder;
2762 if (MSG_IS_DELETED(msginfo->flags))
2763 summaryview->deleted--;
2764 if (MSG_IS_COPY(msginfo->flags)) {
2765 summaryview->copied--;
2766 if (!prefs_common.immediate_exec)
2767 msginfo->to_folder->op_count--;
2769 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_MARKED | MSG_DELETED);
2770 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_COPY);
2771 if (!MSG_IS_MOVE(msginfo->flags)) {
2772 MSG_SET_TMP_FLAGS(msginfo->flags, MSG_MOVE);
2773 summaryview->moved++;
2776 if (!prefs_common.immediate_exec) {
2777 summary_set_row_marks(summaryview, row);
2779 msginfo->to_folder->op_count++;
2780 if (msginfo->to_folder->op_count == 1)
2781 folderview_update_item(msginfo->to_folder, 0);
2785 debug_print(_("Message %d is set to move to %s\n"),
2786 msginfo->msgnum, to_folder->path);
2789 void summary_move_selected_to(SummaryView *summaryview, FolderItem *to_folder)
2793 if (!to_folder) return;
2794 if (!summaryview->folder_item ||
2795 summaryview->folder_item->folder->type == F_NEWS) return;
2796 if (summaryview->folder_item == to_folder) {
2797 alertpanel_notice(_("Destination is same as current folder."));
2801 for (cur = GTK_CLIST(summaryview->ctree)->selection;
2802 cur != NULL; cur = cur->next)
2804 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
2806 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
2808 if (prefs_common.immediate_exec)
2809 summary_execute(summaryview);
2811 summary_status_show(summaryview);
2813 folderview_update_item(to_folder, 0);
2817 void summary_move_to(SummaryView *summaryview)
2819 FolderItem *to_folder;
2821 if (!summaryview->folder_item ||
2822 summaryview->folder_item->folder->type == F_NEWS) return;
2824 to_folder = foldersel_folder_sel(NULL, NULL);
2825 summary_move_selected_to(summaryview, to_folder);
2828 static void summary_copy_row_to(SummaryView *summaryview, GtkCTreeNode *row,
2829 FolderItem *to_folder)
2831 gboolean changed = FALSE;
2832 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2835 g_return_if_fail(to_folder != NULL);
2837 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2838 if (MSG_IS_COPY(msginfo->flags)) {
2839 if (!prefs_common.immediate_exec) {
2840 msginfo->to_folder->op_count--;
2841 if (msginfo->to_folder->op_count == 0) {
2842 folderview_update_item(msginfo->to_folder, 0);
2847 msginfo->to_folder = to_folder;
2848 if (MSG_IS_DELETED(msginfo->flags))
2849 summaryview->deleted--;
2850 if (MSG_IS_MOVE(msginfo->flags)) {
2851 summaryview->moved--;
2852 if (!prefs_common.immediate_exec)
2853 msginfo->to_folder->op_count--;
2855 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_MARKED | MSG_DELETED);
2856 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_MOVE);
2857 if (!MSG_IS_COPY(msginfo->flags)) {
2858 MSG_SET_TMP_FLAGS(msginfo->flags, MSG_COPY);
2859 summaryview->copied++;
2862 if (!prefs_common.immediate_exec) {
2863 summary_set_row_marks(summaryview, row);
2865 msginfo->to_folder->op_count++;
2866 if (msginfo->to_folder->op_count == 1)
2867 folderview_update_item(msginfo->to_folder, 0);
2871 debug_print(_("Message %d is set to copy to %s\n"),
2872 msginfo->msgnum, to_folder->path);
2875 void summary_copy_selected_to(SummaryView *summaryview, FolderItem *to_folder)
2879 if (!to_folder) return;
2880 if (!summaryview->folder_item ||
2881 summaryview->folder_item->folder->type == F_NEWS) return;
2882 if (summaryview->folder_item == to_folder) {
2884 (_("Destination to copy is same as current folder."));
2888 for (cur = GTK_CLIST(summaryview->ctree)->selection;
2889 cur != NULL; cur = cur->next)
2891 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
2893 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
2895 if (prefs_common.immediate_exec)
2896 summary_execute(summaryview);
2898 summary_status_show(summaryview);
2900 folderview_update_item(to_folder, 0);
2904 void summary_copy_to(SummaryView *summaryview)
2906 FolderItem *to_folder;
2908 if (!summaryview->folder_item ||
2909 summaryview->folder_item->folder->type == F_NEWS) return;
2911 to_folder = foldersel_folder_sel(NULL, NULL);
2912 summary_copy_selected_to(summaryview, to_folder);
2915 void summary_select_all(SummaryView *summaryview)
2917 if (summaryview->messages >= 500) {
2918 STATUSBAR_PUSH(summaryview->mainwin,
2919 _("Selecting all messages..."));
2920 main_window_cursor_wait(summaryview->mainwin);
2923 gtk_clist_select_all(GTK_CLIST(summaryview->ctree));
2925 if (summaryview->messages >= 500) {
2926 STATUSBAR_POP(summaryview->mainwin);
2927 main_window_cursor_normal(summaryview->mainwin);
2931 void summary_unselect_all(SummaryView *summaryview)
2933 gtk_sctree_unselect_all(GTK_SCTREE(summaryview->ctree));
2936 void summary_save_as(SummaryView *summaryview)
2938 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2942 if (!summaryview->selected) return;
2943 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
2944 if (!msginfo) return;
2946 dest = filesel_select_file(_("Save as"), NULL);
2948 if (is_file_exist(dest)) {
2951 aval = alertpanel(_("Overwrite"),
2952 _("Overwrite existing file?"),
2953 _("OK"), _("Cancel"), NULL);
2954 if (G_ALERTDEFAULT != aval) return;
2957 src = procmsg_get_message_file(msginfo);
2958 copy_file(src, dest);
2962 void summary_print(SummaryView *summaryview)
2964 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2965 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2971 if (clist->selection == NULL) return;
2973 cmdline = input_dialog(_("Print"),
2974 _("Enter the print command line:\n"
2975 "(`%s' will be replaced with file name)"),
2976 prefs_common.print_cmd);
2977 if (!cmdline) return;
2978 if (!(p = strchr(cmdline, '%')) || *(p + 1) != 's' ||
2979 strchr(p + 2, '%')) {
2980 alertpanel_error(_("Print command line is invalid:\n`%s'"),
2986 for (cur = clist->selection; cur != NULL; cur = cur->next) {
2987 msginfo = gtk_ctree_node_get_row_data
2988 (ctree, GTK_CTREE_NODE(cur->data));
2989 if (msginfo) procmsg_print_message(msginfo, cmdline);
2995 void summary_execute(SummaryView *summaryview)
2997 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2998 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2999 GtkCTreeNode *node, *next;
3001 if (!summaryview->folder_item ||
3002 summaryview->folder_item->folder->type == F_NEWS) return;
3004 gtk_clist_freeze(clist);
3006 /*if (summaryview->folder_item->prefs->enable_thread) */
3007 if (prefs_common.enable_thread)
3008 summary_unthread_for_exec(summaryview);
3010 summary_execute_move(summaryview);
3011 summary_execute_copy(summaryview);
3012 summary_execute_delete(summaryview);
3014 node = GTK_CTREE_NODE(clist->row_list);
3015 while (node != NULL) {
3016 next = gtkut_ctree_node_next(ctree, node);
3017 if (gtk_ctree_node_get_row_data(ctree, node) == NULL) {
3018 if (node == summaryview->displayed) {
3019 messageview_clear(summaryview->messageview);
3020 summaryview->displayed = NULL;
3022 if (GTK_CTREE_ROW(node)->children != NULL)
3023 g_warning("summary_execute(): children != NULL\n");
3025 gtk_ctree_remove_node(ctree, node);
3030 /*if (summaryview->folder_item->prefs->enable_thread) */
3031 if (prefs_common.enable_thread)
3032 summary_thread_build(summaryview, FALSE);
3034 summaryview->selected = clist->selection ?
3035 GTK_CTREE_NODE(clist->selection->data) : NULL;
3037 if (!GTK_CLIST(summaryview->ctree)->row_list) {
3038 menu_set_insensitive_all
3039 (GTK_MENU_SHELL(summaryview->popupmenu));
3040 gtk_widget_grab_focus(summaryview->folderview->ctree);
3042 gtk_widget_grab_focus(summaryview->ctree);
3044 summary_update_status(summaryview);
3045 summary_status_show(summaryview);
3047 summary_write_cache(summaryview);
3049 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
3051 gtk_clist_thaw(clist);
3054 static void summary_execute_move(SummaryView *summaryview)
3056 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3059 summaryview->folder_table = g_hash_table_new(NULL, NULL);
3061 /* search moving messages and execute */
3062 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_move_func,
3065 if (summaryview->mlist) {
3066 procmsg_move_messages(summaryview->mlist);
3068 folder_item_scan_foreach(summaryview->folder_table);
3069 folderview_update_item_foreach(summaryview->folder_table);
3071 for (cur = summaryview->mlist; cur != NULL; cur = cur->next)
3072 procmsg_msginfo_free((MsgInfo *)cur->data);
3073 g_slist_free(summaryview->mlist);
3074 summaryview->mlist = NULL;
3077 g_hash_table_destroy(summaryview->folder_table);
3078 summaryview->folder_table = NULL;
3081 static void summary_execute_move_func(GtkCTree *ctree, GtkCTreeNode *node,
3084 SummaryView *summaryview = data;
3087 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3089 if (msginfo && MSG_IS_MOVE(msginfo->flags) && msginfo->to_folder) {
3090 g_hash_table_insert(summaryview->folder_table,
3091 msginfo->to_folder, GINT_TO_POINTER(1));
3093 summaryview->mlist =
3094 g_slist_append(summaryview->mlist, msginfo);
3095 gtk_ctree_node_set_row_data(ctree, node, NULL);
3097 if (msginfo->msgid && *msginfo->msgid &&
3098 node == g_hash_table_lookup(summaryview->msgid_table,
3100 g_hash_table_remove(summaryview->msgid_table,
3105 static void summary_execute_copy(SummaryView *summaryview)
3107 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3109 summaryview->folder_table = g_hash_table_new(NULL, NULL);
3111 /* search copying messages and execute */
3112 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_copy_func,
3115 if (summaryview->mlist) {
3116 procmsg_copy_messages(summaryview->mlist);
3118 folder_item_scan_foreach(summaryview->folder_table);
3119 folderview_update_item_foreach(summaryview->folder_table);
3121 g_slist_free(summaryview->mlist);
3122 summaryview->mlist = NULL;
3125 g_hash_table_destroy(summaryview->folder_table);
3126 summaryview->folder_table = NULL;
3129 static void summary_execute_copy_func(GtkCTree *ctree, GtkCTreeNode *node,
3132 SummaryView *summaryview = data;
3135 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3137 if (msginfo && MSG_IS_COPY(msginfo->flags) && msginfo->to_folder) {
3138 g_hash_table_insert(summaryview->folder_table,
3139 msginfo->to_folder, GINT_TO_POINTER(1));
3141 summaryview->mlist =
3142 g_slist_append(summaryview->mlist, msginfo);
3144 MSG_UNSET_TMP_FLAGS(msginfo->flags, MSG_COPY);
3145 summary_set_row_marks(summaryview, node);
3149 static void summary_execute_delete(SummaryView *summaryview)
3151 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3155 trash = summaryview->folder_item->folder->trash;
3156 if (summaryview->folder_item->folder->type == F_MH) {
3157 g_return_if_fail(trash != NULL);
3159 if (summaryview->folder_item == trash) return;
3161 /* search deleting messages and execute */
3162 gtk_ctree_pre_recursive
3163 (ctree, NULL, summary_execute_delete_func, summaryview);
3165 if (!summaryview->mlist) return;
3167 for(cur = summaryview->mlist ; cur != NULL ; cur = cur->next) {
3168 MsgInfo * msginfo = cur->data;
3169 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_DELETED);
3172 folder_item_move_msgs_with_dest(trash, summaryview->mlist);
3174 for (cur = summaryview->mlist; cur != NULL; cur = cur->next)
3175 procmsg_msginfo_free((MsgInfo *)cur->data);
3177 g_slist_free(summaryview->mlist);
3178 summaryview->mlist = NULL;
3180 folder_item_scan(trash);
3181 folderview_update_item(trash, FALSE);
3184 static void summary_execute_delete_func(GtkCTree *ctree, GtkCTreeNode *node,
3187 SummaryView *summaryview = data;
3190 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3192 if (msginfo && MSG_IS_DELETED(msginfo->flags)) {
3193 summaryview->mlist =
3194 g_slist_append(summaryview->mlist, msginfo);
3195 gtk_ctree_node_set_row_data(ctree, node, NULL);
3197 if (msginfo->msgid && *msginfo->msgid &&
3198 node == g_hash_table_lookup(summaryview->msgid_table,
3200 g_hash_table_remove(summaryview->msgid_table,
3203 if (msginfo->subject &&
3204 node == subject_table_lookup(summaryview->subject_table,
3205 msginfo->subject)) {
3206 gchar *s = msginfo->subject + (g_strncasecmp(msginfo->subject, "Re: ", 4) == 0 ? 4 : 0);
3207 g_hash_table_remove(summaryview->subject_table, s);
3212 /* thread functions */
3214 void summary_thread_build(SummaryView *summaryview, gboolean init)
3216 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3219 GtkCTreeNode *parent;
3222 debug_print(_("Building threads..."));
3223 STATUSBAR_PUSH(summaryview->mainwin, _("Building threads..."));
3224 main_window_cursor_wait(summaryview->mainwin);
3226 gtk_signal_handler_block_by_func(GTK_OBJECT(ctree),
3227 summary_tree_expanded, summaryview);
3228 gtk_clist_freeze(GTK_CLIST(ctree));
3230 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3232 next = GTK_CTREE_ROW(node)->sibling;
3234 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3235 if (msginfo && msginfo->inreplyto) {
3236 parent = g_hash_table_lookup(summaryview->msgid_table,
3237 msginfo->inreplyto);
3238 if (parent && parent != node) {
3239 gtk_ctree_move(ctree, node, parent, NULL);
3240 gtk_ctree_expand(ctree, node);
3247 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3249 /* for optimization */
3251 if (prefs_common.expand_thread) {
3253 next = GTK_CTREE_ROW(node)->sibling;
3254 if (GTK_CTREE_ROW(node)->children)
3255 gtk_ctree_expand(ctree, node);
3258 } else if (prefs_common.bold_unread) {
3260 next = GTK_CTREE_ROW(node)->sibling;
3261 if (GTK_CTREE_ROW(node)->children)
3262 summary_set_row_marks
3263 (summaryview, node);
3269 next = GTK_CTREE_NODE_NEXT(node);
3270 if (prefs_common.expand_thread)
3271 gtk_ctree_expand(ctree, node);
3272 if (prefs_common.bold_unread &&
3273 GTK_CTREE_ROW(node)->children)
3274 summary_set_row_marks(summaryview, node);
3279 gtk_clist_thaw(GTK_CLIST(ctree));
3280 gtk_signal_handler_unblock_by_func(GTK_OBJECT(ctree),
3281 summary_tree_expanded, summaryview);
3283 debug_print(_("done.\n"));
3284 STATUSBAR_POP(summaryview->mainwin);
3285 main_window_cursor_normal(summaryview->mainwin);
3288 void summary_unthread(SummaryView *summaryview)
3290 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3292 GtkCTreeNode *child;
3293 GtkCTreeNode *sibling;
3294 GtkCTreeNode *next_child;
3296 debug_print(_("Unthreading..."));
3297 STATUSBAR_PUSH(summaryview->mainwin, _("Unthreading..."));
3298 main_window_cursor_wait(summaryview->mainwin);
3300 gtk_signal_handler_block_by_func(GTK_OBJECT(ctree),
3301 summary_tree_collapsed, summaryview);
3302 gtk_clist_freeze(GTK_CLIST(ctree));
3304 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3305 node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
3306 child = GTK_CTREE_ROW(node)->children;
3307 sibling = GTK_CTREE_ROW(node)->sibling;
3309 while (child != NULL) {
3310 next_child = GTK_CTREE_ROW(child)->sibling;
3311 gtk_ctree_move(ctree, child, NULL, sibling);
3316 gtk_clist_thaw(GTK_CLIST(ctree));
3317 gtk_signal_handler_unblock_by_func(GTK_OBJECT(ctree),
3318 summary_tree_collapsed, summaryview);
3320 debug_print(_("done.\n"));
3321 STATUSBAR_POP(summaryview->mainwin);
3322 main_window_cursor_normal(summaryview->mainwin);
3325 static void summary_unthread_for_exec(SummaryView *summaryview)
3328 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3330 debug_print(_("Unthreading for execution..."));
3332 gtk_clist_freeze(GTK_CLIST(ctree));
3334 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3335 node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
3336 summary_unthread_for_exec_func(ctree, node, NULL);
3339 gtk_clist_thaw(GTK_CLIST(ctree));
3341 debug_print(_("done.\n"));
3344 static void summary_unthread_for_exec_func(GtkCTree *ctree, GtkCTreeNode *node,
3348 GtkCTreeNode *top_parent;
3349 GtkCTreeNode *child;
3350 GtkCTreeNode *sibling;
3352 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3355 (!MSG_IS_MOVE(msginfo->flags) &&
3356 !MSG_IS_DELETED(msginfo->flags)))
3358 child = GTK_CTREE_ROW(node)->children;
3361 for (top_parent = node;
3362 GTK_CTREE_ROW(top_parent)->parent != NULL;
3363 top_parent = GTK_CTREE_ROW(top_parent)->parent)
3365 sibling = GTK_CTREE_ROW(top_parent)->sibling;
3367 while (child != NULL) {
3368 GtkCTreeNode *next_child;
3370 next_child = GTK_CTREE_ROW(child)->sibling;
3371 gtk_ctree_move(ctree, child, NULL, sibling);
3376 void summary_filter(SummaryView *summaryview)
3378 if (!prefs_common.fltlist) return;
3380 debug_print(_("filtering..."));
3381 STATUSBAR_PUSH(summaryview->mainwin, _("Filtering..."));
3382 main_window_cursor_wait(summaryview->mainwin);
3384 gtk_clist_freeze(GTK_CLIST(summaryview->ctree));
3386 if (prefs_filtering == NULL) {
3387 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree), NULL,
3388 GTK_CTREE_FUNC(summary_filter_func),
3391 gtk_clist_thaw(GTK_CLIST(summaryview->ctree));
3393 if (prefs_common.immediate_exec)
3394 summary_execute(summaryview);
3396 summary_status_show(summaryview);
3399 summaryview->folder_table = g_hash_table_new(NULL, NULL);
3401 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree), NULL,
3402 GTK_CTREE_FUNC(summary_filter_func),
3405 gtk_clist_thaw(GTK_CLIST(summaryview->ctree));
3407 folder_item_scan_foreach(summaryview->folder_table);
3408 folderview_update_item_foreach(summaryview->folder_table);
3410 g_hash_table_destroy(summaryview->folder_table);
3411 summaryview->folder_table = NULL;
3413 summary_show(summaryview, summaryview->folder_item, FALSE);
3416 debug_print(_("done.\n"));
3417 STATUSBAR_POP(summaryview->mainwin);
3418 main_window_cursor_normal(summaryview->mainwin);
3421 static void summary_filter_func(GtkCTree *ctree, GtkCTreeNode *node,
3424 MsgInfo *msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3425 SummaryView *summaryview = data;
3429 if (prefs_filtering == NULL) {
3431 file = procmsg_get_message_file_path(msginfo);
3432 dest = filter_get_dest_folder(prefs_common.fltlist, file);
3435 if (dest && strcmp2(dest->path, FILTER_NOT_RECEIVE) != 0 &&
3436 summaryview->folder_item != dest)
3437 summary_move_row_to(summaryview, node, dest);
3440 filter_msginfo_move_or_delete(prefs_filtering, msginfo,
3441 summaryview->folder_table);
3446 #define N_COLOR_LABELS colorlabel_get_color_count()
3448 static void summary_colorlabel_menu_item_activate_cb(GtkWidget *widget,
3451 guint color = GPOINTER_TO_UINT(data);
3452 SummaryView *summaryview;
3454 summaryview = gtk_object_get_data(GTK_OBJECT(widget), "summaryview");
3455 g_return_if_fail(summaryview != NULL);
3457 /* "dont_toggle" state set? */
3458 if (gtk_object_get_data(GTK_OBJECT(summaryview->colorlabel_menu),
3462 summary_set_colorlabel(summaryview, color, NULL);
3465 /* summary_set_colorlabel_color() - labelcolor parameter is the color *flag*
3466 * for the messsage; not the color index */
3467 void summary_set_colorlabel_color(GtkCTree *ctree, GtkCTreeNode *node,
3471 GtkStyle *style, *prev_style, *ctree_style;
3475 msginfo = gtk_ctree_node_get_row_data(ctree, node);
3476 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_CLABEL_FLAG_MASK);
3477 MSG_SET_COLORLABEL_VALUE(msginfo->flags, labelcolor);
3479 color_index = labelcolor == 0 ? -1 : (gint)labelcolor - 1;
3480 ctree_style = gtk_widget_get_style(GTK_WIDGET(ctree));
3481 prev_style = gtk_ctree_node_get_row_style(ctree, node);
3483 if (color_index < 0 || color_index >= N_COLOR_LABELS) {
3484 if (!prev_style) return;
3485 style = gtk_style_copy(prev_style);
3486 color = ctree_style->fg[GTK_STATE_NORMAL];
3487 style->fg[GTK_STATE_NORMAL] = color;
3488 color = ctree_style->fg[GTK_STATE_SELECTED];
3489 style->fg[GTK_STATE_SELECTED] = color;
3492 style = gtk_style_copy(prev_style);
3494 style = gtk_style_copy(ctree_style);
3495 color = colorlabel_get_color(color_index);
3496 style->fg[GTK_STATE_NORMAL] = color;
3497 style->fg[GTK_STATE_SELECTED] = color;
3500 gtk_ctree_node_set_row_style(ctree, node, style);
3503 void summary_set_colorlabel(SummaryView *summaryview, guint labelcolor,
3506 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3507 GtkCList *clist = GTK_CLIST(summaryview->ctree);
3510 for (cur = clist->selection; cur != NULL; cur = cur->next)
3511 summary_set_colorlabel_color(ctree, GTK_CTREE_NODE(cur->data),
3515 static void summary_colorlabel_menu_item_activate_item_cb(GtkMenuItem *menu_item,
3518 SummaryView *summaryview;
3520 GtkCheckMenuItem **items;
3524 summaryview = (SummaryView *)data;
3525 g_return_if_fail(summaryview != NULL);
3527 sel = GTK_CLIST(summaryview->ctree)->selection;
3530 menu = GTK_MENU_SHELL(summaryview->colorlabel_menu);
3531 g_return_if_fail(menu != NULL);
3533 Xalloca(items, (N_COLOR_LABELS + 1) * sizeof(GtkWidget *), return);
3535 /* NOTE: don't return prematurely because we set the "dont_toggle"
3536 * state for check menu items */
3537 gtk_object_set_data(GTK_OBJECT(menu), "dont_toggle",
3538 GINT_TO_POINTER(1));
3540 /* clear items. get item pointers. */
3541 for (n = 0, cur = menu->children; cur != NULL; cur = cur->next) {
3542 if (GTK_IS_CHECK_MENU_ITEM(cur->data)) {
3543 gtk_check_menu_item_set_state
3544 (GTK_CHECK_MENU_ITEM(cur->data), FALSE);
3545 items[n] = GTK_CHECK_MENU_ITEM(cur->data);
3550 if (n == (N_COLOR_LABELS + 1)) {
3551 /* iterate all messages and set the state of the appropriate
3553 for (; sel != NULL; sel = sel->next) {
3557 msginfo = gtk_ctree_node_get_row_data
3558 (GTK_CTREE(summaryview->ctree),
3559 GTK_CTREE_NODE(sel->data));
3561 clabel = MSG_GET_COLORLABEL_VALUE(msginfo->flags);
3562 if (!items[clabel]->active)
3563 gtk_check_menu_item_set_state
3564 (items[clabel], TRUE);
3568 g_warning("invalid number of color elements (%d)\n", n);
3570 /* reset "dont_toggle" state */
3571 gtk_object_set_data(GTK_OBJECT(menu), "dont_toggle",
3572 GINT_TO_POINTER(0));
3575 static void summary_colorlabel_menu_create(SummaryView *summaryview)
3577 GtkWidget *label_menuitem;
3582 label_menuitem = gtk_item_factory_get_item(summaryview->popupfactory,
3584 gtk_signal_connect(GTK_OBJECT(label_menuitem), "activate",
3585 GTK_SIGNAL_FUNC(summary_colorlabel_menu_item_activate_item_cb),
3588 gtk_widget_show(label_menuitem);
3590 menu = gtk_menu_new();
3592 /* create sub items. for the menu item activation callback we pass the
3593 * index of label_colors[] as data parameter. for the None color we
3594 * pass an invalid (high) value. also we attach a data pointer so we
3595 * can always get back the SummaryView pointer. */
3597 item = gtk_check_menu_item_new_with_label(_("None"));
3598 gtk_menu_append(GTK_MENU(menu), item);
3599 gtk_signal_connect(GTK_OBJECT(item), "activate",
3600 GTK_SIGNAL_FUNC(summary_colorlabel_menu_item_activate_cb),
3601 GUINT_TO_POINTER(0));
3602 gtk_object_set_data(GTK_OBJECT(item), "summaryview", summaryview);
3603 gtk_widget_show(item);
3605 item = gtk_menu_item_new();
3606 gtk_menu_append(GTK_MENU(menu), item);
3607 gtk_widget_show(item);
3609 /* create pixmap/label menu items */
3610 for (i = 0; i < N_COLOR_LABELS; i++) {
3611 item = colorlabel_create_check_color_menu_item(i);
3612 gtk_menu_append(GTK_MENU(menu), item);
3613 gtk_signal_connect(GTK_OBJECT(item), "activate",
3614 GTK_SIGNAL_FUNC(summary_colorlabel_menu_item_activate_cb),
3615 GUINT_TO_POINTER(i + 1));
3616 gtk_object_set_data(GTK_OBJECT(item), "summaryview",
3618 gtk_widget_show(item);
3621 gtk_widget_show(menu);
3622 gtk_menu_item_set_submenu(GTK_MENU_ITEM(label_menuitem), menu);
3623 summaryview->colorlabel_menu = menu;
3626 /* callback functions */
3628 static void summary_toggle_pressed(GtkWidget *eventbox, GdkEventButton *event,
3629 SummaryView *summaryview)
3634 if (!summaryview->msg_is_toggled_on && summaryview->selected)
3635 summary_display_msg(summaryview, summaryview->selected, FALSE);
3637 summary_toggle_view(summaryview);
3640 static void summary_button_pressed(GtkWidget *ctree, GdkEventButton *event,
3641 SummaryView *summaryview)
3645 if (event->button == 3) {
3647 summary_set_add_sender_menu(summaryview);
3648 gtk_menu_popup(GTK_MENU(summaryview->popupmenu), NULL, NULL,
3649 NULL, NULL, event->button, event->time);
3650 } else if (event->button == 2) {
3651 summaryview->display_msg = TRUE;
3652 } else if (event->button == 1) {
3653 if (!prefs_common.emulate_emacs &&
3654 summaryview->msg_is_toggled_on)
3655 summaryview->display_msg = TRUE;
3659 static void summary_button_released(GtkWidget *ctree, GdkEventButton *event,
3660 SummaryView *summaryview)
3664 void summary_pass_key_press_event(SummaryView *summaryview, GdkEventKey *event)
3666 summary_key_pressed(summaryview->ctree, event, summaryview);
3669 #define BREAK_ON_MODIFIER_KEY() \
3670 if ((event->state & (GDK_MOD1_MASK|GDK_CONTROL_MASK)) != 0) break
3672 #define RETURN_IF_LOCKED() \
3673 if (summaryview->mainwin->lock_count) return
3675 #define KEY_PRESS_EVENT_STOP() \
3676 if (gtk_signal_n_emissions_by_name \
3677 (GTK_OBJECT(ctree), "key_press_event") > 0) { \
3678 gtk_signal_emit_stop_by_name(GTK_OBJECT(ctree), \
3679 "key_press_event"); \
3682 static void summary_key_pressed(GtkWidget *widget, GdkEventKey *event,
3683 SummaryView *summaryview)
3685 GtkCTree *ctree = GTK_CTREE(widget);
3687 FolderItem *to_folder;
3691 switch (event->keyval) {
3692 case GDK_g: /* Go */
3695 BREAK_ON_MODIFIER_KEY();
3696 KEY_PRESS_EVENT_STOP();
3697 to_folder = foldersel_folder_sel(NULL, NULL);
3699 debug_print(_("Go to %s\n"), to_folder->path);
3700 folderview_select(summaryview->folderview, to_folder);
3703 case GDK_w: /* Write new message */
3704 BREAK_ON_MODIFIER_KEY();
3705 if (summaryview->folder_item) {
3707 ac = summaryview->folder_item->folder->account;
3708 if (ac && ac->protocol == A_NNTP)
3709 compose_new_with_recipient
3710 (ac, summaryview->folder_item->path);
3712 compose_new_with_folderitem(ac, summaryview->folder_item);
3716 case GDK_D: /* Empty trash */
3718 BREAK_ON_MODIFIER_KEY();
3719 KEY_PRESS_EVENT_STOP();
3720 main_window_empty_trash(summaryview->mainwin, TRUE);
3722 case GDK_Q: /* Quit */
3724 BREAK_ON_MODIFIER_KEY();
3726 if (prefs_common.confirm_on_exit) {
3727 if (alertpanel(_("Exit"), _("Exit this program?"),
3728 _("OK"), _("Cancel"), NULL)
3729 == G_ALERTDEFAULT) {
3730 manage_window_focus_in
3731 (summaryview->mainwin->window,
3733 app_will_exit(NULL, summaryview->mainwin);
3737 case GDK_Left: /* Move focus */
3739 gtk_widget_grab_focus(summaryview->folderview->ctree);
3745 if (!summaryview->selected) {
3746 node = gtk_ctree_node_nth(ctree, 0);
3748 gtk_ctree_select(ctree, node);
3753 switch (event->keyval) {
3754 case GDK_space: /* Page down or go to the next */
3755 if (summaryview->displayed != summaryview->selected) {
3756 summary_display_msg(summaryview,
3757 summaryview->selected, FALSE);
3760 if (!textview_scroll_page(summaryview->messageview->textview,
3762 summary_select_next_unread(summaryview);
3764 case GDK_n: /* Next */
3766 BREAK_ON_MODIFIER_KEY();
3767 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
3769 case GDK_BackSpace: /* Page up */
3771 textview_scroll_page(summaryview->messageview->textview, TRUE);
3773 case GDK_p: /* Prev */
3775 BREAK_ON_MODIFIER_KEY();
3776 summary_step(summaryview, GTK_SCROLL_STEP_BACKWARD);
3778 case GDK_v: /* Toggle summary mode / message mode */
3780 BREAK_ON_MODIFIER_KEY();
3782 if (!summaryview->msg_is_toggled_on && summaryview->selected)
3783 summary_display_msg(summaryview,
3784 summaryview->selected, FALSE);
3786 summary_toggle_view(summaryview);
3788 case GDK_Return: /* Scroll up/down one line */
3789 if (summaryview->displayed != summaryview->selected) {
3790 summary_display_msg(summaryview,
3791 summaryview->selected, FALSE);
3794 textview_scroll_one_line(summaryview->messageview->textview,
3795 (event->state & GDK_MOD1_MASK) != 0);
3797 case GDK_asterisk: /* Mark */
3798 summary_mark(summaryview);
3800 case GDK_exclam: /* Mark as unread */
3801 summary_mark_as_unread(summaryview);
3803 case GDK_d: /* Delete */
3805 BREAK_ON_MODIFIER_KEY();
3806 summary_delete(summaryview);
3808 case GDK_u: /* Unmark */
3810 BREAK_ON_MODIFIER_KEY();
3811 summary_unmark(summaryview);
3813 case GDK_o: /* Move */
3815 BREAK_ON_MODIFIER_KEY();
3816 summary_move_to(summaryview);
3818 case GDK_O: /* Copy */
3820 BREAK_ON_MODIFIER_KEY();
3821 summary_copy_to(summaryview);
3823 case GDK_x: /* Execute */
3826 BREAK_ON_MODIFIER_KEY();
3827 KEY_PRESS_EVENT_STOP();
3828 summary_execute(summaryview);
3830 case GDK_a: /* Reply to the message */
3831 BREAK_ON_MODIFIER_KEY();
3832 summary_reply_cb(summaryview,
3833 COMPOSE_REPLY_TO_ALL_WITHOUT_QUOTE, NULL);
3835 case GDK_A: /* Reply to the message with quotation */
3836 BREAK_ON_MODIFIER_KEY();
3837 summary_reply_cb(summaryview,
3838 COMPOSE_REPLY_TO_ALL_WITH_QUOTE, NULL);
3840 case GDK_f: /* Forward the message */
3841 BREAK_ON_MODIFIER_KEY();
3842 summary_reply_cb(summaryview, COMPOSE_FORWARD, NULL);
3845 BREAK_ON_MODIFIER_KEY();
3846 summary_reply_cb(summaryview, COMPOSE_FORWARD_AS_ATTACH, NULL);
3848 case GDK_y: /* Save the message */
3849 BREAK_ON_MODIFIER_KEY();
3850 summary_save_as(summaryview);
3857 #undef BREAK_ON_MODIFIER_KEY
3858 #undef RETURN_IF_LOCKED
3859 #undef KEY_PRESS_EVENT_STOP
3861 static void summary_open_row(GtkSCTree *sctree, SummaryView *summaryview)
3863 if (summaryview->folder_item->stype == F_DRAFT ||
3864 summaryview->folder_item->stype == F_OUTBOX ||
3865 summaryview->folder_item->stype == F_QUEUE)
3866 summary_reedit(summaryview);
3868 summary_open_msg(summaryview);
3870 summaryview->display_msg = FALSE;
3873 static void summary_tree_expanded(GtkCTree *ctree, GtkCTreeNode *node,
3874 SummaryView *summaryview)
3876 summary_set_row_marks(summaryview, node);
3879 static void summary_tree_collapsed(GtkCTree *ctree, GtkCTreeNode *node,
3880 SummaryView *summaryview)
3882 summary_set_row_marks(summaryview, node);
3885 static void summary_selected(GtkCTree *ctree, GtkCTreeNode *row,
3886 gint column, SummaryView *summaryview)
3890 summary_status_show(summaryview);
3891 summary_set_menu_sensitive(summaryview);
3893 if (GTK_CLIST(ctree)->selection &&
3894 GTK_CLIST(ctree)->selection->next) {
3895 summaryview->display_msg = FALSE;
3899 summaryview->selected = row;
3901 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3905 if (MSG_IS_MARKED(msginfo->flags)) {
3906 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_MARKED);
3907 CHANGE_FLAGS(msginfo);
3908 summary_set_row_marks(summaryview, row);
3910 summary_mark_row(summaryview, row);
3913 if (MSG_IS_UNREAD(msginfo->flags)) {
3914 summary_mark_row_as_read(summaryview, row);
3915 summary_status_show(summaryview);
3916 } else if (!MSG_IS_REPLIED(msginfo->flags) &&
3917 !MSG_IS_FORWARDED(msginfo->flags)) {
3918 summary_mark_row_as_unread(summaryview, row);
3919 summary_status_show(summaryview);
3926 if (summaryview->display_msg)
3927 summary_display_msg(summaryview, row, FALSE);
3929 summaryview->display_msg = FALSE;
3932 static void summary_col_resized(GtkCList *clist, gint column, gint width,
3933 SummaryView *summaryview)
3937 prefs_common.summary_col_mark = width;
3940 prefs_common.summary_col_unread = width;
3943 prefs_common.summary_col_mime = width;
3946 prefs_common.summary_col_number = width;
3949 prefs_common.summary_col_score = width;
3952 prefs_common.summary_col_size = width;
3955 prefs_common.summary_col_date = width;
3958 prefs_common.summary_col_from = width;
3961 prefs_common.summary_col_subject = width;
3968 static void summary_reply_cb(SummaryView *summaryview, guint action,
3972 GList *sel = GTK_CLIST(summaryview->ctree)->selection;
3974 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
3975 summaryview->selected);
3976 if (!msginfo) return;
3978 switch ((ComposeReplyMode)action) {
3980 compose_reply(msginfo, prefs_common.reply_with_quote,
3983 case COMPOSE_REPLY_WITH_QUOTE:
3984 compose_reply(msginfo, TRUE, FALSE, FALSE);
3986 case COMPOSE_REPLY_WITHOUT_QUOTE:
3987 compose_reply(msginfo, FALSE, FALSE, FALSE);
3989 case COMPOSE_REPLY_TO_SENDER:
3990 compose_reply(msginfo, prefs_common.reply_with_quote,
3993 case COMPOSE_FOLLOWUP_AND_REPLY_TO:
3994 compose_followup_and_reply_to(msginfo,
3995 prefs_common.reply_with_quote,
3998 case COMPOSE_REPLY_TO_SENDER_WITH_QUOTE:
3999 compose_reply(msginfo, TRUE, FALSE, TRUE);
4001 case COMPOSE_REPLY_TO_SENDER_WITHOUT_QUOTE:
4002 compose_reply(msginfo, FALSE, FALSE, TRUE);
4004 case COMPOSE_REPLY_TO_ALL:
4005 compose_reply(msginfo, prefs_common.reply_with_quote,
4008 case COMPOSE_REPLY_TO_ALL_WITH_QUOTE:
4009 compose_reply(msginfo, TRUE, TRUE, FALSE);
4011 case COMPOSE_REPLY_TO_ALL_WITHOUT_QUOTE:
4012 compose_reply(msginfo, FALSE, TRUE, FALSE);
4014 case COMPOSE_FORWARD:
4016 compose_forward(NULL, msginfo, FALSE);
4019 /* if (sel->next) FALL THROUGH */
4020 case COMPOSE_FORWARD_AS_ATTACH:
4022 GSList *msginfo_list = NULL;
4023 for ( ; sel != NULL; sel = sel->next)
4024 msginfo_list = g_slist_append(msginfo_list,
4025 gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
4026 GTK_CTREE_NODE(sel->data)));
4027 compose_forward_multiple(NULL, msginfo_list);
4028 g_slist_free(msginfo_list);
4032 g_warning("summary_reply_cb(): invalid action: %d\n", action);
4035 summary_set_marks_selected(summaryview);
4038 static void summary_show_all_header_cb(SummaryView *summaryview,
4039 guint action, GtkWidget *widget)
4041 header_window_show_cb(summaryview->mainwin, action, widget);
4044 static void summary_num_clicked(GtkWidget *button, SummaryView *summaryview)
4046 summary_sort(summaryview, SORT_BY_NUMBER);
4049 static void summary_score_clicked(GtkWidget *button,
4050 SummaryView *summaryview)
4052 summary_sort(summaryview, SORT_BY_SCORE);
4055 static void summary_size_clicked(GtkWidget *button, SummaryView *summaryview)
4057 summary_sort(summaryview, SORT_BY_SIZE);
4060 static void summary_date_clicked(GtkWidget *button, SummaryView *summaryview)
4062 summary_sort(summaryview, SORT_BY_DATE);
4065 static void summary_from_clicked(GtkWidget *button, SummaryView *summaryview)
4067 summary_sort(summaryview, SORT_BY_FROM);
4070 static void summary_subject_clicked(GtkWidget *button,
4071 SummaryView *summaryview)
4073 summary_sort(summaryview, SORT_BY_SUBJECT);
4076 static void summary_mark_clicked(GtkWidget *button,
4077 SummaryView *summaryview)
4079 summary_sort(summaryview, SORT_BY_LABEL);
4082 void summary_change_display_item(SummaryView *summaryview)
4084 GtkCList *clist = GTK_CLIST(summaryview->ctree);
4086 gtk_clist_set_column_visibility(clist, S_COL_MARK, prefs_common.show_mark);
4087 gtk_clist_set_column_visibility(clist, S_COL_UNREAD, prefs_common.show_unread);
4088 gtk_clist_set_column_visibility(clist, S_COL_MIME, prefs_common.show_mime);
4089 gtk_clist_set_column_visibility(clist, S_COL_NUMBER, prefs_common.show_number);
4090 gtk_clist_set_column_visibility(clist, S_COL_SCORE, prefs_common.show_score);
4091 gtk_clist_set_column_visibility(clist, S_COL_SIZE, prefs_common.show_size);
4092 gtk_clist_set_column_visibility(clist, S_COL_DATE, prefs_common.show_date);
4093 gtk_clist_set_column_visibility(clist, S_COL_FROM, prefs_common.show_from);
4094 gtk_clist_set_column_visibility(clist, S_COL_SUBJECT, prefs_common.show_subject);
4097 static void summary_start_drag(GtkWidget *widget, gint button, GdkEvent *event,
4098 SummaryView *summaryview)
4100 GtkTargetList *list;
4101 GdkDragContext *context;
4103 g_return_if_fail(summaryview != NULL);
4104 g_return_if_fail(summaryview->folder_item != NULL);
4105 g_return_if_fail(summaryview->folder_item->folder != NULL);
4106 if (summaryview->folder_item->folder->type == F_NEWS ||
4107 summaryview->selected == NULL)
4110 list = gtk_target_list_new(summary_drag_types, 1);
4112 context = gtk_drag_begin(widget, list,
4113 GDK_ACTION_MOVE, button, event);
4114 gtk_drag_set_icon_default(context);
4117 static void summary_drag_data_get(GtkWidget *widget,
4118 GdkDragContext *drag_context,
4119 GtkSelectionData *selection_data,
4122 SummaryView *summaryview)
4124 if (info == TARGET_MAIL_URI_LIST) {
4125 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4128 gchar *mail_list = NULL, *tmp1, *tmp2;
4130 for (cur = GTK_CLIST(ctree)->selection;
4131 cur != NULL; cur = cur->next) {
4132 msginfo = gtk_ctree_node_get_row_data
4133 (ctree, GTK_CTREE_NODE(cur->data));
4134 tmp2 = procmsg_get_message_file_path(msginfo);
4135 if (!tmp2) continue;
4136 tmp1 = g_strconcat("file:/", tmp2, NULL);
4142 tmp2 = g_strconcat(mail_list, tmp1, NULL);
4149 if (mail_list != NULL) {
4150 gtk_selection_data_set(selection_data,
4151 selection_data->target, 8,
4152 mail_list, strlen(mail_list));
4155 } else if (info == TARGET_DUMMY) {
4156 if (GTK_CLIST(summaryview->ctree)->selection)
4157 gtk_selection_data_set(selection_data,
4158 selection_data->target, 8,
4164 /* custom compare functions for sorting */
4166 static gint summary_cmp_by_num(GtkCList *clist,
4167 gconstpointer ptr1, gconstpointer ptr2)
4169 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4170 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4172 return msginfo1->msgnum - msginfo2->msgnum;
4175 static gint summary_cmp_by_size(GtkCList *clist,
4176 gconstpointer ptr1, gconstpointer ptr2)
4178 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4179 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4181 return msginfo1->size - msginfo2->size;
4184 static gint summary_cmp_by_date(GtkCList *clist,
4185 gconstpointer ptr1, gconstpointer ptr2)
4187 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4188 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4190 return msginfo1->date_t - msginfo2->date_t;
4193 static gint summary_cmp_by_from(GtkCList *clist,
4194 gconstpointer ptr1, gconstpointer ptr2)
4196 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4197 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4199 if (!msginfo1->fromname)
4200 return (msginfo2->fromname != NULL);
4201 if (!msginfo2->fromname)
4204 return strcasecmp(msginfo1->fromname, msginfo2->fromname);
4207 static gint summary_cmp_by_subject(GtkCList *clist,
4208 gconstpointer ptr1, gconstpointer ptr2)
4210 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4211 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4213 if (!msginfo1->subject)
4214 return (msginfo2->subject != NULL);
4215 if (!msginfo2->subject)
4218 return strcasecmp(msginfo1->subject, msginfo2->subject);
4221 static gint summary_cmp_by_label(GtkCList *clist,
4222 gconstpointer ptr1, gconstpointer ptr2)
4224 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4225 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4227 return MSG_GET_COLORLABEL(msginfo1->flags) - MSG_GET_COLORLABEL(msginfo2->flags);
4230 static gint summary_cmp_by_score(GtkCList *clist,
4231 gconstpointer ptr1, gconstpointer ptr2)
4233 MsgInfo *msginfo1 = ((GtkCListRow *)ptr1)->data;
4234 MsgInfo *msginfo2 = ((GtkCListRow *)ptr2)->data;
4237 /* if score are equal, sort by date */
4239 diff = msginfo1->threadscore - msginfo2->threadscore;
4243 return summary_cmp_by_date(clist, ptr1, ptr2);
4246 static void summary_ignore_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
4248 SummaryView *summaryview = (SummaryView *) data;
4251 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4252 if (MSG_IS_NEW(msginfo->flags))
4253 summaryview->newmsgs--;
4254 if (MSG_IS_UNREAD(msginfo->flags))
4255 summaryview->unread--;
4256 MSG_SET_PERM_FLAGS(msginfo->flags, MSG_IGNORE_THREAD);
4258 CHANGE_FLAGS(msginfo);
4260 summary_set_row_marks(summaryview, row);
4261 debug_print(_("Message %d is marked as ignore thread\n"),
4265 static void summary_ignore_thread(SummaryView *summaryview)
4267 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4270 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next) {
4271 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data), GTK_CTREE_FUNC(summary_ignore_thread_func), summaryview);
4274 summary_status_show(summaryview);
4277 static void summary_unignore_thread_func(GtkCTree *ctree, GtkCTreeNode *row, gpointer data)
4279 SummaryView *summaryview = (SummaryView *) data;
4282 msginfo = gtk_ctree_node_get_row_data(ctree, row);
4283 if (MSG_IS_NEW(msginfo->flags))
4284 summaryview->newmsgs++;
4285 if (MSG_IS_UNREAD(msginfo->flags))
4286 summaryview->unread++;
4287 MSG_UNSET_PERM_FLAGS(msginfo->flags, MSG_IGNORE_THREAD);
4289 CHANGE_FLAGS(msginfo);
4291 summary_set_row_marks(summaryview, row);
4292 debug_print(_("Message %d is marked as unignore thread\n"),
4296 static void summary_unignore_thread(SummaryView *summaryview)
4298 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
4301 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next) {
4302 gtk_ctree_pre_recursive(ctree, GTK_CTREE_NODE(cur->data), GTK_CTREE_FUNC(summary_unignore_thread_func), summaryview);
4305 summary_status_show(summaryview);