2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2002 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>
54 #include "mainwindow.h"
55 #include "folderview.h"
56 #include "summaryview.h"
57 #include "messageview.h"
58 #include "foldersel.h"
60 #include "procheader.h"
61 #include "sourcewindow.h"
62 #include "prefs_common.h"
63 #include "prefs_summary_column.h"
64 #include "prefs_filter.h"
65 #include "prefs_filtering.h"
70 #include "stock_pixmap.h"
72 #include "alertpanel.h"
73 #include "inputdialog.h"
74 #include "statusbar.h"
77 #include "colorlabel.h"
80 #include "addressbook.h"
81 #include "addr_compl.h"
83 #include "prefs_folder_item.h"
84 #include "filtering.h"
85 #include "string_match.h"
88 #define SUMMARY_COL_MARK_WIDTH 10
89 #define SUMMARY_COL_UNREAD_WIDTH 13
90 #define SUMMARY_COL_LOCKED_WIDTH 13
91 #define SUMMARY_COL_MIME_WIDTH 11
93 static GdkFont *boldfont;
94 static GdkFont *smallfont;
96 static GtkStyle *bold_style;
97 static GtkStyle *bold_marked_style;
98 static GtkStyle *bold_deleted_style;
99 static GtkStyle *small_style;
100 static GtkStyle *small_marked_style;
101 static GtkStyle *small_deleted_style;
103 static GdkPixmap *markxpm;
104 static GdkBitmap *markxpmmask;
105 static GdkPixmap *deletedxpm;
106 static GdkBitmap *deletedxpmmask;
108 static GdkPixmap *newxpm;
109 static GdkBitmap *newxpmmask;
110 static GdkPixmap *unreadxpm;
111 static GdkBitmap *unreadxpmmask;
112 static GdkPixmap *repliedxpm;
113 static GdkBitmap *repliedxpmmask;
114 static GdkPixmap *forwardedxpm;
115 static GdkBitmap *forwardedxpmmask;
116 static GdkPixmap *ignorethreadxpm;
117 static GdkBitmap *ignorethreadxpmmask;
118 static GdkPixmap *lockedxpm;
119 static GdkBitmap *lockedxpmmask;
121 static GdkPixmap *clipxpm;
122 static GdkBitmap *clipxpmmask;
123 static GdkPixmap *keyxpm;
124 static GdkBitmap *keyxpmmask;
125 static GdkPixmap *clipkeyxpm;
126 static GdkBitmap *clipkeyxpmmask;
128 static void summary_free_msginfo_func (GtkCTree *ctree,
131 static void summary_set_marks_func (GtkCTree *ctree,
135 static void summary_set_menu_sensitive (SummaryView *summaryview);
137 static void summary_set_hide_read_msgs_menu (SummaryView *summaryview,
140 static guint summary_get_msgnum (SummaryView *summaryview,
143 static GtkCTreeNode *summary_find_prev_msg
144 (SummaryView *summaryview,
145 GtkCTreeNode *current_node);
146 static GtkCTreeNode *summary_find_next_msg
147 (SummaryView *summaryview,
148 GtkCTreeNode *current_node);
150 static GtkCTreeNode *summary_find_prev_flagged_msg
151 (SummaryView *summaryview,
152 GtkCTreeNode *current_node,
154 gboolean start_from_prev);
155 static GtkCTreeNode *summary_find_next_flagged_msg
156 (SummaryView *summaryview,
157 GtkCTreeNode *current_node,
159 gboolean start_from_next);
161 static GtkCTreeNode *summary_find_msg_by_msgnum
162 (SummaryView *summaryview,
165 static void summary_update_status (SummaryView *summaryview);
167 /* display functions */
168 static void summary_status_show (SummaryView *summaryview);
169 static void summary_set_column_titles (SummaryView *summaryview);
170 static void summary_set_ctree_from_list (SummaryView *summaryview,
172 static void summary_set_header (SummaryView *summaryview,
175 static void summary_display_msg (SummaryView *summaryview,
177 static void summary_display_msg_full (SummaryView *summaryview,
180 gboolean all_headers);
181 static void summary_set_row_marks (SummaryView *summaryview,
184 /* message handling */
185 static void summary_mark_row (SummaryView *summaryview,
187 static void summary_lock_row (SummaryView *summaryview,
189 static void summary_mark_row_as_read (SummaryView *summaryview,
191 static void summary_mark_row_as_unread (SummaryView *summaryview,
193 static void summary_delete_row (SummaryView *summaryview,
195 static void summary_unmark_row (SummaryView *summaryview,
197 static void summary_move_row_to (SummaryView *summaryview,
199 FolderItem *to_folder);
200 static void summary_copy_row_to (SummaryView *summaryview,
202 FolderItem *to_folder);
204 static void summary_delete_duplicated_func
207 SummaryView *summaryview);
209 static void summary_execute_move (SummaryView *summaryview);
210 static void summary_execute_move_func (GtkCTree *ctree,
213 static void summary_execute_copy (SummaryView *summaryview);
214 static void summary_execute_copy_func (GtkCTree *ctree,
217 static void summary_execute_delete (SummaryView *summaryview);
218 static void summary_execute_delete_func (GtkCTree *ctree,
222 static void summary_thread_init (SummaryView *summaryview);
223 static void summary_ignore_thread (SummaryView *summaryview);
224 static void summary_unignore_thread (SummaryView *summaryview);
226 static void summary_unthread_for_exec (SummaryView *summaryview);
227 static void summary_unthread_for_exec_func (GtkCTree *ctree,
231 void summary_simplify_subject(SummaryView *summaryview, gchar * rexp,
235 void summary_processing(SummaryView *summaryview, GSList * mlist);
237 static void summary_filter_func (GtkCTree *ctree,
241 static void summary_colorlabel_menu_item_activate_cb
244 static void summary_colorlabel_menu_item_activate_item_cb
245 (GtkMenuItem *label_menu_item,
247 static void summary_colorlabel_menu_create(SummaryView *summaryview);
249 static GtkWidget *summary_ctree_create (SummaryView *summaryview);
251 /* callback functions */
252 static void summary_toggle_pressed (GtkWidget *eventbox,
253 GdkEventButton *event,
254 SummaryView *summaryview);
255 static void summary_button_pressed (GtkWidget *ctree,
256 GdkEventButton *event,
257 SummaryView *summaryview);
258 static void summary_button_released (GtkWidget *ctree,
259 GdkEventButton *event,
260 SummaryView *summaryview);
261 static void summary_key_pressed (GtkWidget *ctree,
263 SummaryView *summaryview);
264 static void summary_searchbar_pressed (GtkWidget *ctree,
266 SummaryView *summaryview);
267 static void summary_searchtype_changed (GtkMenuItem *widget,
269 static void summary_open_row (GtkSCTree *sctree,
270 SummaryView *summaryview);
271 static void summary_tree_expanded (GtkCTree *ctree,
273 SummaryView *summaryview);
274 static void summary_tree_collapsed (GtkCTree *ctree,
276 SummaryView *summaryview);
277 static void summary_selected (GtkCTree *ctree,
280 SummaryView *summaryview);
281 static void summary_col_resized (GtkCList *clist,
284 SummaryView *summaryview);
285 static void summary_reply_cb (SummaryView *summaryview,
288 static void summary_execute_cb (SummaryView *summaryview,
291 static void summary_show_all_header_cb (SummaryView *summaryview,
295 static void summary_add_address_cb (SummaryView *summaryview,
298 static void summary_create_filter_cb (SummaryView *summaryview,
302 static void summary_mark_clicked (GtkWidget *button,
303 SummaryView *summaryview);
304 static void summary_unread_clicked (GtkWidget *button,
305 SummaryView *summaryview);
306 static void summary_mime_clicked (GtkWidget *button,
307 SummaryView *summaryview);
308 static void summary_num_clicked (GtkWidget *button,
309 SummaryView *summaryview);
310 static void summary_score_clicked (GtkWidget *button,
311 SummaryView *summaryview);
312 static void summary_size_clicked (GtkWidget *button,
313 SummaryView *summaryview);
314 static void summary_date_clicked (GtkWidget *button,
315 SummaryView *summaryview);
316 static void summary_from_clicked (GtkWidget *button,
317 SummaryView *summaryview);
318 static void summary_subject_clicked (GtkWidget *button,
319 SummaryView *summaryview);
320 static void summary_score_clicked (GtkWidget *button,
321 SummaryView *summaryview);
322 static void summary_locked_clicked (GtkWidget *button,
323 SummaryView *summaryview);
325 static void summary_start_drag (GtkWidget *widget,
328 SummaryView *summaryview);
329 static void summary_drag_data_get (GtkWidget *widget,
330 GdkDragContext *drag_context,
331 GtkSelectionData *selection_data,
334 SummaryView *summaryview);
336 /* custom compare functions for sorting */
338 static gint summary_cmp_by_mark (GtkCList *clist,
341 static gint summary_cmp_by_unread (GtkCList *clist,
344 static gint summary_cmp_by_mime (GtkCList *clist,
347 static gint summary_cmp_by_num (GtkCList *clist,
350 static gint summary_cmp_by_size (GtkCList *clist,
353 static gint summary_cmp_by_date (GtkCList *clist,
356 static gint summary_cmp_by_from (GtkCList *clist,
359 static gint summary_cmp_by_to (GtkCList *clist,
362 static gint summary_cmp_by_subject (GtkCList *clist,
365 static gint summary_cmp_by_simplified_subject
366 (GtkCList *clist, gconstpointer ptr1, gconstpointer ptr2);
367 static gint summary_cmp_by_score (GtkCList *clist,
370 static gint summary_cmp_by_locked (GtkCList *clist,
371 gconstpointer ptr1, gconstpointer ptr2);
372 static gint summary_cmp_by_label (GtkCList *clist,
376 static void news_flag_crosspost (MsgInfo *msginfo);
378 static void tog_searchbar_cb (GtkWidget *w,
382 GtkTargetEntry summary_drag_types[1] =
384 {"text/plain", GTK_TARGET_SAME_APP, TARGET_DUMMY}
387 static GtkItemFactoryEntry summary_popup_entries[] =
389 {N_("/_Reply"), NULL, summary_reply_cb, COMPOSE_REPLY, NULL},
390 {N_("/Repl_y to sender"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_SENDER, NULL},
391 {N_("/Follow-up and reply to"), NULL, summary_reply_cb, COMPOSE_FOLLOWUP_AND_REPLY_TO, NULL},
392 {N_("/Reply to a_ll"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_ALL, NULL},
393 {N_("/_Forward"), NULL, summary_reply_cb, COMPOSE_FORWARD, NULL},
394 {N_("/Redirect"), NULL, summary_reply_cb, COMPOSE_REDIRECT, NULL},
395 {N_("/---"), NULL, NULL, 0, "<Separator>"},
396 {N_("/Re-_edit"), NULL, summary_reedit, 0, NULL},
397 {N_("/---"), NULL, NULL, 0, "<Separator>"},
398 {N_("/Select _thread"), NULL, summary_select_thread, 0, NULL},
399 {N_("/Select _all"), NULL, summary_select_all, 0, NULL},
400 {N_("/---"), NULL, NULL, 0, "<Separator>"},
401 {N_("/M_ove..."), NULL, summary_move_to, 0, NULL},
402 {N_("/_Copy..."), NULL, summary_copy_to, 0, NULL},
403 {N_("/_Delete"), NULL, summary_delete, 0, NULL},
404 {N_("/Cancel a news message"), NULL, summary_cancel, 0, NULL},
405 {N_("/E_xecute"), NULL, summary_execute_cb, 0, NULL},
406 {N_("/---"), NULL, NULL, 0, "<Separator>"},
407 {N_("/_Mark"), NULL, NULL, 0, "<Branch>"},
408 {N_("/_Mark/_Mark"), NULL, summary_mark, 0, NULL},
409 {N_("/_Mark/_Unmark"), NULL, summary_unmark, 0, NULL},
410 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
411 {N_("/_Mark/Mark as unr_ead"), NULL, summary_mark_as_unread, 0, NULL},
412 {N_("/_Mark/Mark as rea_d"), NULL, summary_mark_as_read, 0, NULL},
413 {N_("/_Mark/Mark all read"), NULL, summary_mark_all_read, 0, NULL},
414 {N_("/_Mark/Ignore thread"), NULL, summary_ignore_thread, 0, NULL},
415 {N_("/_Mark/Unignore thread"), NULL, summary_unignore_thread, 0, NULL},
416 {N_("/Color la_bel"), NULL, NULL, 0, NULL},
418 {N_("/---"), NULL, NULL, 0, "<Separator>"},
419 {N_("/Add sender to address boo_k"),
420 NULL, summary_add_address_cb, 0, NULL},
421 {N_("/Create f_ilter rule"), NULL, NULL, 0, "<Branch>"},
422 {N_("/Create f_ilter rule/_Automatically"),
423 NULL, summary_create_filter_cb, FILTER_BY_AUTO, NULL},
424 {N_("/Create f_ilter rule/by _From"),
425 NULL, summary_create_filter_cb, FILTER_BY_FROM, NULL},
426 {N_("/Create f_ilter rule/by _To"),
427 NULL, summary_create_filter_cb, FILTER_BY_TO, NULL},
428 {N_("/Create f_ilter rule/by _Subject"),
429 NULL, summary_create_filter_cb, FILTER_BY_SUBJECT, NULL},
430 {N_("/---"), NULL, NULL, 0, "<Separator>"},
431 {N_("/_View"), NULL, NULL, 0, "<Branch>"},
432 {N_("/_View/Open in new _window"),
433 NULL, summary_open_msg, 0, NULL},
434 {N_("/_View/_Source"), NULL, summary_view_source, 0, NULL},
435 {N_("/_View/All _header"), NULL, summary_show_all_header_cb, 0, "<ToggleItem>"},
436 {N_("/---"), NULL, NULL, 0, "<Separator>"},
437 {N_("/_Save as..."), NULL, summary_save_as, 0, NULL},
438 {N_("/_Print..."), NULL, summary_print, 0, NULL},
441 static const gchar *const col_label[N_SUMMARY_COLS] = {
442 N_("M"), /* S_COL_MARK */
443 N_("U"), /* S_COL_UNREAD */
445 N_("Subject"), /* S_COL_SUBJECT */
446 N_("From"), /* S_COL_FROM */
447 N_("Date"), /* S_COL_DATE */
448 N_("Size"), /* S_COL_SIZE */
449 N_("No."), /* S_COL_NUMBER */
450 N_("Score"), /* S_COL_SCORE */
451 N_("L") /* S_COL_LOCKED */
454 SummaryView *summary_create(void)
456 SummaryView *summaryview;
458 GtkWidget *scrolledwin;
462 GtkWidget *hbox_search;
463 GtkWidget *statlabel_folder;
464 GtkWidget *statlabel_select;
465 GtkWidget *statlabel_msgs;
467 GtkWidget *toggle_eventbox;
468 GtkWidget *toggle_arrow;
469 GtkWidget *popupmenu;
470 GtkWidget *search_type_opt;
471 GtkWidget *search_type;
472 GtkWidget *search_string;
474 GtkWidget *toggle_search;
475 GtkTooltips *search_tip;
476 GtkItemFactory *popupfactory;
479 debug_print("Creating summary view...\n");
480 summaryview = g_new0(SummaryView, 1);
482 vbox = gtk_vbox_new(FALSE, 3);
484 /* create status label */
485 hbox = gtk_hbox_new(FALSE, 0);
487 search_tip = gtk_tooltips_new();
488 toggle_search = gtk_toggle_button_new();
490 gtk_tooltips_set_tip(GTK_TOOLTIPS(search_tip),
492 _("Toggle quick-search bar"), NULL);
494 gtk_box_pack_start(GTK_BOX(hbox), toggle_search, FALSE, FALSE, 2);
496 hbox_l = gtk_hbox_new(FALSE, 0);
497 gtk_box_pack_start(GTK_BOX(hbox), hbox_l, TRUE, TRUE, 0);
499 statlabel_folder = gtk_label_new("");
500 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_folder, FALSE, FALSE, 2);
501 statlabel_select = gtk_label_new("");
502 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_select, FALSE, FALSE, 12);
504 /* toggle view button */
505 toggle_eventbox = gtk_event_box_new();
506 gtk_box_pack_end(GTK_BOX(hbox), toggle_eventbox, FALSE, FALSE, 4);
507 toggle_arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
508 gtk_container_add(GTK_CONTAINER(toggle_eventbox), toggle_arrow);
509 gtk_signal_connect(GTK_OBJECT(toggle_eventbox), "button_press_event",
510 GTK_SIGNAL_FUNC(summary_toggle_pressed),
514 statlabel_msgs = gtk_label_new("");
515 gtk_box_pack_end(GTK_BOX(hbox), statlabel_msgs, FALSE, FALSE, 4);
517 hbox_spc = gtk_hbox_new(FALSE, 0);
518 gtk_box_pack_end(GTK_BOX(hbox), hbox_spc, FALSE, FALSE, 6);
520 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
521 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
522 GTK_POLICY_AUTOMATIC,
524 gtk_box_pack_start(GTK_BOX(vbox), scrolledwin, TRUE, TRUE, 0);
525 gtk_widget_set_usize(vbox,
526 prefs_common.summaryview_width,
527 prefs_common.summaryview_height);
529 ctree = summary_ctree_create(summaryview);
531 gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
532 GTK_CLIST(ctree)->hadjustment);
533 gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
534 GTK_CLIST(ctree)->vadjustment);
535 gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
538 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
541 hbox_search = gtk_hbox_new(FALSE, 0);
542 gtk_box_pack_start(GTK_BOX(vbox), hbox_search, FALSE, FALSE, 0);
544 search_type_opt = gtk_option_menu_new();
545 gtk_widget_show(search_type_opt);
546 gtk_box_pack_start(GTK_BOX(hbox_search), search_type_opt, FALSE, FALSE, 0);
548 search_type = gtk_menu_new();
549 MENUITEM_ADD (search_type, menuitem, _("Subject"), S_SEARCH_SUBJECT);
550 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
551 GTK_SIGNAL_FUNC(summary_searchtype_changed),
553 MENUITEM_ADD (search_type, menuitem, _("From"), S_SEARCH_FROM);
554 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
555 GTK_SIGNAL_FUNC(summary_searchtype_changed),
557 MENUITEM_ADD (search_type, menuitem, _("To"), S_SEARCH_TO);
558 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
559 GTK_SIGNAL_FUNC(summary_searchtype_changed),
562 gtk_option_menu_set_menu(GTK_OPTION_MENU(search_type_opt), search_type);
563 gtk_widget_show(search_type);
565 search_string = gtk_entry_new();
567 gtk_box_pack_start(GTK_BOX(hbox_search), search_string, FALSE, FALSE, 2);
569 gtk_widget_show(search_string);
570 gtk_widget_show(hbox_search);
572 gtk_signal_connect(GTK_OBJECT(search_string), "key_press_event",
573 GTK_SIGNAL_FUNC(summary_searchbar_pressed),
576 gtk_signal_connect (GTK_OBJECT(toggle_search), "toggled",
577 GTK_SIGNAL_FUNC(tog_searchbar_cb), hbox_search);
579 /* create popup menu */
580 n_entries = sizeof(summary_popup_entries) /
581 sizeof(summary_popup_entries[0]);
582 popupmenu = menu_create_items(summary_popup_entries, n_entries,
583 "<SummaryView>", &popupfactory,
586 summaryview->vbox = vbox;
587 summaryview->scrolledwin = scrolledwin;
588 summaryview->ctree = ctree;
589 summaryview->hbox = hbox;
590 summaryview->hbox_l = hbox_l;
591 summaryview->hbox_search = hbox_search;
592 summaryview->statlabel_folder = statlabel_folder;
593 summaryview->statlabel_select = statlabel_select;
594 summaryview->statlabel_msgs = statlabel_msgs;
595 summaryview->toggle_eventbox = toggle_eventbox;
596 summaryview->toggle_arrow = toggle_arrow;
597 summaryview->toggle_search = toggle_search;
598 summaryview->popupmenu = popupmenu;
599 summaryview->popupfactory = popupfactory;
600 summaryview->lock_count = 0;
601 summaryview->search_type_opt = search_type_opt;
602 summaryview->search_type = search_type;
603 summaryview->search_string = search_string;
605 /* CLAWS: need this to get the SummaryView * from
607 gtk_object_set_data(GTK_OBJECT(ctree), "summaryview", (gpointer)summaryview);
609 gtk_widget_show_all(vbox);
614 void summary_init(SummaryView *summaryview)
619 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_MARK,
620 &markxpm, &markxpmmask);
621 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_DELETED,
622 &deletedxpm, &deletedxpmmask);
623 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_NEW,
624 &newxpm, &newxpmmask);
625 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_UNREAD,
626 &unreadxpm, &unreadxpmmask);
627 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_REPLIED,
628 &repliedxpm, &repliedxpmmask);
629 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_FORWARDED,
630 &forwardedxpm, &forwardedxpmmask);
631 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP,
632 &clipxpm, &clipxpmmask);
633 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_LOCKED,
634 &lockedxpm, &lockedxpmmask);
635 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_IGNORETHREAD,
636 &ignorethreadxpm, &ignorethreadxpmmask);
637 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP_KEY,
638 &clipkeyxpm, &clipkeyxpmmask);
639 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_KEY,
640 &keyxpm, &keyxpmmask);
643 small_style = gtk_style_copy
644 (gtk_widget_get_style(summaryview->ctree));
646 smallfont = gdk_fontset_load(SMALL_FONT);
647 small_style->font = smallfont;
648 small_marked_style = gtk_style_copy(small_style);
649 small_marked_style->fg[GTK_STATE_NORMAL] =
650 summaryview->color_marked;
651 small_deleted_style = gtk_style_copy(small_style);
652 small_deleted_style->fg[GTK_STATE_NORMAL] =
653 summaryview->color_dim;
656 bold_style = gtk_style_copy
657 (gtk_widget_get_style(summaryview->ctree));
659 boldfont = gdk_fontset_load(BOLD_FONT);
660 bold_style->font = boldfont;
661 bold_marked_style = gtk_style_copy(bold_style);
662 bold_marked_style->fg[GTK_STATE_NORMAL] =
663 summaryview->color_marked;
664 bold_deleted_style = gtk_style_copy(bold_style);
665 bold_deleted_style->fg[GTK_STATE_NORMAL] =
666 summaryview->color_dim;
669 style = gtk_style_copy(gtk_widget_get_style
670 (summaryview->statlabel_folder));
672 gtk_widget_set_style(summaryview->statlabel_folder, style);
673 gtk_widget_set_style(summaryview->statlabel_select, style);
674 gtk_widget_set_style(summaryview->statlabel_msgs, style);
676 pixmap = stock_pixmap_widget(summaryview->hbox_l, STOCK_PIXMAP_DIR_OPEN);
677 gtk_box_pack_start(GTK_BOX(summaryview->hbox_l), pixmap, FALSE, FALSE, 4);
678 gtk_box_reorder_child(GTK_BOX(summaryview->hbox_l), pixmap, 0);
679 gtk_widget_show(pixmap);
680 summaryview->folder_pixmap = pixmap;
682 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_QUICKSEARCH);
683 gtk_container_add (GTK_CONTAINER(summaryview->toggle_search), pixmap);
684 gtk_widget_show(pixmap);
686 /* Init summaryview prefs */
687 summaryview->sort_key = SORT_BY_NONE;
688 summaryview->sort_type = SORT_ASCENDING;
690 /* Init summaryview extra data */
691 summaryview->simplify_subject_preg = NULL;
693 summary_clear_list(summaryview);
694 summary_set_column_titles(summaryview);
695 summary_colorlabel_menu_create(summaryview);
696 summary_set_menu_sensitive(summaryview);
700 GtkCTreeNode * summary_find_next_important_score(SummaryView *summaryview,
701 GtkCTreeNode *current_node)
703 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
706 gint best_score = MIN_SCORE;
707 GtkCTreeNode *best_node = NULL;
710 /*node = current_node;*/
711 node = GTK_CTREE_NODE_NEXT(current_node);
713 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
715 for (; node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
716 msginfo = gtk_ctree_node_get_row_data(ctree, node);
717 if (msginfo->score >= summaryview->important_score)
719 if (msginfo->score > best_score) {
720 best_score = msginfo->score;
731 GtkCTreeNode * summary_find_prev_important_score(SummaryView *summaryview,
732 GtkCTreeNode *current_node)
734 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
737 gint best_score = MIN_SCORE;
738 GtkCTreeNode *best_node = NULL;
741 /*node = current_node;*/
742 node = GTK_CTREE_NODE_PREV(current_node);
744 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
746 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
747 msginfo = gtk_ctree_node_get_row_data(ctree, node);
748 if (msginfo->score >= summaryview->important_score)
750 if (msginfo->score > best_score) {
751 best_score = msginfo->score;
762 gboolean summary_show(SummaryView *summaryview, FolderItem *item)
764 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
766 GSList *mlist = NULL;
769 guint selected_msgnum = 0;
770 guint displayed_msgnum = 0;
773 if (summary_is_locked(summaryview)) return FALSE;
776 summary_lock(summaryview);
778 if (item != summaryview->folder_item) {
779 /* changing folder, reset search */
780 gtk_entry_set_text(GTK_ENTRY(summaryview->search_string), "");
783 STATUSBAR_POP(summaryview->mainwin);
785 is_refresh = (item == summaryview->folder_item) ? TRUE : FALSE;
787 selected_msgnum = summary_get_msgnum(summaryview,
788 summaryview->selected);
789 displayed_msgnum = summary_get_msgnum(summaryview,
790 summaryview->displayed);
793 /* process the marks if any */
794 if (summaryview->mainwin->lock_count == 0 &&
795 (summaryview->moved > 0 || summaryview->copied > 0)) {
798 val = alertpanel(_("Process mark"),
799 _("Some marks are left. Process it?"),
800 _("Yes"), _("No"), _("Cancel"));
801 if (G_ALERTDEFAULT == val) {
802 summary_unlock(summaryview);
803 summary_execute(summaryview);
804 summary_lock(summaryview);
805 } else if (G_ALERTALTERNATE == val) {
808 summary_unlock(summaryview);
812 folder_update_op_count();
815 gtk_clist_freeze(GTK_CLIST(ctree));
817 summary_clear_list(summaryview);
818 summary_set_column_titles(summaryview);
820 messageview_clear(summaryview->messageview);
823 if (!item || !item->path || !item->parent || item->no_select ||
824 (item->folder->type == F_MH &&
825 ((buf = folder_item_get_path(item)) == NULL ||
826 change_dir(buf) < 0))) {
828 debug_print("empty folder\n\n");
829 summary_set_hide_read_msgs_menu(summaryview, FALSE);
831 messageview_clear(summaryview->messageview);
832 summary_clear_all(summaryview);
833 summaryview->folder_item = item;
834 gtk_clist_thaw(GTK_CLIST(ctree));
835 summary_unlock(summaryview);
841 summaryview->folder_item = item;
844 gtk_signal_handler_block_by_data(GTK_OBJECT(ctree), summaryview);
846 buf = g_strdup_printf(_("Scanning folder (%s)..."), item->path);
847 debug_print("%s\n", buf);
848 STATUSBAR_PUSH(summaryview->mainwin, buf);
851 main_window_cursor_wait(summaryview->mainwin);
854 mlist = item->folder->get_msg_list(item->folder, item, !update_cache);
857 USE LIST FROM CACHE, WILL NOT DISPLAY ANY MESSAGES DROPED
858 BY OTHER PROGRAMS TO THE FOLDER
860 mlist = folder_item_get_msg_list(item);
862 summary_processing(summaryview, mlist);
864 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
865 MsgInfo * msginfo = (MsgInfo *) cur->data;
867 msginfo->score = score_message(global_scoring, msginfo);
868 if (msginfo->score != MAX_SCORE &&
869 msginfo->score != MIN_SCORE) {
870 msginfo->score += score_message(item->prefs->scoring,
875 if (summaryview->folder_item->hide_read_msgs) {
878 summary_set_hide_read_msgs_menu(summaryview, TRUE);
880 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
881 MsgInfo * msginfo = (MsgInfo *) cur->data;
883 if ((MSG_IS_UNREAD(msginfo->flags)
884 || MSG_IS_MARKED(msginfo->flags)
885 || MSG_IS_LOCKED(msginfo->flags))
886 && !MSG_IS_IGNORE_THREAD(msginfo->flags))
887 not_killed = g_slist_append(not_killed, msginfo);
889 procmsg_msginfo_free(msginfo);
894 summary_set_hide_read_msgs_menu(summaryview, FALSE);
897 if (strlen(gtk_entry_get_text(GTK_ENTRY(summaryview->search_string))) > 0) {
899 gint search_type = GPOINTER_TO_INT(gtk_object_get_user_data(
900 GTK_OBJECT(GTK_MENU_ITEM(gtk_menu_get_active(
901 GTK_MENU(summaryview->search_type))))));
902 gchar *search_string = gtk_entry_get_text(GTK_ENTRY(summaryview->search_string));
903 gchar *searched_header = NULL;
906 for (cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
907 MsgInfo * msginfo = (MsgInfo *) cur->data;
909 switch (search_type) {
910 case S_SEARCH_SUBJECT:
911 searched_header = msginfo->subject;
914 searched_header = msginfo->from;
917 searched_header = msginfo->to;
920 debug_print("unknown search type (%d)\n", search_type);
923 if (searched_header && strcasestr(searched_header, search_string) != NULL)
924 not_killed = g_slist_append(not_killed, msginfo);
926 procmsg_msginfo_free(msginfo);
932 if ((global_scoring || item->prefs->scoring)) {
937 kill_score = prefs_common.kill_score;
938 if (item->prefs->kill_score > kill_score)
939 kill_score = item->prefs->kill_score;
940 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
941 MsgInfo * msginfo = (MsgInfo *) cur->data;
943 if (msginfo->score > kill_score)
944 not_killed = g_slist_append(not_killed, msginfo);
946 procmsg_msginfo_free(msginfo);
952 STATUSBAR_POP(summaryview->mainwin);
954 /* set ctree and hash table from the msginfo list
955 creating thread, and count the number of messages */
956 summary_set_ctree_from_list(summaryview, mlist);
960 if (summaryview->sort_key != SORT_BY_NONE)
961 summary_sort(summaryview, summaryview->sort_key, summaryview->sort_type);
963 gtk_signal_handler_unblock_by_data(GTK_OBJECT(ctree), summaryview);
965 gtk_clist_thaw(GTK_CLIST(ctree));
968 summaryview->displayed =
969 summary_find_msg_by_msgnum(summaryview,
971 if (!summaryview->displayed)
972 messageview_clear(summaryview->messageview);
973 summary_select_by_msgnum(summaryview, selected_msgnum);
974 if (!summaryview->selected) {
975 /* no selected message - select first unread
976 message, but do not display it */
977 node = summary_find_next_flagged_msg(summaryview, NULL,
979 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL)
980 node = gtk_ctree_node_nth
982 item->sort_type == SORT_DESCENDING
983 ? 0 : GTK_CLIST(ctree)->rows - 1);
984 summary_select_node(summaryview, node, FALSE, TRUE);
987 /* select first unread message */
988 if (summaryview->sort_key == SORT_BY_SCORE)
989 node = summary_find_next_important_score(summaryview,
992 node = summary_find_next_flagged_msg(summaryview, NULL,
994 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL) {
995 node = gtk_ctree_node_nth
997 item->sort_type == SORT_DESCENDING
998 ? 0 : GTK_CLIST(ctree)->rows - 1);
1000 if (prefs_common.open_unread_on_enter) {
1001 summary_unlock(summaryview);
1002 summary_select_node(summaryview, node, TRUE, TRUE);
1003 summary_lock(summaryview);
1005 summary_select_node(summaryview, node, FALSE, TRUE);
1008 summary_status_show(summaryview);
1009 summary_set_menu_sensitive(summaryview);
1010 toolbar_set_sensitive(summaryview->mainwin);
1013 STATUSBAR_PUSH(summaryview->mainwin, _("Done."));
1015 main_window_cursor_normal(summaryview->mainwin);
1016 summary_unlock(summaryview);
1022 void summary_clear_list(SummaryView *summaryview)
1024 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1027 gtk_clist_freeze(clist);
1029 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree),
1030 NULL, summary_free_msginfo_func, NULL);
1032 if (summaryview->folder_item) {
1033 summaryview->folder_item->opened = FALSE;
1034 summaryview->folder_item = NULL;
1037 summaryview->display_msg = FALSE;
1039 summaryview->selected = NULL;
1040 summaryview->displayed = NULL;
1041 summaryview->newmsgs = summaryview->unread = 0;
1042 summaryview->messages = summaryview->total_size = 0;
1043 summaryview->deleted = summaryview->moved = 0;
1044 summaryview->copied = 0;
1045 if (summaryview->msgid_table) {
1046 g_hash_table_destroy(summaryview->msgid_table);
1047 summaryview->msgid_table = NULL;
1049 if (summaryview->subject_table) {
1050 g_hash_table_destroy(summaryview->subject_table);
1051 summaryview->subject_table = NULL;
1053 summaryview->mlist = NULL;
1054 if (summaryview->folder_table) {
1055 g_hash_table_destroy(summaryview->folder_table);
1056 summaryview->folder_table = NULL;
1059 gtk_clist_clear(clist);
1060 if (summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
1061 optimal_width = gtk_clist_optimal_column_width
1062 (clist, summaryview->col_pos[S_COL_SUBJECT]);
1063 gtk_clist_set_column_width
1064 (clist, summaryview->col_pos[S_COL_SUBJECT],
1068 gtk_clist_thaw(clist);
1071 void summary_clear_all(SummaryView *summaryview)
1073 summary_clear_list(summaryview);
1074 summary_set_menu_sensitive(summaryview);
1075 toolbar_set_sensitive(summaryview->mainwin);
1076 summary_status_show(summaryview);
1079 void summary_lock(SummaryView *summaryview)
1081 summaryview->lock_count++;
1084 void summary_unlock(SummaryView *summaryview)
1086 if (summaryview->lock_count)
1087 summaryview->lock_count--;
1090 gboolean summary_is_locked(SummaryView *summaryview)
1092 return summaryview->lock_count > 0;
1095 SummarySelection summary_get_selection_type(SummaryView *summaryview)
1097 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1098 SummarySelection selection;
1100 if (!clist->row_list)
1101 selection = SUMMARY_NONE;
1102 else if (!clist->selection)
1103 selection = SUMMARY_SELECTED_NONE;
1104 else if (!clist->selection->next)
1105 selection = SUMMARY_SELECTED_SINGLE;
1107 selection = SUMMARY_SELECTED_MULTIPLE;
1112 static void summary_set_menu_sensitive(SummaryView *summaryview)
1114 GtkItemFactory *ifactory = summaryview->popupfactory;
1115 SummarySelection selection;
1116 GtkWidget *menuitem;
1119 selection = summary_get_selection_type(summaryview);
1120 main_window_set_menu_sensitive(summaryview->mainwin);
1122 if (selection == SUMMARY_NONE) {
1125 submenu = gtk_item_factory_get_widget
1126 (summaryview->popupfactory, "/Mark");
1127 menu_set_insensitive_all(GTK_MENU_SHELL(submenu));
1128 menu_set_insensitive_all
1129 (GTK_MENU_SHELL(summaryview->popupmenu));
1133 if (summaryview->folder_item->folder->type != F_NEWS)
1134 menu_set_sensitive(ifactory, "/Move...", TRUE);
1136 menu_set_sensitive(ifactory, "/Move...", FALSE);
1139 menu_set_sensitive(ifactory, "/Delete", TRUE);
1141 menu_set_sensitive(ifactory, "/Select all", TRUE);
1142 menu_set_sensitive(ifactory, "/Copy...", TRUE);
1143 menu_set_sensitive(ifactory, "/Execute", TRUE);
1145 menu_set_sensitive(ifactory, "/Mark", TRUE);
1146 menu_set_sensitive(ifactory, "/Mark/Mark", TRUE);
1147 menu_set_sensitive(ifactory, "/Mark/Unmark", TRUE);
1149 menu_set_sensitive(ifactory, "/Mark/Mark as unread", TRUE);
1150 menu_set_sensitive(ifactory, "/Mark/Mark as read", TRUE);
1151 menu_set_sensitive(ifactory, "/Mark/Mark all read", TRUE);
1152 menu_set_sensitive(ifactory, "/Mark/Ignore thread", TRUE);
1153 menu_set_sensitive(ifactory, "/Mark/Unignore thread", TRUE);
1155 menu_set_sensitive(ifactory, "/Color label", TRUE);
1157 sens = (selection == SUMMARY_SELECTED_MULTIPLE) ? FALSE : TRUE;
1158 menu_set_sensitive(ifactory, "/Reply", sens);
1159 menu_set_sensitive(ifactory, "/Reply to sender", sens);
1160 menu_set_sensitive(ifactory, "/Reply to all", sens);
1161 menu_set_sensitive(ifactory, "/Forward", TRUE);
1162 menu_set_sensitive(ifactory, "/Redirect", sens);
1164 menu_set_sensitive(ifactory, "/Add sender to address book", sens);
1165 menu_set_sensitive(ifactory, "/Create filter rule", sens);
1167 menu_set_sensitive(ifactory, "/View", sens);
1168 menu_set_sensitive(ifactory, "/View/Open in new window", sens);
1169 menu_set_sensitive(ifactory, "/View/Source", sens);
1170 menu_set_sensitive(ifactory, "/View/All header", sens);
1171 if (summaryview->folder_item->stype == F_OUTBOX ||
1172 summaryview->folder_item->stype == F_DRAFT ||
1173 summaryview->folder_item->stype == F_QUEUE)
1174 menu_set_sensitive(ifactory, "/Re-edit", sens);
1176 menu_set_sensitive(ifactory, "/Re-edit", FALSE);
1178 menu_set_sensitive(ifactory, "/Select thread", sens);
1179 menu_set_sensitive(ifactory, "/Save as...", sens);
1180 menu_set_sensitive(ifactory, "/Print...", TRUE);
1182 if (summaryview->folder_item->folder->account)
1183 sens = summaryview->folder_item->folder->account->protocol
1187 menu_set_sensitive(ifactory, "/Follow-up and reply to", sens);
1188 menu_set_sensitive(ifactory, "/Cancel a news message", sens);
1189 menu_set_sensitive(ifactory, "/Delete", !sens);
1191 summary_lock(summaryview);
1192 menuitem = gtk_item_factory_get_widget(ifactory, "/View/All header");
1193 gtk_check_menu_item_set_active
1194 (GTK_CHECK_MENU_ITEM(menuitem),
1195 summaryview->messageview->textview->show_all_headers);
1196 summary_unlock(summaryview);
1199 void summary_select_prev_unread(SummaryView *summaryview)
1203 node = summary_find_prev_flagged_msg
1204 (summaryview, summaryview->selected, MSG_UNREAD, FALSE);
1209 switch (prefs_common.next_unread_msg_dialog) {
1210 case NEXTUNREADMSGDIALOG_ALWAYS:
1211 val = alertpanel(_("No more unread messages"),
1212 _("No unread message found. "
1213 "Search from the end?"),
1214 _("Yes"), _("No"), NULL);
1216 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1217 val = G_ALERTDEFAULT;
1219 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1220 val = !G_ALERTDEFAULT;
1224 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1226 if (val != G_ALERTDEFAULT) return;
1227 node = summary_find_prev_flagged_msg(summaryview, NULL,
1232 alertpanel_notice(_("No unread messages."));
1234 summary_select_node(summaryview, node, TRUE, FALSE);
1237 void summary_select_next_unread(SummaryView *summaryview)
1239 GtkCTreeNode *node = summaryview->selected;
1240 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1242 node = summary_find_next_flagged_msg
1243 (summaryview, node, MSG_UNREAD, FALSE);
1246 summary_select_node(summaryview, node, TRUE, FALSE);
1248 node = summary_find_next_flagged_msg
1249 (summaryview, NULL, MSG_UNREAD, FALSE);
1253 switch (prefs_common.next_unread_msg_dialog) {
1254 case NEXTUNREADMSGDIALOG_ALWAYS:
1255 val = alertpanel(_("No more unread messages"),
1256 _("No unread message found. "
1257 "Go to next folder?"),
1258 _("Yes"), _("No"), NULL);
1260 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1261 val = G_ALERTDEFAULT;
1263 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1268 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1271 if (val == G_ALERTDEFAULT) {
1272 if (gtk_signal_n_emissions_by_name
1273 (GTK_OBJECT(ctree), "key_press_event") > 0)
1274 gtk_signal_emit_stop_by_name
1277 folderview_select_next_unread(summaryview->folderview);
1283 summary_select_node(summaryview, node, TRUE, FALSE);
1288 void summary_select_prev_new(SummaryView *summaryview)
1292 node = summary_find_prev_flagged_msg
1293 (summaryview, summaryview->selected, MSG_NEW, FALSE);
1298 val = alertpanel(_("No more new messages"),
1299 _("No new message found. "
1300 "Search from the end?"),
1301 _("Yes"), _("No"), NULL);
1302 if (val != G_ALERTDEFAULT) return;
1303 node = summary_find_prev_flagged_msg(summaryview, NULL,
1308 alertpanel_notice(_("No new messages."));
1310 summary_select_node(summaryview, node, TRUE, FALSE);
1313 void summary_select_next_new(SummaryView *summaryview)
1315 GtkCTreeNode *node = summaryview->selected;
1316 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1318 while ((node = summary_find_next_flagged_msg
1319 (summaryview, node, MSG_NEW, FALSE)) == NULL) {
1322 val = alertpanel(_("No more new messages"),
1323 _("No new message found. "
1324 "Go to next folder?"),
1325 _("Yes"), _("Search again"), _("No"));
1326 if (val == G_ALERTDEFAULT) {
1327 if (gtk_signal_n_emissions_by_name
1328 (GTK_OBJECT(ctree), "key_press_event") > 0)
1329 gtk_signal_emit_stop_by_name
1332 folderview_select_next_unread(summaryview->folderview);
1334 } else if (val == G_ALERTALTERNATE)
1341 summary_select_node(summaryview, node, TRUE, FALSE);
1344 void summary_select_prev_marked(SummaryView *summaryview)
1348 node = summary_find_prev_flagged_msg
1349 (summaryview, summaryview->selected, MSG_MARKED, TRUE);
1354 val = alertpanel(_("No more marked messages"),
1355 _("No marked message found. "
1356 "Search from the end?"),
1357 _("Yes"), _("No"), NULL);
1358 if (val != G_ALERTDEFAULT) return;
1359 node = summary_find_prev_flagged_msg(summaryview, NULL,
1364 alertpanel_notice(_("No marked messages."));
1366 summary_select_node(summaryview, node, TRUE, FALSE);
1369 void summary_select_next_marked(SummaryView *summaryview)
1373 node = summary_find_next_flagged_msg
1374 (summaryview, summaryview->selected, MSG_MARKED, TRUE);
1379 val = alertpanel(_("No more marked messages"),
1380 _("No marked message found. "
1381 "Search from the beginning?"),
1382 _("Yes"), _("No"), NULL);
1383 if (val != G_ALERTDEFAULT) return;
1384 node = summary_find_next_flagged_msg(summaryview, NULL,
1389 alertpanel_notice(_("No marked messages."));
1391 summary_select_node(summaryview, node, TRUE, FALSE);
1394 void summary_select_prev_labeled(SummaryView *summaryview)
1398 node = summary_find_prev_flagged_msg
1399 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
1404 val = alertpanel(_("No more labeled messages"),
1405 _("No labeled message found. "
1406 "Search from the end?"),
1407 _("Yes"), _("No"), NULL);
1408 if (val != G_ALERTDEFAULT) return;
1409 node = summary_find_prev_flagged_msg(summaryview, NULL,
1410 MSG_CLABEL_FLAG_MASK, TRUE);
1414 alertpanel_notice(_("No labeled messages."));
1416 summary_select_node(summaryview, node, TRUE, FALSE);
1419 void summary_select_next_labeled(SummaryView *summaryview)
1423 node = summary_find_next_flagged_msg
1424 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
1429 val = alertpanel(_("No more labeled messages"),
1430 _("No labeled message found. "
1431 "Search from the beginning?"),
1432 _("Yes"), _("No"), NULL);
1433 if (val != G_ALERTDEFAULT) return;
1434 node = summary_find_next_flagged_msg(summaryview, NULL,
1435 MSG_CLABEL_FLAG_MASK, TRUE);
1439 alertpanel_notice(_("No labeled messages."));
1441 summary_select_node(summaryview, node, TRUE, FALSE);
1444 void summary_select_by_msgnum(SummaryView *summaryview, guint msgnum)
1448 node = summary_find_msg_by_msgnum(summaryview, msgnum);
1449 summary_select_node(summaryview, node, FALSE, TRUE);
1453 * summary_select_node:
1454 * @summaryview: Summary view.
1455 * @node: Summary tree node.
1456 * @display_msg: TRUE to display the selected message.
1457 * @do_refresh: TRUE to refresh the widget.
1459 * Select @node (bringing it into view by scrolling and expanding its
1460 * thread, if necessary) and unselect all others. If @display_msg is
1461 * TRUE, display the corresponding message in the message view.
1462 * If @do_refresh is TRUE, the widget is refreshed.
1464 void summary_select_node(SummaryView *summaryview, GtkCTreeNode *node,
1465 gboolean display_msg, gboolean do_refresh)
1467 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1470 gtkut_ctree_expand_parent_all(ctree, node);
1473 gtk_widget_grab_focus(GTK_WIDGET(ctree));
1474 gtk_ctree_node_moveto(ctree, node, -1, 0.5, 0);
1476 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1477 if (display_msg && summaryview->displayed == node)
1478 summaryview->displayed = NULL;
1479 summaryview->display_msg = display_msg;
1480 gtk_sctree_select(GTK_SCTREE(ctree), node);
1484 static guint summary_get_msgnum(SummaryView *summaryview, GtkCTreeNode *node)
1486 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1491 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1492 return msginfo->msgnum;
1495 static GtkCTreeNode *summary_find_prev_msg(SummaryView *summaryview,
1496 GtkCTreeNode *current_node)
1498 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1503 node = current_node;
1505 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
1507 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1508 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1509 if (!MSG_IS_DELETED(msginfo->flags)) break;
1515 static GtkCTreeNode *summary_find_next_msg(SummaryView *summaryview,
1516 GtkCTreeNode *current_node)
1518 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1523 node = current_node;
1525 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1527 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1528 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1529 if (!MSG_IS_DELETED(msginfo->flags)) break;
1535 static GtkCTreeNode *summary_find_prev_flagged_msg(SummaryView *summaryview,
1536 GtkCTreeNode *current_node,
1538 gboolean start_from_prev)
1540 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1545 if (start_from_prev)
1546 node = GTK_CTREE_NODE_PREV(current_node);
1548 node = current_node;
1550 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
1552 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1553 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1554 if ((msginfo->flags.perm_flags & flags) != 0) break;
1560 static GtkCTreeNode *summary_find_next_flagged_msg(SummaryView *summaryview,
1561 GtkCTreeNode *current_node,
1563 gboolean start_from_next)
1565 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1570 if (start_from_next)
1571 node = gtkut_ctree_node_next(ctree, current_node);
1573 node = current_node;
1575 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1577 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1578 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1579 /* Find msg with matching flags but ignore messages with
1580 ignore flags, if searching for new or unread messages */
1581 if (!(((flags & (MSG_NEW | MSG_UNREAD)) != 0) && MSG_IS_IGNORE_THREAD(msginfo->flags)) &&
1582 ((msginfo->flags.perm_flags & flags) != 0))
1589 static GtkCTreeNode *summary_find_msg_by_msgnum(SummaryView *summaryview,
1592 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1596 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1598 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1599 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1600 if (msginfo->msgnum == msgnum) break;
1606 static guint attract_hash_func(gconstpointer key)
1612 Xstrdup_a(str, (const gchar *)key, return 0);
1619 for (p += 1; *p != '\0'; p++)
1620 h = (h << 5) - h + *p;
1626 static gint attract_compare_func(gconstpointer a, gconstpointer b)
1628 return subject_compare((const gchar *)a, (const gchar *)b) == 0;
1631 void summary_attract_by_subject(SummaryView *summaryview)
1633 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1634 GtkCList *clist = GTK_CLIST(ctree);
1635 GtkCTreeNode *src_node;
1636 GtkCTreeNode *dst_node, *sibling;
1638 MsgInfo *src_msginfo, *dst_msginfo;
1639 GHashTable *subject_table;
1641 debug_print("Attracting messages by subject...");
1642 STATUSBAR_PUSH(summaryview->mainwin,
1643 _("Attracting messages by subject..."));
1645 main_window_cursor_wait(summaryview->mainwin);
1646 gtk_clist_freeze(clist);
1648 subject_table = g_hash_table_new(attract_hash_func,
1649 attract_compare_func);
1651 for (src_node = GTK_CTREE_NODE(clist->row_list);
1654 tmp = GTK_CTREE_ROW(src_node)->sibling;
1655 src_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(src_node);
1656 if (!src_msginfo) continue;
1657 if (!src_msginfo->subject) continue;
1659 /* find attracting node */
1660 dst_node = g_hash_table_lookup(subject_table,
1661 src_msginfo->subject);
1664 dst_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(dst_node);
1666 /* if the time difference is more than 20 days,
1668 if (ABS(src_msginfo->date_t - dst_msginfo->date_t)
1669 > 60 * 60 * 24 * 20)
1672 sibling = GTK_CTREE_ROW(dst_node)->sibling;
1673 if (src_node != sibling)
1674 gtk_ctree_move(ctree, src_node, NULL, sibling);
1677 g_hash_table_insert(subject_table,
1678 src_msginfo->subject, src_node);
1681 g_hash_table_destroy(subject_table);
1683 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
1685 gtk_clist_thaw(clist);
1687 debug_print("done.\n");
1688 STATUSBAR_POP(summaryview->mainwin);
1690 main_window_cursor_normal(summaryview->mainwin);
1693 static void summary_free_msginfo_func(GtkCTree *ctree, GtkCTreeNode *node,
1696 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, node);
1699 procmsg_msginfo_free(msginfo);
1702 static void summary_set_marks_func(GtkCTree *ctree, GtkCTreeNode *node,
1705 SummaryView *summaryview = data;
1708 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1710 if (MSG_IS_NEWS(msginfo->flags))
1711 news_flag_crosspost(msginfo);
1713 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1714 summaryview->newmsgs++;
1715 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1716 summaryview->unread++;
1717 if (MSG_IS_DELETED(msginfo->flags))
1718 summaryview->deleted++;
1720 summaryview->messages++;
1721 summaryview->total_size += msginfo->size;
1723 summary_set_row_marks(summaryview, node);
1726 static void summary_update_status(SummaryView *summaryview)
1728 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1732 summaryview->newmsgs = summaryview->unread =
1733 summaryview->messages = summaryview->total_size =
1734 summaryview->deleted = summaryview->moved = summaryview->copied = 0;
1736 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1737 node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1738 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
1740 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1741 summaryview->newmsgs++;
1742 if (MSG_IS_UNREAD(msginfo->flags)&& !MSG_IS_IGNORE_THREAD(msginfo->flags))
1743 summaryview->unread++;
1744 if (MSG_IS_DELETED(msginfo->flags))
1745 summaryview->deleted++;
1746 if (MSG_IS_MOVE(msginfo->flags))
1747 summaryview->moved++;
1748 if (MSG_IS_COPY(msginfo->flags))
1749 summaryview->copied++;
1750 summaryview->messages++;
1751 summaryview->total_size += msginfo->size;
1755 static void summary_status_show(SummaryView *summaryview)
1758 gchar *del, *mv, *cp;
1762 GList *rowlist, *cur;
1763 guint n_selected = 0;
1767 if (!summaryview->folder_item) {
1768 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder), "");
1769 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), "");
1770 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), "");
1774 rowlist = GTK_CLIST(summaryview->ctree)->selection;
1775 for (cur = rowlist; cur != NULL; cur = cur->next) {
1776 msginfo = gtk_ctree_node_get_row_data
1777 (GTK_CTREE(summaryview->ctree),
1778 GTK_CTREE_NODE(cur->data));
1779 sel_size += msginfo->size;
1783 if (summaryview->folder_item->folder->type == F_NEWS &&
1784 prefs_common.ng_abbrev_len < strlen(summaryview->folder_item->path)) {
1786 group = get_abbrev_newsgroup_name
1787 (g_basename(summaryview->folder_item->path));
1788 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder), group);
1791 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder),
1792 summaryview->folder_item->path);
1795 if (summaryview->deleted)
1796 del = g_strdup_printf(_("%d deleted"), summaryview->deleted);
1799 if (summaryview->moved)
1800 mv = g_strdup_printf(_("%s%d moved"),
1801 summaryview->deleted ? _(", ") : "",
1802 summaryview->moved);
1805 if (summaryview->copied)
1806 cp = g_strdup_printf(_("%s%d copied"),
1807 summaryview->deleted ||
1808 summaryview->moved ? _(", ") : "",
1809 summaryview->copied);
1813 if (summaryview->deleted || summaryview->moved || summaryview->copied)
1819 sel = g_strdup_printf(" (%s)", to_human_readable(sel_size));
1820 if (n_selected == 1)
1821 itstr = g_strdup(_(" item selected"));
1823 itstr = g_strdup(_(" items selected"));
1826 itstr = g_strdup("");
1829 str = g_strconcat(n_selected ? itos(n_selected) : "",
1830 itstr, sel, spc, del, mv, cp, NULL);
1831 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), str);
1839 if (FOLDER_IS_LOCAL(summaryview->folder_item->folder)) {
1840 str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
1841 summaryview->newmsgs,
1842 summaryview->unread,
1843 summaryview->messages,
1844 to_human_readable(summaryview->total_size));
1846 str = g_strdup_printf(_("%d new, %d unread, %d total"),
1847 summaryview->newmsgs,
1848 summaryview->unread,
1849 summaryview->messages);
1851 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), str);
1855 static void summary_set_column_titles(SummaryView *summaryview)
1857 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1863 SummaryColumnType type;
1864 gboolean single_char;
1865 GtkJustification justify;
1867 static FolderSortKey sort_by[N_SUMMARY_COLS] = {
1880 for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
1881 type = summaryview->col_state[pos].type;
1883 /* CLAWS: mime and unread are single char headers */
1884 single_char = (type == S_COL_MIME || type == S_COL_UNREAD);
1885 justify = (type == S_COL_NUMBER || type == S_COL_SIZE)
1886 ? GTK_JUSTIFY_RIGHT : GTK_JUSTIFY_LEFT;
1893 if (prefs_common.trans_hdr)
1894 title = gettext(col_label[type]);
1896 title = col_label[type];
1898 /* CLAWS: dummies for mark and locked headers */
1904 title = gettext(col_label[type]);
1907 if (type == S_COL_MIME) {
1908 label = gtk_pixmap_new(clipxpm, clipxpmmask);
1909 gtk_widget_show(label);
1910 gtk_clist_set_column_widget(clist, pos, label);
1914 gtk_clist_set_column_title(clist, pos, title);
1918 /* CLAWS: changed so that locked and mark headers
1919 * show a pixmap instead of single character */
1920 hbox = gtk_hbox_new(FALSE, 4);
1922 if (type == S_COL_LOCKED)
1923 label = gtk_pixmap_new(lockedxpm, lockedxpmmask);
1924 else if (type == S_COL_MARK)
1925 label = gtk_pixmap_new(markxpm, markxpmmask);
1927 label = gtk_label_new(title);
1929 if (justify == GTK_JUSTIFY_RIGHT)
1930 gtk_box_pack_end(GTK_BOX(hbox), label,
1933 gtk_box_pack_start(GTK_BOX(hbox), label,
1936 if (summaryview->sort_key == sort_by[type]) {
1937 arrow = gtk_arrow_new
1938 (summaryview->sort_type == SORT_ASCENDING
1939 ? GTK_ARROW_DOWN : GTK_ARROW_UP,
1941 if (justify == GTK_JUSTIFY_RIGHT)
1942 gtk_box_pack_start(GTK_BOX(hbox), arrow,
1945 gtk_box_pack_end(GTK_BOX(hbox), arrow,
1949 gtk_widget_show_all(hbox);
1950 gtk_clist_set_column_widget(clist, pos, hbox);
1954 void summary_sort(SummaryView *summaryview,
1955 FolderSortKey sort_key, FolderSortType sort_type)
1957 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1958 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1959 GtkCListCompareFunc cmp_func = NULL;
1963 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mark;
1965 case SORT_BY_UNREAD:
1966 cmp_func = (GtkCListCompareFunc)summary_cmp_by_unread;
1969 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mime;
1971 case SORT_BY_NUMBER:
1972 cmp_func = (GtkCListCompareFunc)summary_cmp_by_num;
1975 cmp_func = (GtkCListCompareFunc)summary_cmp_by_size;
1978 cmp_func = (GtkCListCompareFunc)summary_cmp_by_date;
1981 cmp_func = summaryview->folder_item->stype != F_OUTBOX ?
1982 (GtkCListCompareFunc) summary_cmp_by_from :
1983 (GtkCListCompareFunc) summary_cmp_by_to;
1985 case SORT_BY_SUBJECT:
1986 if (summaryview->simplify_subject_preg)
1987 cmp_func = (GtkCListCompareFunc)summary_cmp_by_simplified_subject;
1989 cmp_func = (GtkCListCompareFunc)summary_cmp_by_subject;
1992 cmp_func = (GtkCListCompareFunc)summary_cmp_by_score;
1994 case SORT_BY_LOCKED:
1995 cmp_func = (GtkCListCompareFunc)summary_cmp_by_locked;
1998 cmp_func = (GtkCListCompareFunc)summary_cmp_by_label;
2007 summaryview->sort_key = sort_key;
2008 summaryview->sort_type = sort_type;
2010 summary_set_column_titles(summaryview);
2011 summary_set_menu_sensitive(summaryview);
2012 if(cmp_func != NULL) {
2013 debug_print("Sorting summary...");
2014 STATUSBAR_PUSH(summaryview->mainwin, _("Sorting summary..."));
2016 main_window_cursor_wait(summaryview->mainwin);
2018 gtk_clist_set_compare_func(clist, cmp_func);
2020 gtk_clist_set_sort_type(clist, (GtkSortType)sort_type);
2022 gtk_sctree_sort_recursive(ctree, NULL);
2024 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
2026 main_window_cursor_normal(summaryview->mainwin);
2028 debug_print("done.\n");
2029 STATUSBAR_POP(summaryview->mainwin);
2033 gboolean summary_insert_gnode_func(GtkCTree *ctree, guint depth, GNode *gnode,
2034 GtkCTreeNode *cnode, gpointer data)
2036 SummaryView *summaryview = (SummaryView *)data;
2037 MsgInfo *msginfo = (MsgInfo *)gnode->data;
2038 gchar *text[N_SUMMARY_COLS];
2039 gint *col_pos = summaryview->col_pos;
2040 const gchar *msgid = msginfo->msgid;
2041 GHashTable *msgid_table = summaryview->msgid_table;
2043 summary_set_header(summaryview, text, msginfo);
2045 gtk_ctree_set_node_info(ctree, cnode, text[col_pos[S_COL_SUBJECT]], 2,
2046 NULL, NULL, NULL, NULL, FALSE,
2047 gnode->parent->parent ? TRUE : FALSE);
2048 #define SET_TEXT(col) \
2049 gtk_ctree_node_set_text(ctree, cnode, col_pos[col], \
2052 SET_TEXT(S_COL_NUMBER);
2053 SET_TEXT(S_COL_SCORE);
2054 SET_TEXT(S_COL_SIZE);
2055 SET_TEXT(S_COL_DATE);
2056 SET_TEXT(S_COL_FROM);
2057 SET_TEXT(S_COL_SUBJECT);
2061 GTKUT_CTREE_NODE_SET_ROW_DATA(cnode, msginfo);
2062 summary_set_marks_func(ctree, cnode, summaryview);
2064 if (msgid && msgid[0] != '\0')
2065 g_hash_table_insert(msgid_table, (gchar *)msgid, cnode);
2070 static void summary_set_ctree_from_list(SummaryView *summaryview,
2073 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2075 GtkCTreeNode *node = NULL;
2076 GHashTable *msgid_table;
2077 GHashTable *subject_table;
2082 debug_print("\tSetting summary from message data...");
2083 STATUSBAR_PUSH(summaryview->mainwin,
2084 _("Setting summary from message data..."));
2087 msgid_table = g_hash_table_new(g_str_hash, g_str_equal);
2088 summaryview->msgid_table = msgid_table;
2089 subject_table = g_hash_table_new(g_str_hash, g_str_equal);
2090 summaryview->subject_table = subject_table;
2092 if (prefs_common.use_addr_book)
2093 start_address_completion();
2095 for (cur = mlist ; cur != NULL; cur = cur->next) {
2096 msginfo = (MsgInfo *)cur->data;
2097 msginfo->threadscore = msginfo->score;
2100 if (global_scoring || summaryview->folder_item->prefs->scoring) {
2101 summaryview->important_score = prefs_common.important_score;
2102 if (summaryview->folder_item->prefs->important_score >
2103 summaryview->important_score)
2104 summaryview->important_score =
2105 summaryview->folder_item->prefs->important_score;
2108 if (summaryview->threaded) {
2109 GNode *root, *gnode;
2111 root = procmsg_get_thread_tree(mlist);
2113 for (gnode = root->children; gnode != NULL;
2114 gnode = gnode->next) {
2115 node = gtk_ctree_insert_gnode
2116 (ctree, NULL, node, gnode,
2117 summary_insert_gnode_func, summaryview);
2120 g_node_destroy(root);
2122 summary_thread_init(summaryview);
2124 gchar *text[N_SUMMARY_COLS];
2126 mlist = g_slist_reverse(mlist);
2127 for (; mlist != NULL; mlist = mlist->next) {
2128 msginfo = (MsgInfo *)mlist->data;
2130 summary_set_header(summaryview, text, msginfo);
2132 node = gtk_ctree_insert_node
2133 (ctree, NULL, node, text, 2,
2134 NULL, NULL, NULL, NULL, FALSE, FALSE);
2135 GTKUT_CTREE_NODE_SET_ROW_DATA(node, msginfo);
2136 summary_set_marks_func(ctree, node, summaryview);
2138 if (msginfo->msgid && msginfo->msgid[0] != '\0')
2139 g_hash_table_insert(msgid_table,
2140 msginfo->msgid, node);
2142 subject_table_insert(subject_table,
2146 mlist = g_slist_reverse(mlist);
2149 if (prefs_common.enable_hscrollbar &&
2150 summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
2153 optimal_width = gtk_clist_optimal_column_width
2154 (GTK_CLIST(ctree), summaryview->col_pos[S_COL_SUBJECT]);
2155 gtk_clist_set_column_width(GTK_CLIST(ctree),
2156 summaryview->col_pos[S_COL_SUBJECT],
2160 if (prefs_common.use_addr_book)
2161 end_address_completion();
2163 debug_print("done.\n");
2164 STATUSBAR_POP(summaryview->mainwin);
2166 debug_print("\tmsgid hash table size = %d\n",
2167 g_hash_table_size(msgid_table));
2168 debug_print("\tsubject hash table size = %d\n",
2169 g_hash_table_size(subject_table));
2173 static gchar *summary_complete_address(const gchar *addr)
2176 gchar *res, *tmp, *email_addr;
2178 Xstrdup_a(email_addr, addr, return NULL);
2179 extract_address(email_addr);
2180 g_return_val_if_fail(*email_addr, NULL);
2183 * completion stuff must be already initialized
2186 if (1 < (count = complete_address(email_addr))) {
2187 tmp = get_complete_address(1);
2188 res = procheader_get_fromname(tmp);
2195 static void summary_set_header(SummaryView *summaryview, gchar *text[],
2198 static gchar date_modified[80];
2199 static gchar *to = NULL;
2200 static gchar col_score[11];
2201 static gchar buf[BUFFSIZE];
2202 gint *col_pos = summaryview->col_pos;
2204 text[col_pos[S_COL_MARK]] = NULL;
2205 text[col_pos[S_COL_UNREAD]] = NULL;
2206 text[col_pos[S_COL_MIME]] = NULL;
2207 text[col_pos[S_COL_LOCKED]] = NULL;
2208 text[col_pos[S_COL_NUMBER]] = itos(msginfo->msgnum);
2209 text[col_pos[S_COL_SIZE]] = to_human_readable(msginfo->size);
2210 text[col_pos[S_COL_SCORE]] = itos_buf(col_score, msginfo->score);
2212 if (msginfo->date_t) {
2213 procheader_date_get_localtime(date_modified,
2214 sizeof(date_modified),
2216 text[col_pos[S_COL_DATE]] = date_modified;
2217 } else if (msginfo->date)
2218 text[col_pos[S_COL_DATE]] = msginfo->date;
2220 text[col_pos[S_COL_DATE]] = _("(No Date)");
2222 text[col_pos[S_COL_FROM]] = msginfo->fromname ? msginfo->fromname :
2224 if (prefs_common.swap_from && msginfo->from && msginfo->to &&
2225 !MSG_IS_NEWS(msginfo->flags)) {
2228 Xstrdup_a(addr, msginfo->from, return);
2229 extract_address(addr);
2231 if (prefs_common.use_addr_book) {
2232 if (account_find_from_address(addr)) {
2233 addr = summary_complete_address(msginfo->to);
2235 to = g_strconcat("-->", addr == NULL ? msginfo->to : addr, NULL);
2236 text[col_pos[S_COL_FROM]] = to;
2240 if (cur_account && cur_account->address && !strcmp( addr, cur_account->address)) {
2242 to = g_strconcat("-->", msginfo->to, NULL);
2243 text[col_pos[S_COL_FROM]] = to;
2249 * CLAWS: note that the "text[col_pos[S_COL_FROM]] != to" is really a hack,
2250 * checking whether the above block (which handles the special case of
2251 * the --> in sent boxes) was executed.
2253 if (text[col_pos[S_COL_FROM]] != to && prefs_common.use_addr_book && msginfo->from) {
2254 gchar *from = summary_complete_address(msginfo->from);
2258 text[col_pos[S_COL_FROM]] = to;
2262 if (summaryview->simplify_subject_preg != NULL)
2263 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ?
2264 string_remove_match(buf, BUFFSIZE, msginfo->subject,
2265 summaryview->simplify_subject_preg) :
2268 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ? msginfo->subject :
2272 #define CHANGE_FLAGS(msginfo) \
2274 if (msginfo->folder->folder->change_flags != NULL) \
2275 msginfo->folder->folder->change_flags(msginfo->folder->folder, \
2280 static void summary_display_msg(SummaryView *summaryview, GtkCTreeNode *row)
2282 summary_display_msg_full(summaryview, row, FALSE, FALSE);
2285 static void summary_display_msg_full(SummaryView *summaryview,
2287 gboolean new_window, gboolean all_headers)
2289 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2294 if (!new_window && summaryview->displayed == row) return;
2295 g_return_if_fail(row != NULL);
2297 if (summary_is_locked(summaryview)) return;
2298 summary_lock(summaryview);
2300 STATUSBAR_POP(summaryview->mainwin);
2303 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2305 filename = procmsg_get_message_file(msginfo);
2307 summary_unlock(summaryview);
2312 if (new_window || !prefs_common.mark_as_read_on_new_window) {
2313 if (MSG_IS_NEW(msginfo->flags) && MSG_IS_IGNORE_THREAD(msginfo->flags))
2314 summaryview->newmsgs--;
2315 if (MSG_IS_UNREAD(msginfo->flags) && MSG_IS_IGNORE_THREAD(msginfo->flags))
2316 summaryview->unread--;
2317 if (MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)) {
2318 procmsg_msginfo_unset_flags
2319 (msginfo, MSG_NEW | MSG_UNREAD, 0);
2320 summary_set_row_marks(summaryview, row);
2321 gtk_clist_thaw(GTK_CLIST(ctree));
2322 summary_status_show(summaryview);
2324 flags = msginfo->flags;
2329 MessageView *msgview;
2331 msgview = messageview_create_with_new_window(summaryview->mainwin);
2332 messageview_show(msgview, msginfo, all_headers);
2334 MessageView *msgview;
2336 msgview = summaryview->messageview;
2338 summaryview->displayed = row;
2339 if (!messageview_is_visible(msgview))
2340 main_window_toggle_message_view(summaryview->mainwin);
2341 messageview_show(msgview, msginfo, all_headers);
2342 if (msgview->type == MVIEW_TEXT ||
2343 (msgview->type == MVIEW_MIME &&
2344 (GTK_CLIST(msgview->mimeview->ctree)->row_list == NULL ||
2345 gtk_notebook_get_current_page
2346 (GTK_NOTEBOOK(msgview->mimeview->notebook)) == 0)))
2347 gtk_widget_grab_focus(summaryview->ctree);
2349 gtkut_ctree_node_move_if_on_the_edge(ctree, row);
2352 summary_set_menu_sensitive(summaryview);
2353 toolbar_set_sensitive(summaryview->mainwin);
2355 summary_unlock(summaryview);
2358 void summary_display_msg_selected(SummaryView *summaryview,
2359 gboolean all_headers)
2361 if (summary_is_locked(summaryview)) return;
2362 summaryview->displayed = NULL;
2363 summary_display_msg_full(summaryview, summaryview->selected, FALSE,
2367 void summary_redisplay_msg(SummaryView *summaryview)
2371 if (summaryview->displayed) {
2372 node = summaryview->displayed;
2373 summaryview->displayed = NULL;
2374 summary_display_msg(summaryview, node);
2378 void summary_open_msg(SummaryView *summaryview)
2380 if (!summaryview->selected) return;
2382 summary_display_msg_full(summaryview, summaryview->selected,
2386 void summary_view_source(SummaryView * summaryview)
2388 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2390 SourceWindow *srcwin;
2392 if (!summaryview->selected) return;
2394 srcwin = source_window_create();
2395 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
2396 source_window_show_msg(srcwin, msginfo);
2397 source_window_show(srcwin);
2400 void summary_reedit(SummaryView *summaryview)
2404 if (!summaryview->selected) return;
2405 if (!summaryview->folder_item) return;
2406 if (summaryview->folder_item->stype != F_OUTBOX &&
2407 summaryview->folder_item->stype != F_DRAFT &&
2408 summaryview->folder_item->stype != F_QUEUE) return;
2410 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
2411 summaryview->selected);
2412 if (!msginfo) return;
2414 compose_reedit(msginfo);
2417 void summary_step(SummaryView *summaryview, GtkScrollType type)
2419 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2422 if (summary_is_locked(summaryview)) return;
2424 if (type == GTK_SCROLL_STEP_FORWARD) {
2425 node = gtkut_ctree_node_next(ctree, summaryview->selected);
2427 gtkut_ctree_expand_parent_all(ctree, node);
2431 if (summaryview->selected) {
2432 node = GTK_CTREE_NODE_PREV(summaryview->selected);
2437 if (messageview_is_visible(summaryview->messageview))
2438 summaryview->display_msg = TRUE;
2440 gtk_signal_emit_by_name(GTK_OBJECT(ctree), "scroll_vertical",
2444 void summary_toggle_view(SummaryView *summaryview)
2446 if (!messageview_is_visible(summaryview->messageview) &&
2447 summaryview->selected)
2448 summary_display_msg(summaryview,
2449 summaryview->selected);
2451 main_window_toggle_message_view(summaryview->mainwin);
2454 static gboolean summary_search_unread_recursive(GtkCTree *ctree,
2460 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2461 if (msginfo && MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2463 node = GTK_CTREE_ROW(node)->children;
2465 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2468 if (summary_search_unread_recursive(ctree, node) == TRUE)
2470 node = GTK_CTREE_ROW(node)->sibling;
2476 static gboolean summary_have_unread_children(SummaryView *summaryview,
2479 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2481 if (!node) return FALSE;
2483 node = GTK_CTREE_ROW(node)->children;
2486 if (summary_search_unread_recursive(ctree, node) == TRUE)
2488 node = GTK_CTREE_ROW(node)->sibling;
2494 static void summary_set_row_marks(SummaryView *summaryview, GtkCTreeNode *row)
2496 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2497 GtkStyle *style = NULL;
2500 gint *col_pos = summaryview->col_pos;
2502 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2503 if (!msginfo) return;
2505 flags = msginfo->flags;
2507 gtk_ctree_node_set_foreground(ctree, row, NULL);
2509 /* set new/unread column */
2510 if (MSG_IS_IGNORE_THREAD(flags)) {
2511 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_UNREAD],
2512 ignorethreadxpm, ignorethreadxpmmask);
2513 } else if (MSG_IS_NEW(flags)) {
2514 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_UNREAD],
2515 newxpm, newxpmmask);
2516 } else if (MSG_IS_UNREAD(flags)) {
2517 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_UNREAD],
2518 unreadxpm, unreadxpmmask);
2519 } else if (MSG_IS_REPLIED(flags)) {
2520 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_UNREAD],
2521 repliedxpm, repliedxpmmask);
2522 } else if (MSG_IS_FORWARDED(flags)) {
2523 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_UNREAD],
2524 forwardedxpm, forwardedxpmmask);
2526 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_UNREAD],
2530 if (prefs_common.bold_unread &&
2531 ((MSG_IS_UNREAD(flags) && !MSG_IS_IGNORE_THREAD(flags)) ||
2532 (!GTK_CTREE_ROW(row)->expanded &&
2533 GTK_CTREE_ROW(row)->children &&
2534 summary_have_unread_children(summaryview, row))))
2537 /* set mark column */
2538 if (MSG_IS_DELETED(flags)) {
2539 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
2540 deletedxpm, deletedxpmmask);
2542 style = bold_deleted_style;
2544 style = small_deleted_style;
2546 gtk_ctree_node_set_foreground
2547 (ctree, row, &summaryview->color_dim);
2548 } else if (MSG_IS_MARKED(flags)) {
2549 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
2550 markxpm, markxpmmask);
2551 } else if (MSG_IS_MOVE(flags)) {
2552 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], "o");
2554 style = bold_marked_style;
2556 style = small_marked_style;
2558 gtk_ctree_node_set_foreground
2559 (ctree, row, &summaryview->color_marked);
2560 } else if (MSG_IS_COPY(flags)) {
2561 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], "O");
2563 style = bold_marked_style;
2565 style = small_marked_style;
2567 gtk_ctree_node_set_foreground
2568 (ctree, row, &summaryview->color_marked);
2570 else if ((global_scoring ||
2571 summaryview->folder_item->prefs->scoring) &&
2572 (msginfo->score >= summaryview->important_score) &&
2573 (MSG_IS_MARKED(msginfo->flags) || MSG_IS_MOVE(msginfo->flags) || MSG_IS_COPY(msginfo->flags))) {
2574 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, "!");
2575 gtk_ctree_node_set_foreground(ctree, row,
2576 &summaryview->color_important);
2578 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], NULL);
2581 if (MSG_IS_LOCKED(flags)) {
2582 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_LOCKED],
2583 lockedxpm, lockedxpmmask);
2586 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_LOCKED], NULL);
2589 if (MSG_IS_MIME(flags) && MSG_IS_ENCRYPTED(flags)) {
2590 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2591 clipkeyxpm, clipkeyxpmmask);
2592 } else if (MSG_IS_ENCRYPTED(flags)) {
2593 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2594 keyxpm, keyxpmmask);
2595 } else if (MSG_IS_MIME(flags)) {
2596 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2597 clipxpm, clipxpmmask);
2599 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MIME], NULL);
2602 style = small_style;
2604 gtk_ctree_node_set_row_style(ctree, row, style);
2606 if (MSG_GET_COLORLABEL(flags))
2607 summary_set_colorlabel_color(ctree, row, MSG_GET_COLORLABEL_VALUE(flags));
2610 void summary_set_marks_selected(SummaryView *summaryview)
2614 for (cur = GTK_CLIST(summaryview->ctree)->selection; cur != NULL;
2616 summary_set_row_marks(summaryview, GTK_CTREE_NODE(cur->data));
2619 static void summary_mark_row(SummaryView *summaryview, GtkCTreeNode *row)
2621 gboolean changed = FALSE;
2622 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2625 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2626 if (MSG_IS_DELETED(msginfo->flags))
2627 summaryview->deleted--;
2628 if (MSG_IS_MOVE(msginfo->flags)) {
2629 summaryview->moved--;
2632 if (MSG_IS_COPY(msginfo->flags)) {
2633 summaryview->copied--;
2636 if (changed && !prefs_common.immediate_exec) {
2637 msginfo->to_folder->op_count--;
2638 if (msginfo->to_folder->op_count == 0)
2639 folderview_update_item(msginfo->to_folder, 0);
2641 msginfo->to_folder = NULL;
2642 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_MOVE | MSG_COPY);
2643 procmsg_msginfo_set_flags(msginfo, MSG_MARKED, 0);
2644 summary_set_row_marks(summaryview, row);
2645 debug_print("Message %s/%d is marked\n", msginfo->folder->path, msginfo->msgnum);
2648 static void summary_lock_row(SummaryView *summaryview, GtkCTreeNode *row)
2650 /* almost verbatim summary_mark_row(); may want a menu action? */
2651 gboolean changed = FALSE;
2652 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2655 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2656 if (MSG_IS_DELETED(msginfo->flags))
2657 summaryview->deleted--;
2658 if (MSG_IS_MOVE(msginfo->flags)) {
2659 summaryview->moved--;
2662 if (MSG_IS_COPY(msginfo->flags)) {
2663 summaryview->copied--;
2666 if (changed && !prefs_common.immediate_exec) {
2667 msginfo->to_folder->op_count--;
2668 if (msginfo->to_folder->op_count == 0)
2669 folderview_update_item(msginfo->to_folder, 0);
2671 msginfo->to_folder = NULL;
2672 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_MOVE | MSG_COPY);
2673 procmsg_msginfo_set_flags(msginfo, MSG_LOCKED, 0);
2674 summary_set_row_marks(summaryview, row);
2675 debug_print("Message %d is locked\n", msginfo->msgnum);
2678 void summary_mark(SummaryView *summaryview)
2680 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2683 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2684 summary_mark_row(summaryview, GTK_CTREE_NODE(cur->data));
2686 /* summary_step(summaryview, GTK_SCROLL_STEP_FORWARD); */
2687 summary_status_show(summaryview);
2690 static void summary_mark_row_as_read(SummaryView *summaryview,
2693 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2696 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2698 if(!MSG_IS_NEW(msginfo->flags) && !MSG_IS_UNREAD(msginfo->flags))
2701 if (MSG_IS_NEW(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2702 summaryview->newmsgs--;
2703 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2704 summaryview->unread--;
2706 procmsg_msginfo_unset_flags(msginfo, MSG_NEW | MSG_UNREAD, 0);
2707 summary_set_row_marks(summaryview, row);
2708 debug_print("Message %d is marked as read\n",
2712 void summary_mark_as_read(SummaryView *summaryview)
2714 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2717 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2718 summary_mark_row_as_read(summaryview,
2719 GTK_CTREE_NODE(cur->data));
2721 summary_status_show(summaryview);
2724 void summary_mark_all_read(SummaryView *summaryview)
2726 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2727 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2730 gtk_clist_freeze(clist);
2731 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
2732 node = gtkut_ctree_node_next(ctree, node))
2733 summary_mark_row_as_read(summaryview, node);
2734 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
2735 node = gtkut_ctree_node_next(ctree, node)) {
2736 if (!GTK_CTREE_ROW(node)->expanded)
2737 summary_set_row_marks(summaryview, node);
2739 gtk_clist_thaw(clist);
2741 summary_status_show(summaryview);
2744 static void summary_mark_row_as_unread(SummaryView *summaryview,
2747 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2750 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2751 if (MSG_IS_DELETED(msginfo->flags)) {
2752 msginfo->to_folder = NULL;
2753 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, 0);
2754 summaryview->deleted--;
2757 if (!MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2758 summaryview->unread++;
2760 procmsg_msginfo_unset_flags(msginfo, MSG_REPLIED | MSG_FORWARDED, 0);
2761 procmsg_msginfo_set_flags(msginfo, MSG_UNREAD, 0);
2762 debug_print("Message %d is marked as unread\n",
2765 summary_set_row_marks(summaryview, row);
2768 void summary_mark_as_unread(SummaryView *summaryview)
2770 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2773 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2774 summary_mark_row_as_unread(summaryview,
2775 GTK_CTREE_NODE(cur->data));
2777 summary_status_show(summaryview);
2780 static gboolean check_permission(SummaryView *summaryview, MsgInfo * msginfo)
2785 switch (summaryview->folder_item->folder->type) {
2790 security : checks if one the accounts correspond to
2791 the author of the post
2795 for(cur = account_get_list() ; cur != NULL ; cur = cur->next) {
2796 PrefsAccount * account;
2799 account = cur->data;
2800 if (account->name && *account->name)
2802 g_strdup_printf("%s <%s>",
2807 g_strdup_printf("%s",
2810 if (g_strcasecmp(from_name, msginfo->from) == 0) {
2819 alertpanel_error(_("You're not the author of the article\n"));
2829 static void summary_delete_row(SummaryView *summaryview, GtkCTreeNode *row)
2831 gboolean changed = FALSE;
2832 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2835 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2837 if (!check_permission(summaryview, msginfo))
2840 if (MSG_IS_LOCKED(msginfo->flags)) return;
2842 if (MSG_IS_DELETED(msginfo->flags)) return;
2844 if (MSG_IS_MOVE(msginfo->flags)) {
2845 summaryview->moved--;
2848 if (MSG_IS_COPY(msginfo->flags)) {
2849 summaryview->copied--;
2852 if (changed && !prefs_common.immediate_exec) {
2853 msginfo->to_folder->op_count--;
2854 if (msginfo->to_folder->op_count == 0)
2855 folderview_update_item(msginfo->to_folder, 0);
2857 msginfo->to_folder = NULL;
2858 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED, MSG_MOVE | MSG_COPY);
2859 procmsg_msginfo_set_flags(msginfo, MSG_DELETED, 0);
2860 summaryview->deleted++;
2862 if (!prefs_common.immediate_exec &&
2863 summaryview->folder_item->stype != F_TRASH)
2864 summary_set_row_marks(summaryview, row);
2866 debug_print("Message %s/%d is set to delete\n",
2867 msginfo->folder->path, msginfo->msgnum);
2870 void summary_cancel(SummaryView *summaryview)
2873 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2875 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
2876 summaryview->selected);
2877 if (!msginfo) return;
2879 if (!check_permission(summaryview, msginfo))
2882 news_cancel_article(summaryview->folder_item->folder, msginfo);
2884 if (summary_is_locked(summaryview)) return;
2886 summary_lock(summaryview);
2888 gtk_clist_freeze(clist);
2890 summary_update_status(summaryview);
2891 summary_status_show(summaryview);
2893 gtk_clist_thaw(clist);
2895 summary_unlock(summaryview);
2898 void summary_delete(SummaryView *summaryview)
2900 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2901 FolderItem *item = summaryview->folder_item;
2903 GtkCTreeNode *sel_last = NULL;
2908 if (!item || item->folder->type == F_NEWS) return;
2911 if (summary_is_locked(summaryview)) return;
2913 /* if current folder is trash, ask for confirmation */
2914 if (item->stype == F_TRASH) {
2917 aval = alertpanel(_("Delete message(s)"),
2918 _("Do you really want to delete message(s) from the trash?"),
2919 _("Yes"), _("No"), NULL);
2920 if (aval != G_ALERTDEFAULT) return;
2923 /* next code sets current row focus right. We need to find a row
2924 * that is not deleted. */
2925 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next) {
2926 sel_last = GTK_CTREE_NODE(cur->data);
2927 summary_delete_row(summaryview, sel_last);
2930 node = summary_find_next_msg(summaryview, sel_last);
2932 node = summary_find_prev_msg(summaryview, sel_last);
2935 if (sel_last && node == gtkut_ctree_node_next(ctree, sel_last))
2936 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
2937 else if (sel_last && node == GTK_CTREE_NODE_PREV(sel_last))
2938 summary_step(summaryview, GTK_SCROLL_STEP_BACKWARD);
2942 messageview_is_visible(summaryview->messageview),
2946 if (prefs_common.immediate_exec || item->stype == F_TRASH)
2947 summary_execute(summaryview);
2949 summary_status_show(summaryview);
2952 void summary_delete_duplicated(SummaryView *summaryview)
2954 if (!summaryview->folder_item ||
2955 summaryview->folder_item->folder->type == F_NEWS) return;
2956 if (summaryview->folder_item->stype == F_TRASH) return;
2958 main_window_cursor_wait(summaryview->mainwin);
2959 debug_print("Deleting duplicated messages...");
2960 STATUSBAR_PUSH(summaryview->mainwin,
2961 _("Deleting duplicated messages..."));
2963 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree), NULL,
2964 GTK_CTREE_FUNC(summary_delete_duplicated_func),
2967 if (prefs_common.immediate_exec)
2968 summary_execute(summaryview);
2970 summary_status_show(summaryview);
2972 debug_print("done.\n");
2973 STATUSBAR_POP(summaryview->mainwin);
2974 main_window_cursor_normal(summaryview->mainwin);
2977 static void summary_delete_duplicated_func(GtkCTree *ctree, GtkCTreeNode *node,
2978 SummaryView *summaryview)
2980 GtkCTreeNode *found;
2981 MsgInfo *msginfo = GTK_CTREE_ROW(node)->row.data;
2983 if (!msginfo->msgid || !*msginfo->msgid) return;
2985 found = g_hash_table_lookup(summaryview->msgid_table, msginfo->msgid);
2987 if (found && found != node)
2988 summary_delete_row(summaryview, node);
2991 static void summary_unmark_row(SummaryView *summaryview, GtkCTreeNode *row)
2993 gboolean changed = FALSE;
2994 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2997 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2998 if (MSG_IS_DELETED(msginfo->flags))
2999 summaryview->deleted--;
3000 if (MSG_IS_MOVE(msginfo->flags)) {
3001 summaryview->moved--;
3004 if (MSG_IS_COPY(msginfo->flags)) {
3005 summaryview->copied--;
3008 if (changed && !prefs_common.immediate_exec) {
3009 msginfo->to_folder->op_count--;
3010 if (msginfo->to_folder->op_count == 0)
3011 folderview_update_item(msginfo->to_folder, 0);
3013 msginfo->to_folder = NULL;
3014 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_MOVE | MSG_COPY);
3015 summary_set_row_marks(summaryview, row);
3017 debug_print("Message %s/%d is unmarked\n",
3018 msginfo->folder->path, msginfo->msgnum);
3021 void summary_unmark(SummaryView *summaryview)
3023 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3026 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
3027 summary_unmark_row(summaryview, GTK_CTREE_NODE(cur->data));
3029 summary_status_show(summaryview);
3032 static void summary_move_row_to(SummaryView *summaryview, GtkCTreeNode *row,
3033 FolderItem *to_folder)
3035 gboolean changed = FALSE;
3036 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3039 g_return_if_fail(to_folder != NULL);
3041 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3042 if (MSG_IS_MOVE(msginfo->flags)) {
3043 if (!prefs_common.immediate_exec) {
3044 msginfo->to_folder->op_count--;
3045 if (msginfo->to_folder->op_count == 0) {
3046 folderview_update_item(msginfo->to_folder, 0);
3051 msginfo->to_folder = to_folder;
3052 if (MSG_IS_DELETED(msginfo->flags))
3053 summaryview->deleted--;
3054 if (MSG_IS_COPY(msginfo->flags)) {
3055 summaryview->copied--;
3056 if (!prefs_common.immediate_exec)
3057 msginfo->to_folder->op_count--;
3059 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_COPY);
3060 if (!MSG_IS_MOVE(msginfo->flags)) {
3061 procmsg_msginfo_set_flags(msginfo, 0, MSG_MOVE);
3062 summaryview->moved++;
3065 if (!prefs_common.immediate_exec) {
3066 summary_set_row_marks(summaryview, row);
3068 msginfo->to_folder->op_count++;
3069 if (msginfo->to_folder->op_count == 1)
3070 folderview_update_item(msginfo->to_folder, 0);
3074 debug_print("Message %d is set to move to %s\n",
3075 msginfo->msgnum, to_folder->path);
3078 void summary_move_selected_to(SummaryView *summaryview, FolderItem *to_folder)
3082 if (!to_folder) return;
3083 if (!summaryview->folder_item ||
3084 summaryview->folder_item->folder->type == F_NEWS) return;
3086 if (summary_is_locked(summaryview)) return;
3088 if (summaryview->folder_item == to_folder) {
3089 alertpanel_notice(_("Destination is same as current folder."));
3093 for (cur = GTK_CLIST(summaryview->ctree)->selection;
3094 cur != NULL; cur = cur->next)
3096 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
3098 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
3100 if (prefs_common.immediate_exec)
3101 summary_execute(summaryview);
3103 summary_status_show(summaryview);
3105 folderview_update_item(to_folder, 0);
3109 void summary_move_to(SummaryView *summaryview)
3111 FolderItem *to_folder;
3113 if (!summaryview->folder_item ||
3114 summaryview->folder_item->folder->type == F_NEWS) return;
3116 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
3117 FOLDER_SEL_MOVE, NULL);
3118 summary_move_selected_to(summaryview, to_folder);
3121 static void summary_copy_row_to(SummaryView *summaryview, GtkCTreeNode *row,
3122 FolderItem *to_folder)
3124 gboolean changed = FALSE;
3125 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3128 g_return_if_fail(to_folder != NULL);
3130 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3131 if (MSG_IS_COPY(msginfo->flags)) {
3132 if (!prefs_common.immediate_exec) {
3133 msginfo->to_folder->op_count--;
3134 if (msginfo->to_folder->op_count == 0) {
3135 folderview_update_item(msginfo->to_folder, 0);
3140 msginfo->to_folder = to_folder;
3141 if (MSG_IS_DELETED(msginfo->flags))
3142 summaryview->deleted--;
3143 if (MSG_IS_MOVE(msginfo->flags)) {
3144 summaryview->moved--;
3145 if (!prefs_common.immediate_exec)
3146 msginfo->to_folder->op_count--;
3148 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_MOVE);
3149 if (!MSG_IS_COPY(msginfo->flags)) {
3150 procmsg_msginfo_set_flags(msginfo, 0, MSG_COPY);
3151 summaryview->copied++;
3154 if (!prefs_common.immediate_exec) {
3155 summary_set_row_marks(summaryview, row);
3157 msginfo->to_folder->op_count++;
3158 if (msginfo->to_folder->op_count == 1)
3159 folderview_update_item(msginfo->to_folder, 0);
3163 debug_print("Message %d is set to copy to %s\n",
3164 msginfo->msgnum, to_folder->path);
3167 void summary_copy_selected_to(SummaryView *summaryview, FolderItem *to_folder)
3171 if (!to_folder) return;
3172 if (!summaryview->folder_item) return;
3174 if (summary_is_locked(summaryview)) return;
3176 if (summaryview->folder_item == to_folder) {
3178 (_("Destination to copy is same as current folder."));
3182 for (cur = GTK_CLIST(summaryview->ctree)->selection;
3183 cur != NULL; cur = cur->next)
3185 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
3187 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
3189 if (prefs_common.immediate_exec)
3190 summary_execute(summaryview);
3192 summary_status_show(summaryview);
3194 folderview_update_item(to_folder, 0);
3198 void summary_copy_to(SummaryView *summaryview)
3200 FolderItem *to_folder;
3202 if (!summaryview->folder_item) return;
3204 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
3205 FOLDER_SEL_COPY, NULL);
3206 summary_copy_selected_to(summaryview, to_folder);
3209 void summary_add_address(SummaryView *summaryview)
3214 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
3215 summaryview->selected);
3216 if (!msginfo) return;
3218 Xstrdup_a(from, msginfo->from, return);
3219 eliminate_address_comment(from);
3220 extract_address(from);
3221 addressbook_add_contact(msginfo->fromname, from, NULL);
3224 void summary_select_all(SummaryView *summaryview)
3226 if (summaryview->messages >= 500) {
3227 STATUSBAR_PUSH(summaryview->mainwin,
3228 _("Selecting all messages..."));
3229 main_window_cursor_wait(summaryview->mainwin);
3232 gtk_clist_select_all(GTK_CLIST(summaryview->ctree));
3234 if (summaryview->messages >= 500) {
3235 STATUSBAR_POP(summaryview->mainwin);
3236 main_window_cursor_normal(summaryview->mainwin);
3240 void summary_unselect_all(SummaryView *summaryview)
3242 gtk_sctree_unselect_all(GTK_SCTREE(summaryview->ctree));
3245 void summary_save_as(SummaryView *summaryview)
3247 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3249 gchar *filename = NULL;
3252 if (!summaryview->selected) return;
3253 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
3254 if (!msginfo) return;
3256 if (msginfo->subject) {
3257 Xstrdup_a(filename, msginfo->subject, return);
3258 subst_for_filename(filename);
3260 dest = filesel_select_file(_("Save as"), filename);
3262 if (is_file_exist(dest)) {
3265 aval = alertpanel(_("Overwrite"),
3266 _("Overwrite existing file?"),
3267 _("OK"), _("Cancel"), NULL);
3268 if (G_ALERTDEFAULT != aval) return;
3271 src = procmsg_get_message_file(msginfo);
3272 if (copy_file(src, dest, TRUE) < 0) {
3273 alertpanel_error(_("Can't save the file `%s'."),
3279 void summary_print(SummaryView *summaryview)
3281 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3282 GtkCList *clist = GTK_CLIST(summaryview->ctree);
3288 if (clist->selection == NULL) return;
3290 cmdline = input_dialog(_("Print"),
3291 _("Enter the print command line:\n"
3292 "(`%s' will be replaced with file name)"),
3293 prefs_common.print_cmd);
3294 if (!cmdline) return;
3295 if (!(p = strchr(cmdline, '%')) || *(p + 1) != 's' ||
3296 strchr(p + 2, '%')) {
3297 alertpanel_error(_("Print command line is invalid:\n`%s'"),