2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2003 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>
51 #include "mainwindow.h"
52 #include "folderview.h"
53 #include "summaryview.h"
54 #include "messageview.h"
55 #include "foldersel.h"
57 #include "procheader.h"
58 #include "sourcewindow.h"
59 #include "prefs_common.h"
60 #include "prefs_summary_column.h"
61 #include "prefs_filtering.h"
66 #include "stock_pixmap.h"
68 #include "alertpanel.h"
69 #include "inputdialog.h"
70 #include "statusbar.h"
72 #include "colorlabel.h"
75 #include "addressbook.h"
76 #include "addr_compl.h"
78 #include "folder_item_prefs.h"
79 #include "filtering.h"
80 #include "string_match.h"
84 #include "matcher_parser.h"
86 #include "description_window.h"
89 #define SUMMARY_COL_MARK_WIDTH 10
90 #define SUMMARY_COL_STATUS_WIDTH 13
91 #define SUMMARY_COL_LOCKED_WIDTH 13
92 #define SUMMARY_COL_MIME_WIDTH 11
95 static GtkStyle *bold_style;
96 static GtkStyle *bold_marked_style;
97 static GtkStyle *bold_deleted_style;
98 static GtkStyle *small_style;
99 static GtkStyle *small_marked_style;
100 static GtkStyle *small_deleted_style;
102 static GdkPixmap *markxpm;
103 static GdkBitmap *markxpmmask;
104 static GdkPixmap *deletedxpm;
105 static GdkBitmap *deletedxpmmask;
107 static GdkPixmap *newxpm;
108 static GdkBitmap *newxpmmask;
109 static GdkPixmap *unreadxpm;
110 static GdkBitmap *unreadxpmmask;
111 static GdkPixmap *repliedxpm;
112 static GdkBitmap *repliedxpmmask;
113 static GdkPixmap *forwardedxpm;
114 static GdkBitmap *forwardedxpmmask;
115 static GdkPixmap *ignorethreadxpm;
116 static GdkBitmap *ignorethreadxpmmask;
117 static GdkPixmap *lockedxpm;
118 static GdkBitmap *lockedxpmmask;
120 static GdkPixmap *clipxpm;
121 static GdkBitmap *clipxpmmask;
122 static GdkPixmap *keyxpm;
123 static GdkBitmap *keyxpmmask;
124 static GdkPixmap *clipkeyxpm;
125 static GdkBitmap *clipkeyxpmmask;
126 static GdkPixmap *gpgsignedxpm;
127 static GdkBitmap *gpgsignedxpmmask;
129 static void summary_free_msginfo_func (GtkCTree *ctree,
132 static void summary_set_marks_func (GtkCTree *ctree,
136 static void summary_set_menu_sensitive (SummaryView *summaryview);
138 static void summary_set_hide_read_msgs_menu (SummaryView *summaryview,
141 static guint summary_get_msgnum (SummaryView *summaryview,
144 static GtkCTreeNode *summary_find_prev_msg
145 (SummaryView *summaryview,
146 GtkCTreeNode *current_node);
147 static GtkCTreeNode *summary_find_next_msg
148 (SummaryView *summaryview,
149 GtkCTreeNode *current_node);
151 static GtkCTreeNode *summary_find_prev_flagged_msg
152 (SummaryView *summaryview,
153 GtkCTreeNode *current_node,
155 gboolean start_from_prev);
156 static GtkCTreeNode *summary_find_next_flagged_msg
157 (SummaryView *summaryview,
158 GtkCTreeNode *current_node,
160 gboolean start_from_next);
162 static GtkCTreeNode *summary_find_msg_by_msgnum
163 (SummaryView *summaryview,
166 static void summary_update_status (SummaryView *summaryview);
168 /* display functions */
169 static void summary_status_show (SummaryView *summaryview);
170 static void summary_set_column_titles (SummaryView *summaryview);
171 static void summary_set_ctree_from_list (SummaryView *summaryview,
173 static void summary_set_header (SummaryView *summaryview,
176 static void summary_display_msg (SummaryView *summaryview,
178 static void summary_display_msg_full (SummaryView *summaryview,
181 gboolean all_headers);
182 static void summary_set_row_marks (SummaryView *summaryview,
185 /* message handling */
186 static void summary_mark_row (SummaryView *summaryview,
188 static void summary_lock_row (SummaryView *summaryview,
190 static void summary_unlock_row (SummaryView *summaryview,
192 static void summary_mark_row_as_read (SummaryView *summaryview,
194 static void summary_mark_row_as_unread (SummaryView *summaryview,
196 static void summary_delete_row (SummaryView *summaryview,
198 static void summary_unmark_row (SummaryView *summaryview,
200 static void summary_move_row_to (SummaryView *summaryview,
202 FolderItem *to_folder);
203 static void summary_copy_row_to (SummaryView *summaryview,
205 FolderItem *to_folder);
207 static void summary_delete_duplicated_func
210 SummaryView *summaryview);
212 static void summary_execute_move (SummaryView *summaryview);
213 static void summary_execute_move_func (GtkCTree *ctree,
216 static void summary_execute_copy (SummaryView *summaryview);
217 static void summary_execute_copy_func (GtkCTree *ctree,
220 static void summary_execute_delete (SummaryView *summaryview);
221 static void summary_execute_delete_func (GtkCTree *ctree,
225 static void summary_thread_init (SummaryView *summaryview);
226 static void summary_ignore_thread (SummaryView *summaryview);
227 static void summary_unignore_thread (SummaryView *summaryview);
229 static void summary_unthread_for_exec (SummaryView *summaryview);
230 static void summary_unthread_for_exec_func (GtkCTree *ctree,
234 void summary_simplify_subject(SummaryView *summaryview, gchar * rexp,
238 void summary_processing(SummaryView *summaryview, GSList * mlist);
240 static void summary_filter_func (GtkCTree *ctree,
244 static void summary_colorlabel_menu_item_activate_cb
247 static void summary_colorlabel_menu_item_activate_item_cb
248 (GtkMenuItem *label_menu_item,
250 static void summary_colorlabel_menu_create(SummaryView *summaryview);
252 static GtkWidget *summary_ctree_create (SummaryView *summaryview);
254 /* callback functions */
255 static gint 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 gint summary_key_pressed (GtkWidget *ctree,
266 SummaryView *summaryview);
267 static gint summary_searchbar_pressed (GtkWidget *ctree,
269 SummaryView *summaryview);
270 static void summary_searchbar_focus_evt (GtkWidget *ctree,
271 GdkEventFocus *event,
272 SummaryView *summaryview);
273 static void summary_searchtype_changed (GtkMenuItem *widget,
275 static void summary_open_row (GtkSCTree *sctree,
276 SummaryView *summaryview);
277 static void summary_tree_expanded (GtkCTree *ctree,
279 SummaryView *summaryview);
280 static void summary_tree_collapsed (GtkCTree *ctree,
282 SummaryView *summaryview);
283 static void summary_selected (GtkCTree *ctree,
286 SummaryView *summaryview);
287 static void summary_col_resized (GtkCList *clist,
290 SummaryView *summaryview);
291 static void summary_reply_cb (SummaryView *summaryview,
294 static void summary_execute_cb (SummaryView *summaryview,
297 static void summary_show_all_header_cb (SummaryView *summaryview,
301 static void summary_add_address_cb (SummaryView *summaryview,
304 static void summary_create_filter_cb (SummaryView *summaryview,
308 static void summary_mark_clicked (GtkWidget *button,
309 SummaryView *summaryview);
310 static void summary_status_clicked (GtkWidget *button,
311 SummaryView *summaryview);
312 static void summary_mime_clicked (GtkWidget *button,
313 SummaryView *summaryview);
314 static void summary_num_clicked (GtkWidget *button,
315 SummaryView *summaryview);
316 static void summary_score_clicked (GtkWidget *button,
317 SummaryView *summaryview);
318 static void summary_size_clicked (GtkWidget *button,
319 SummaryView *summaryview);
320 static void summary_date_clicked (GtkWidget *button,
321 SummaryView *summaryview);
322 static void summary_from_clicked (GtkWidget *button,
323 SummaryView *summaryview);
324 static void summary_subject_clicked (GtkWidget *button,
325 SummaryView *summaryview);
326 static void summary_score_clicked (GtkWidget *button,
327 SummaryView *summaryview);
328 static void summary_locked_clicked (GtkWidget *button,
329 SummaryView *summaryview);
331 static void summary_start_drag (GtkWidget *widget,
334 SummaryView *summaryview);
335 static void summary_drag_data_get (GtkWidget *widget,
336 GdkDragContext *drag_context,
337 GtkSelectionData *selection_data,
340 SummaryView *summaryview);
342 /* custom compare functions for sorting */
344 static gint summary_cmp_by_mark (GtkCList *clist,
347 static gint summary_cmp_by_status (GtkCList *clist,
350 static gint summary_cmp_by_mime (GtkCList *clist,
353 static gint summary_cmp_by_num (GtkCList *clist,
356 static gint summary_cmp_by_size (GtkCList *clist,
359 static gint summary_cmp_by_date (GtkCList *clist,
362 static gint summary_cmp_by_from (GtkCList *clist,
365 static gint summary_cmp_by_simplified_subject
369 static gint summary_cmp_by_score (GtkCList *clist,
372 static gint summary_cmp_by_label (GtkCList *clist,
375 static gint summary_cmp_by_to (GtkCList *clist,
378 static gint summary_cmp_by_subject (GtkCList *clist,
381 static gint summary_cmp_by_locked (GtkCList *clist,
385 static void news_flag_crosspost (MsgInfo *msginfo);
387 static void tog_searchbar_cb (GtkWidget *w,
390 static void summary_find_answers (SummaryView *summaryview,
393 static gboolean summary_update_msg (gpointer source, gpointer data);
395 GtkTargetEntry summary_drag_types[1] =
397 {"text/plain", GTK_TARGET_SAME_APP, TARGET_DUMMY}
400 static GtkItemFactoryEntry summary_popup_entries[] =
402 {N_("/_Reply"), "<control>R", summary_reply_cb, COMPOSE_REPLY, NULL},
403 {N_("/Repl_y to"), NULL, NULL, 0, "<Branch>"},
404 {N_("/Repl_y to/_all"), "<shift><control>R", summary_reply_cb, COMPOSE_REPLY_TO_ALL, NULL},
405 {N_("/Repl_y to/_sender"), NULL, summary_reply_cb, COMPOSE_REPLY_TO_SENDER, NULL},
406 {N_("/Repl_y to/mailing _list"),
407 "<control>L", summary_reply_cb, COMPOSE_REPLY_TO_LIST, NULL},
408 {N_("/Follow-up and reply to"), NULL, summary_reply_cb, COMPOSE_FOLLOWUP_AND_REPLY_TO, NULL},
409 {N_("/---"), NULL, NULL, 0, "<Separator>"},
410 {N_("/_Forward"), "<control><alt>F", summary_reply_cb, COMPOSE_FORWARD, NULL},
411 {N_("/Redirect"), NULL, summary_reply_cb, COMPOSE_REDIRECT, NULL},
412 {N_("/---"), NULL, NULL, 0, "<Separator>"},
413 {N_("/Re-_edit"), NULL, summary_reedit, 0, NULL},
414 {N_("/---"), NULL, NULL, 0, "<Separator>"},
415 {N_("/M_ove..."), "<control>O", summary_move_to, 0, NULL},
416 {N_("/_Copy..."), "<shift><control>O", summary_copy_to, 0, NULL},
417 {N_("/_Delete"), "<control>D", summary_delete, 0, NULL},
418 {N_("/Cancel a news message"), NULL, summary_cancel, 0, NULL},
419 {N_("/E_xecute"), "X", summary_execute_cb, 0, NULL},
420 {N_("/---"), NULL, NULL, 0, "<Separator>"},
421 {N_("/_Mark"), NULL, NULL, 0, "<Branch>"},
422 {N_("/_Mark/_Mark"), "<shift>asterisk", summary_mark, 0, NULL},
423 {N_("/_Mark/_Unmark"), "U", summary_unmark, 0, NULL},
424 {N_("/_Mark/---"), NULL, NULL, 0, "<Separator>"},
425 {N_("/_Mark/Mark as unr_ead"), "<shift>exclam", summary_mark_as_unread, 0, NULL},
426 {N_("/_Mark/Mark as rea_d"), NULL, summary_mark_as_read, 0, NULL},
427 {N_("/_Mark/Mark all read"), NULL, summary_mark_all_read, 0, NULL},
428 {N_("/_Mark/Ignore thread"), NULL, summary_ignore_thread, 0, NULL},
429 {N_("/_Mark/Unignore thread"), NULL, summary_unignore_thread, 0, NULL},
430 {N_("/_Mark/Lock"), NULL, summary_msgs_lock, 0, NULL},
431 {N_("/_Mark/Unlock"), NULL, summary_msgs_unlock, 0, NULL},
432 {N_("/Color la_bel"), NULL, NULL, 0, NULL},
434 {N_("/---"), NULL, NULL, 0, "<Separator>"},
435 {N_("/Add sender to address boo_k"),
436 NULL, summary_add_address_cb, 0, NULL},
437 {N_("/Create f_ilter rule"), NULL, NULL, 0, "<Branch>"},
438 {N_("/Create f_ilter rule/_Automatically"),
439 NULL, summary_create_filter_cb, FILTER_BY_AUTO, NULL},
440 {N_("/Create f_ilter rule/by _From"),
441 NULL, summary_create_filter_cb, FILTER_BY_FROM, NULL},
442 {N_("/Create f_ilter rule/by _To"),
443 NULL, summary_create_filter_cb, FILTER_BY_TO, NULL},
444 {N_("/Create f_ilter rule/by _Subject"),
445 NULL, summary_create_filter_cb, FILTER_BY_SUBJECT, NULL},
446 {N_("/---"), NULL, NULL, 0, "<Separator>"},
447 {N_("/_View"), NULL, NULL, 0, "<Branch>"},
448 {N_("/_View/Open in new _window"),
449 "<control><alt>N", summary_open_msg, 0, NULL},
450 {N_("/_View/_Source"), "<control>U", summary_view_source, 0, NULL},
451 {N_("/_View/All _header"), "<control>H", summary_show_all_header_cb, 0, "<ToggleItem>"},
452 {N_("/---"), NULL, NULL, 0, "<Separator>"},
453 {N_("/_Save as..."), "<control>S", summary_save_as, 0, NULL},
454 {N_("/_Print..."), NULL, summary_print, 0, NULL},
455 {N_("/---"), NULL, NULL, 0, "<Separator>"},
456 {N_("/Select _all"), "<control>A", summary_select_all, 0, NULL},
457 {N_("/Select t_hread"), NULL, summary_select_thread, 0, NULL}
458 }; /* see also list in menu_connect_identical_items() in menu.c if this changes */
460 static const gchar *const col_label[N_SUMMARY_COLS] = {
461 N_("M"), /* S_COL_MARK */
462 N_("S"), /* S_COL_STATUS */
464 N_("Subject"), /* S_COL_SUBJECT */
465 N_("From"), /* S_COL_FROM */
466 N_("Date"), /* S_COL_DATE */
467 N_("Size"), /* S_COL_SIZE */
468 N_("No."), /* S_COL_NUMBER */
469 N_("Score"), /* S_COL_SCORE */
470 N_("L") /* S_COL_LOCKED */
474 * Strings describing how to use Extended Search
476 * When adding new lines, remember to put 2 strings for each line
478 static gchar *search_descr_strings[] = {
479 "a", N_("all messages"),
480 "ag #", N_("messages whose age is greather than #"),
481 "al #", N_("messages whose age is greather than #"),
482 "b S", N_("messages which contain S in the message body"),
483 "B S", N_("messages which contain S in the whole message"),
484 "c S", N_("messages carbon-copied to S"),
485 "C S", N_("message is either to: or cc: to S"),
486 "D", N_("deleted messages"), /** how I can filter deleted messages **/
487 "e S", N_("messages which contain S in the Sender field"),
488 "E S", N_("true if execute \"S\" succeeds"),
489 "f S", N_("messages originating from user S"),
490 "F", N_("forwarded messages"),
491 "h S", N_("messages which contain header S"),
492 "i S", N_("messages which contain S in Message-Id header"),
493 "I S", N_("messages which contain S in inreplyto header"),
494 "L", N_("locked messages"),
495 "n S", N_("messages which are in newsgroup S"),
496 "N", N_("new messages"),
497 "O", N_("old messages"),
498 "r", N_("messages which have been replied to"),
499 "R", N_("read messages"),
500 "s S", N_("messages which contain S in subject"),
501 "se #", N_("messages whose score is equal to #"),
502 "sg #", N_("messages whose score is greater than #"),
503 "sl #", N_("messages whose score is lower than #"),
504 "Se #", N_("messages whose size is equal to #"),
505 "Sg #", N_("messages whose size is greater than #"),
506 "Ss #", N_("messages whose size is smaller than #"),
507 "t S", N_("messages which have been sent to S"),
508 "T", N_("marked messages"),
509 "U", N_("unread messages"),
510 "x S", N_("messages which contain S in References header"),
511 "y S", N_("messages which contain S in X-Label header"),
513 "&", N_("logical AND operator"),
514 "|", N_("logical OR operator"),
515 "! or ~", N_("logical NOT operator"),
516 "%", N_("case sensitive search"),
520 static DescriptionWindow search_descr = {
523 N_("Extended Search symbols"),
527 static void search_description_cb(GtkWidget *widget)
529 description_window_create(&search_descr);
532 SummaryView *summary_create(void)
534 SummaryView *summaryview;
536 GtkWidget *scrolledwin;
540 GtkWidget *hbox_search;
541 GtkWidget *statlabel_folder;
542 GtkWidget *statlabel_select;
543 GtkWidget *statlabel_msgs;
545 GtkWidget *toggle_eventbox;
546 GtkWidget *toggle_arrow;
547 GtkWidget *popupmenu;
548 GtkWidget *search_type_opt;
549 GtkWidget *search_type;
550 GtkWidget *search_string;
551 GtkWidget *search_hbbox;
552 GtkWidget *search_description;
554 GtkWidget *toggle_search;
555 GtkTooltips *search_tip;
556 GtkItemFactory *popupfactory;
559 debug_print("Creating summary view...\n");
560 summaryview = g_new0(SummaryView, 1);
562 #define SUMMARY_VBOX_SPACING 3
563 vbox = gtk_vbox_new(FALSE, SUMMARY_VBOX_SPACING);
565 /* create status label */
566 hbox = gtk_hbox_new(FALSE, 0);
568 search_tip = gtk_tooltips_new();
569 toggle_search = gtk_toggle_button_new();
571 gtk_tooltips_set_tip(GTK_TOOLTIPS(search_tip),
573 _("Toggle quick-search bar"), NULL);
575 gtk_box_pack_start(GTK_BOX(hbox), toggle_search, FALSE, FALSE, 2);
577 hbox_l = gtk_hbox_new(FALSE, 0);
578 gtk_box_pack_start(GTK_BOX(hbox), hbox_l, TRUE, TRUE, 0);
580 statlabel_folder = gtk_label_new("");
581 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_folder, FALSE, FALSE, 2);
582 statlabel_select = gtk_label_new("");
583 gtk_box_pack_start(GTK_BOX(hbox_l), statlabel_select, FALSE, FALSE, 12);
585 /* toggle view button */
586 toggle_eventbox = gtk_event_box_new();
587 gtk_box_pack_end(GTK_BOX(hbox), toggle_eventbox, FALSE, FALSE, 4);
588 toggle_arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
589 gtk_container_add(GTK_CONTAINER(toggle_eventbox), toggle_arrow);
590 gtk_signal_connect(GTK_OBJECT(toggle_eventbox), "button_press_event",
591 GTK_SIGNAL_FUNC(summary_toggle_pressed),
595 statlabel_msgs = gtk_label_new("");
596 gtk_box_pack_end(GTK_BOX(hbox), statlabel_msgs, FALSE, FALSE, 4);
598 hbox_spc = gtk_hbox_new(FALSE, 0);
599 gtk_box_pack_end(GTK_BOX(hbox), hbox_spc, FALSE, FALSE, 6);
601 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
602 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
603 GTK_POLICY_AUTOMATIC,
605 gtk_box_pack_start(GTK_BOX(vbox), scrolledwin, TRUE, TRUE, 0);
606 gtk_widget_set_usize(vbox,
607 prefs_common.summaryview_width,
608 prefs_common.summaryview_height);
610 ctree = summary_ctree_create(summaryview);
612 gtk_scrolled_window_set_hadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
613 GTK_CLIST(ctree)->hadjustment);
614 gtk_scrolled_window_set_vadjustment(GTK_SCROLLED_WINDOW(scrolledwin),
615 GTK_CLIST(ctree)->vadjustment);
616 gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
619 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
622 hbox_search = gtk_hbox_new(FALSE, 0);
623 gtk_box_pack_start(GTK_BOX(vbox), hbox_search, FALSE, FALSE, 0);
625 search_type_opt = gtk_option_menu_new();
626 gtk_widget_show(search_type_opt);
627 gtk_box_pack_start(GTK_BOX(hbox_search), search_type_opt, FALSE, FALSE, 0);
629 search_type = gtk_menu_new();
630 MENUITEM_ADD (search_type, menuitem, _("Subject"), S_SEARCH_SUBJECT);
631 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
632 GTK_SIGNAL_FUNC(summary_searchtype_changed),
634 MENUITEM_ADD (search_type, menuitem, _("From"), S_SEARCH_FROM);
635 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
636 GTK_SIGNAL_FUNC(summary_searchtype_changed),
638 MENUITEM_ADD (search_type, menuitem, _("To"), S_SEARCH_TO);
639 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
640 GTK_SIGNAL_FUNC(summary_searchtype_changed),
642 MENUITEM_ADD (search_type, menuitem, _("Extended"), S_SEARCH_EXTENDED);
643 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
644 GTK_SIGNAL_FUNC(summary_searchtype_changed),
647 gtk_option_menu_set_menu(GTK_OPTION_MENU(search_type_opt), search_type);
649 gtk_option_menu_set_history(GTK_OPTION_MENU(search_type_opt), prefs_common.summary_quicksearch_type);
651 gtk_widget_show(search_type);
653 search_string = gtk_entry_new();
655 gtk_box_pack_start(GTK_BOX(hbox_search), search_string, FALSE, FALSE, 2);
657 gtkut_button_set_create(&search_hbbox, &search_description, _("Extended Symbols"),
658 NULL, NULL, NULL, NULL);
659 gtk_signal_connect(GTK_OBJECT(search_description), "clicked",
660 GTK_SIGNAL_FUNC(search_description_cb), NULL);
661 gtk_box_pack_start(GTK_BOX(hbox_search), search_hbbox, FALSE, FALSE, 2);
663 gtk_widget_show(search_string);
664 gtk_widget_show(hbox_search);
666 gtk_signal_connect(GTK_OBJECT(search_string), "key_press_event",
667 GTK_SIGNAL_FUNC(summary_searchbar_pressed),
670 gtk_signal_connect(GTK_OBJECT(search_string), "focus_in_event",
671 GTK_SIGNAL_FUNC(summary_searchbar_focus_evt),
674 gtk_signal_connect(GTK_OBJECT(search_string), "focus_out_event",
675 GTK_SIGNAL_FUNC(summary_searchbar_focus_evt),
678 gtk_signal_connect (GTK_OBJECT(toggle_search), "toggled",
679 GTK_SIGNAL_FUNC(tog_searchbar_cb), summaryview);
681 /* create popup menu */
682 n_entries = sizeof(summary_popup_entries) /
683 sizeof(summary_popup_entries[0]);
684 popupmenu = menu_create_items(summary_popup_entries, n_entries,
685 "<SummaryView>", &popupfactory,
688 summaryview->vbox = vbox;
689 summaryview->scrolledwin = scrolledwin;
690 summaryview->ctree = ctree;
691 summaryview->hbox = hbox;
692 summaryview->hbox_l = hbox_l;
693 summaryview->hbox_search = hbox_search;
694 summaryview->statlabel_folder = statlabel_folder;
695 summaryview->statlabel_select = statlabel_select;
696 summaryview->statlabel_msgs = statlabel_msgs;
697 summaryview->toggle_eventbox = toggle_eventbox;
698 summaryview->toggle_arrow = toggle_arrow;
699 summaryview->toggle_search = toggle_search;
700 summaryview->popupmenu = popupmenu;
701 summaryview->popupfactory = popupfactory;
702 summaryview->lock_count = 0;
703 summaryview->search_type_opt = search_type_opt;
704 summaryview->search_type = search_type;
705 summaryview->search_string = search_string;
706 summaryview->search_description = search_description;
707 summaryview->msginfo_update_callback_id =
708 hooks_register_hook(MSGINFO_UPDATE_HOOKLIST, summary_update_msg, (gpointer) summaryview);
710 summaryview->target_list = gtk_target_list_new(summary_drag_types, 1);
712 /* CLAWS: need this to get the SummaryView * from
714 gtk_object_set_data(GTK_OBJECT(ctree), "summaryview", (gpointer)summaryview);
716 gtk_widget_show_all(vbox);
721 void summary_init(SummaryView *summaryview)
723 static GdkFont *boldfont = NULL;
724 static GdkFont *smallfont = NULL;
728 gtk_widget_realize(summaryview->ctree);
729 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_MARK,
730 &markxpm, &markxpmmask);
731 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_DELETED,
732 &deletedxpm, &deletedxpmmask);
733 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_NEW,
734 &newxpm, &newxpmmask);
735 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_UNREAD,
736 &unreadxpm, &unreadxpmmask);
737 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_REPLIED,
738 &repliedxpm, &repliedxpmmask);
739 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_FORWARDED,
740 &forwardedxpm, &forwardedxpmmask);
741 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP,
742 &clipxpm, &clipxpmmask);
743 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_LOCKED,
744 &lockedxpm, &lockedxpmmask);
745 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_IGNORETHREAD,
746 &ignorethreadxpm, &ignorethreadxpmmask);
747 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_CLIP_KEY,
748 &clipkeyxpm, &clipkeyxpmmask);
749 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_KEY,
750 &keyxpm, &keyxpmmask);
751 stock_pixmap_gdk(summaryview->ctree, STOCK_PIXMAP_GPG_SIGNED,
752 &gpgsignedxpm, &gpgsignedxpmmask);
755 bold_style = gtk_style_copy
756 (gtk_widget_get_style(summaryview->ctree));
758 boldfont = gtkut_font_load(BOLD_FONT);
760 bold_style->font = boldfont;
761 bold_marked_style = gtk_style_copy(bold_style);
762 bold_marked_style->fg[GTK_STATE_NORMAL] =
763 summaryview->color_marked;
764 bold_deleted_style = gtk_style_copy(bold_style);
765 bold_deleted_style->fg[GTK_STATE_NORMAL] =
766 summaryview->color_dim;
770 small_style = gtk_style_copy
771 (gtk_widget_get_style(summaryview->ctree));
773 smallfont = gtkut_font_load(SMALL_FONT);
774 small_style->font = smallfont;
775 small_marked_style = gtk_style_copy(small_style);
776 small_marked_style->fg[GTK_STATE_NORMAL] =
777 summaryview->color_marked;
778 small_deleted_style = gtk_style_copy(small_style);
779 small_deleted_style->fg[GTK_STATE_NORMAL] =
780 summaryview->color_dim;
783 style = gtk_style_copy(gtk_widget_get_style
784 (summaryview->statlabel_folder));
786 style->font = smallfont;
787 gtk_widget_set_style(summaryview->statlabel_folder, style);
788 gtk_widget_set_style(summaryview->statlabel_select, style);
789 gtk_widget_set_style(summaryview->statlabel_msgs, style);
791 pixmap = stock_pixmap_widget(summaryview->hbox_l, STOCK_PIXMAP_DIR_OPEN);
792 gtk_box_pack_start(GTK_BOX(summaryview->hbox_l), pixmap, FALSE, FALSE, 4);
793 gtk_box_reorder_child(GTK_BOX(summaryview->hbox_l), pixmap, 0);
794 gtk_widget_show(pixmap);
795 summaryview->folder_pixmap = pixmap;
797 pixmap = stock_pixmap_widget(summaryview->hbox, STOCK_PIXMAP_QUICKSEARCH);
798 gtk_container_add (GTK_CONTAINER(summaryview->toggle_search), pixmap);
799 gtk_widget_show(pixmap);
800 summaryview->quick_search_pixmap = pixmap;
802 /* Init summaryview prefs */
803 summaryview->sort_key = SORT_BY_NONE;
804 summaryview->sort_type = SORT_ASCENDING;
806 /* Init summaryview extra data */
807 summaryview->simplify_subject_preg = NULL;
809 summary_clear_list(summaryview);
810 summary_set_column_titles(summaryview);
811 summary_colorlabel_menu_create(summaryview);
812 summary_set_menu_sensitive(summaryview);
816 GtkCTreeNode * summary_find_next_important_score(SummaryView *summaryview,
817 GtkCTreeNode *current_node)
819 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
822 gint best_score = MIN_SCORE;
823 GtkCTreeNode *best_node = NULL;
826 /*node = current_node;*/
827 node = GTK_CTREE_NODE_NEXT(current_node);
829 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
831 for (; node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
832 msginfo = gtk_ctree_node_get_row_data(ctree, node);
833 if (msginfo->score >= summaryview->important_score)
835 if (msginfo->score > best_score) {
836 best_score = msginfo->score;
847 GtkCTreeNode * summary_find_prev_important_score(SummaryView *summaryview,
848 GtkCTreeNode *current_node)
850 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
853 gint best_score = MIN_SCORE;
854 GtkCTreeNode *best_node = NULL;
857 /*node = current_node;*/
858 node = GTK_CTREE_NODE_PREV(current_node);
860 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
862 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
863 msginfo = gtk_ctree_node_get_row_data(ctree, node);
864 if (msginfo->score >= summaryview->important_score)
866 if (msginfo->score > best_score) {
867 best_score = msginfo->score;
878 #define CURRENTLY_DISPLAYED(m) \
879 ( (m->msgnum == displayed_msgnum) \
880 && (!g_strcasecmp(m->folder->name,item->name)) )
882 gboolean summary_show(SummaryView *summaryview, FolderItem *item)
884 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
886 GSList *mlist = NULL;
889 guint selected_msgnum = 0;
890 guint displayed_msgnum = 0;
893 if (summary_is_locked(summaryview)) return FALSE;
896 summary_lock(summaryview);
898 if (item != summaryview->folder_item) {
899 /* changing folder, reset search */
900 gtk_entry_set_text(GTK_ENTRY(summaryview->search_string), "");
903 /* STATUSBAR_POP(summaryview->mainwin); */
905 is_refresh = (item == summaryview->folder_item) ? TRUE : FALSE;
907 selected_msgnum = summary_get_msgnum(summaryview,
908 summaryview->selected);
909 displayed_msgnum = summary_get_msgnum(summaryview,
910 summaryview->displayed);
913 /* process the marks if any */
914 if (summaryview->mainwin->lock_count == 0 &&
915 (summaryview->moved > 0 || summaryview->copied > 0)) {
918 val = alertpanel(_("Process mark"),
919 _("Some marks are left. Process it?"),
920 _("Yes"), _("No"), _("Cancel"));
921 if (G_ALERTDEFAULT == val) {
922 summary_unlock(summaryview);
923 summary_execute(summaryview);
924 summary_lock(summaryview);
925 } else if (G_ALERTALTERNATE == val) {
928 summary_unlock(summaryview);
932 folder_update_op_count();
935 gtk_clist_freeze(GTK_CLIST(ctree));
937 summary_clear_list(summaryview);
938 summary_set_column_titles(summaryview);
941 if (!item || !item->path || !item->parent || item->no_select) {
943 debug_print("empty folder\n\n");
944 summary_set_hide_read_msgs_menu(summaryview, FALSE);
945 summary_clear_all(summaryview);
946 summaryview->folder_item = item;
947 gtk_clist_thaw(GTK_CLIST(ctree));
948 summary_unlock(summaryview);
955 messageview_clear(summaryview->messageview);
957 summaryview->folder_item = item;
960 gtk_signal_handler_block_by_data(GTK_OBJECT(ctree), summaryview);
962 buf = g_strdup_printf(_("Scanning folder (%s)..."), item->path);
963 debug_print("%s\n", buf);
964 STATUSBAR_PUSH(summaryview->mainwin, buf);
967 main_window_cursor_wait(summaryview->mainwin);
970 mlist = item->folder->get_msg_list(item->folder, item, !update_cache);
973 USE LIST FROM CACHE, WILL NOT DISPLAY ANY MESSAGES DROPED
974 BY OTHER PROGRAMS TO THE FOLDER
976 mlist = folder_item_get_msg_list(item);
978 summary_processing(summaryview, mlist);
980 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
981 MsgInfo * msginfo = (MsgInfo *) cur->data;
983 msginfo->score = score_message(global_scoring, msginfo);
984 if (msginfo->score != MAX_SCORE &&
985 msginfo->score != MIN_SCORE) {
986 msginfo->score += score_message(item->prefs->scoring,
991 if (summaryview->folder_item->hide_read_msgs) {
994 summary_set_hide_read_msgs_menu(summaryview, TRUE);
996 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
997 MsgInfo * msginfo = (MsgInfo *) cur->data;
999 if ((MSG_IS_UNREAD(msginfo->flags)
1000 || MSG_IS_MARKED(msginfo->flags)
1001 || MSG_IS_LOCKED(msginfo->flags)
1002 || CURRENTLY_DISPLAYED(msginfo))
1003 && !MSG_IS_IGNORE_THREAD(msginfo->flags))
1004 not_killed = g_slist_prepend(not_killed, msginfo);
1006 procmsg_msginfo_free(msginfo);
1008 g_slist_free(mlist);
1011 summary_set_hide_read_msgs_menu(summaryview, FALSE);
1014 if (strlen(gtk_entry_get_text(GTK_ENTRY(summaryview->search_string))) > 0) {
1016 gint search_type = GPOINTER_TO_INT(gtk_object_get_user_data(
1017 GTK_OBJECT(GTK_MENU_ITEM(gtk_menu_get_active(
1018 GTK_MENU(summaryview->search_type))))));
1019 gchar *search_string = gtk_entry_get_text(GTK_ENTRY(summaryview->search_string));
1020 gchar *searched_header = NULL;
1021 MatcherList * tmp_list = NULL;
1023 if (search_type == S_SEARCH_EXTENDED) {
1024 char *newstr = NULL;
1026 newstr = expand_search_string(search_string);
1028 tmp_list = matcher_parser_get_cond(newstr);
1036 for (cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
1037 MsgInfo * msginfo = (MsgInfo *) cur->data;
1039 switch (search_type) {
1040 case S_SEARCH_SUBJECT:
1041 searched_header = msginfo->subject;
1044 searched_header = msginfo->from;
1047 searched_header = msginfo->to;
1049 case S_SEARCH_EXTENDED:
1052 debug_print("unknown search type (%d)\n", search_type);
1055 if (search_type != S_SEARCH_EXTENDED) {
1056 if (searched_header && strcasestr(searched_header, search_string) != NULL)
1057 not_killed = g_slist_prepend(not_killed, msginfo);
1059 procmsg_msginfo_free(msginfo);
1061 if ((tmp_list != NULL) && matcherlist_match(tmp_list, msginfo))
1062 not_killed = g_slist_prepend(not_killed, msginfo);
1064 procmsg_msginfo_free(msginfo);
1067 if (search_type == S_SEARCH_EXTENDED && tmp_list != NULL) {
1068 matcherlist_free(tmp_list);
1072 g_slist_free(mlist);
1076 if ((global_scoring || item->prefs->scoring)) {
1081 kill_score = prefs_common.kill_score;
1082 if (item->prefs->kill_score > kill_score)
1083 kill_score = item->prefs->kill_score;
1084 for(cur = mlist ; cur != NULL ; cur = g_slist_next(cur)) {
1085 MsgInfo * msginfo = (MsgInfo *) cur->data;
1087 if (msginfo->score > kill_score)
1088 not_killed = g_slist_prepend(not_killed, msginfo);
1090 procmsg_msginfo_free(msginfo);
1092 g_slist_free(mlist);
1096 STATUSBAR_POP(summaryview->mainwin);
1098 /* set ctree and hash table from the msginfo list, and
1099 create the thread */
1100 summary_set_ctree_from_list(summaryview, mlist);
1102 g_slist_free(mlist);
1104 if (summaryview->sort_key != SORT_BY_NONE)
1105 summary_sort(summaryview, summaryview->sort_key, summaryview->sort_type);
1107 gtk_signal_handler_unblock_by_data(GTK_OBJECT(ctree), summaryview);
1109 gtk_clist_thaw(GTK_CLIST(ctree));
1112 summaryview->displayed =
1113 summary_find_msg_by_msgnum(summaryview,
1115 if (!summaryview->displayed)
1116 messageview_clear(summaryview->messageview);
1117 summary_select_by_msgnum(summaryview, selected_msgnum);
1118 if (!summaryview->selected) {
1119 /* no selected message - select first unread
1120 message, but do not display it */
1121 node = summary_find_next_flagged_msg(summaryview, NULL,
1123 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL)
1124 node = gtk_ctree_node_nth
1126 item->sort_type == SORT_DESCENDING
1127 ? 0 : GTK_CLIST(ctree)->rows - 1);
1128 summary_select_node(summaryview, node, FALSE, TRUE);
1131 /* select first unread message */
1132 if (summaryview->sort_key == SORT_BY_SCORE)
1133 node = summary_find_next_important_score(summaryview,
1136 node = summary_find_next_flagged_msg(summaryview, NULL,
1138 if (node == NULL && GTK_CLIST(ctree)->row_list != NULL) {
1139 node = gtk_ctree_node_nth
1141 item->sort_type == SORT_DESCENDING
1142 ? 0 : GTK_CLIST(ctree)->rows - 1);
1144 if (prefs_common.open_unread_on_enter ||
1145 prefs_common.always_show_msg) {
1146 summary_unlock(summaryview);
1147 summary_select_node(summaryview, node, TRUE, TRUE);
1148 summary_lock(summaryview);
1150 summary_select_node(summaryview, node, FALSE, TRUE);
1153 summary_set_column_titles(summaryview);
1154 summary_status_show(summaryview);
1155 summary_set_menu_sensitive(summaryview);
1156 toolbar_main_set_sensitive(summaryview->mainwin);
1159 STATUSBAR_PUSH(summaryview->mainwin, _("Done."));
1160 STATUSBAR_POP(summaryview->mainwin);
1161 main_window_cursor_normal(summaryview->mainwin);
1162 summary_unlock(summaryview);
1168 #undef CURRENTLY_DISPLAYED
1170 void summary_clear_list(SummaryView *summaryview)
1172 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1175 gtk_clist_freeze(clist);
1177 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree),
1178 NULL, summary_free_msginfo_func, NULL);
1180 if (summaryview->folder_item) {
1181 summaryview->folder_item->opened = FALSE;
1182 summaryview->folder_item = NULL;
1185 summaryview->display_msg = FALSE;
1187 summaryview->selected = NULL;
1188 summaryview->displayed = NULL;
1189 summaryview->total_size = 0;
1190 summaryview->deleted = summaryview->moved = 0;
1191 summaryview->copied = 0;
1192 if (summaryview->msgid_table) {
1193 g_hash_table_destroy(summaryview->msgid_table);
1194 summaryview->msgid_table = NULL;
1196 if (summaryview->subject_table) {
1197 g_hash_table_destroy(summaryview->subject_table);
1198 summaryview->subject_table = NULL;
1200 summaryview->mlist = NULL;
1202 gtk_clist_clear(clist);
1203 if (summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
1204 optimal_width = gtk_clist_optimal_column_width
1205 (clist, summaryview->col_pos[S_COL_SUBJECT]);
1206 gtk_clist_set_column_width
1207 (clist, summaryview->col_pos[S_COL_SUBJECT],
1211 gtk_clist_thaw(clist);
1214 void summary_clear_all(SummaryView *summaryview)
1216 messageview_clear(summaryview->messageview);
1217 summary_clear_list(summaryview);
1218 summary_set_menu_sensitive(summaryview);
1219 toolbar_main_set_sensitive(summaryview->mainwin);
1220 summary_status_show(summaryview);
1223 void summary_lock(SummaryView *summaryview)
1225 summaryview->lock_count++;
1228 void summary_unlock(SummaryView *summaryview)
1230 if (summaryview->lock_count)
1231 summaryview->lock_count--;
1234 gboolean summary_is_locked(SummaryView *summaryview)
1236 return summaryview->lock_count > 0;
1239 SummarySelection summary_get_selection_type(SummaryView *summaryview)
1241 GtkCList *clist = GTK_CLIST(summaryview->ctree);
1242 SummarySelection selection;
1244 if (!clist->row_list)
1245 selection = SUMMARY_NONE;
1246 else if (!clist->selection)
1247 selection = SUMMARY_SELECTED_NONE;
1248 else if (!clist->selection->next)
1249 selection = SUMMARY_SELECTED_SINGLE;
1251 selection = SUMMARY_SELECTED_MULTIPLE;
1256 GSList *summary_get_selected_msg_list(SummaryView *summaryview)
1258 GSList *mlist = NULL;
1262 for (cur = GTK_CLIST(summaryview->ctree)->selection; cur != NULL;
1264 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(cur->data);
1265 mlist = g_slist_prepend(mlist, msginfo);
1268 mlist = g_slist_reverse(mlist);
1273 GSList *summary_get_msg_list(SummaryView *summaryview)
1275 GSList *mlist = NULL;
1280 ctree = GTK_CTREE(summaryview->ctree);
1282 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1283 node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1284 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
1285 mlist = g_slist_prepend(mlist, msginfo);
1288 mlist = g_slist_reverse(mlist);
1293 static void summary_set_menu_sensitive(SummaryView *summaryview)
1295 GtkItemFactory *ifactory = summaryview->popupfactory;
1296 SensitiveCond state;
1298 GtkWidget *menuitem;
1301 static const struct {
1305 {"/Reply" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1306 {"/Reply to" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1307 {"/Reply to/all" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1308 {"/Reply to/sender" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1309 {"/Reply to/mailing list" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1310 {"/Follow-up and reply to" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST|M_NEWS},
1312 {"/Forward" , M_HAVE_ACCOUNT|M_TARGET_EXIST},
1313 {"/Redirect" , M_HAVE_ACCOUNT|M_SINGLE_TARGET_EXIST},
1315 {"/Re-edit" , M_HAVE_ACCOUNT|M_ALLOW_REEDIT},
1317 {"/Move..." , M_TARGET_EXIST|M_ALLOW_DELETE|M_UNLOCKED|M_NOT_NEWS},
1318 {"/Copy..." , M_TARGET_EXIST|M_EXEC|M_UNLOCKED},
1319 {"/Delete" , M_TARGET_EXIST|M_ALLOW_DELETE|M_UNLOCKED|M_NOT_NEWS},
1320 {"/Cancel a news message" , M_TARGET_EXIST|M_ALLOW_DELETE|M_UNLOCKED|M_NEWS},
1321 {"/Execute" , M_DELAY_EXEC},
1323 {"/Mark" , M_TARGET_EXIST},
1324 {"/Mark/Mark" , M_TARGET_EXIST},
1325 {"/Mark/Unmark" , M_TARGET_EXIST},
1326 {"/Mark/Mark as unread" , M_TARGET_EXIST},
1327 {"/Mark/Mark all read" , M_TARGET_EXIST},
1328 {"/Mark/Ignore thread" , M_TARGET_EXIST},
1329 {"/Mark/Lock" , M_TARGET_EXIST},
1330 {"/Mark/Unlock" , M_TARGET_EXIST},
1331 {"/Color label" , M_TARGET_EXIST},
1333 {"/Add sender to address book" , M_SINGLE_TARGET_EXIST},
1334 {"/Create filter rule" , M_SINGLE_TARGET_EXIST|M_UNLOCKED},
1336 {"/View" , M_SINGLE_TARGET_EXIST},
1337 {"/View/Open in new window" , M_SINGLE_TARGET_EXIST},
1338 {"/View/Source" , M_SINGLE_TARGET_EXIST},
1339 {"/View/All header" , M_SINGLE_TARGET_EXIST},
1341 {"/Save as..." , M_TARGET_EXIST|M_UNLOCKED},
1342 {"/Print..." , M_TARGET_EXIST|M_UNLOCKED},
1344 {"/Select thread" , M_SINGLE_TARGET_EXIST},
1345 {"/Select all" , M_TARGET_EXIST},
1350 main_window_set_menu_sensitive(summaryview->mainwin);
1352 state = main_window_get_current_state(summaryview->mainwin);
1354 for (i = 0; entry[i].entry != NULL; i++) {
1355 sensitive = ((entry[i].cond & state) == entry[i].cond);
1356 menu_set_sensitive(ifactory, entry[i].entry, sensitive);
1360 summary_lock(summaryview);
1361 menuitem = gtk_item_factory_get_widget(ifactory, "/View/All header");
1362 gtk_check_menu_item_set_active
1363 (GTK_CHECK_MENU_ITEM(menuitem),
1364 summaryview->messageview->mimeview->textview->show_all_headers);
1365 summary_unlock(summaryview);
1368 void summary_select_prev_unread(SummaryView *summaryview)
1372 node = summary_find_prev_flagged_msg
1373 (summaryview, summaryview->selected, MSG_UNREAD, FALSE);
1378 switch (prefs_common.next_unread_msg_dialog) {
1379 case NEXTUNREADMSGDIALOG_ALWAYS:
1380 val = alertpanel(_("No more unread messages"),
1381 _("No unread message found. "
1382 "Search from the end?"),
1383 _("Yes"), _("No"), NULL);
1385 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1386 val = G_ALERTDEFAULT;
1388 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1389 val = !G_ALERTDEFAULT;
1393 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1395 if (val != G_ALERTDEFAULT) return;
1396 node = summary_find_prev_flagged_msg(summaryview, NULL,
1401 alertpanel_notice(_("No unread messages."));
1403 summary_select_node(summaryview, node, TRUE, FALSE);
1406 void summary_select_next_unread(SummaryView *summaryview)
1408 GtkCTreeNode *node = summaryview->selected;
1409 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1411 node = summary_find_next_flagged_msg
1412 (summaryview, node, MSG_UNREAD, FALSE);
1415 summary_select_node(summaryview, node, TRUE, FALSE);
1417 node = summary_find_next_flagged_msg
1418 (summaryview, NULL, MSG_UNREAD, FALSE);
1422 switch (prefs_common.next_unread_msg_dialog) {
1423 case NEXTUNREADMSGDIALOG_ALWAYS:
1424 val = alertpanel(_("No more unread messages"),
1425 _("No unread message found. "
1426 "Go to next folder?"),
1427 _("Yes"), _("No"), NULL);
1429 case NEXTUNREADMSGDIALOG_ASSUME_YES:
1430 val = G_ALERTDEFAULT;
1432 case NEXTUNREADMSGDIALOG_ASSUME_NO:
1437 _("Internal error: unexpected value for prefs_common.next_unread_msg_dialog\n"));
1440 if (val == G_ALERTDEFAULT) {
1441 if (gtk_signal_n_emissions_by_name
1442 (GTK_OBJECT(ctree), "key_press_event") > 0)
1443 gtk_signal_emit_stop_by_name
1446 folderview_select_next_unread(summaryview->folderview);
1452 summary_select_node(summaryview, node, TRUE, FALSE);
1457 void summary_select_prev_new(SummaryView *summaryview)
1461 node = summary_find_prev_flagged_msg
1462 (summaryview, summaryview->selected, MSG_NEW, FALSE);
1467 val = alertpanel(_("No more new messages"),
1468 _("No new message found. "
1469 "Search from the end?"),
1470 _("Yes"), _("No"), NULL);
1471 if (val != G_ALERTDEFAULT) return;
1472 node = summary_find_prev_flagged_msg(summaryview, NULL,
1477 alertpanel_notice(_("No new messages."));
1479 summary_select_node(summaryview, node, TRUE, FALSE);
1482 void summary_select_next_new(SummaryView *summaryview)
1484 GtkCTreeNode *node = summaryview->selected;
1485 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1487 while ((node = summary_find_next_flagged_msg
1488 (summaryview, node, MSG_NEW, FALSE)) == NULL) {
1491 val = alertpanel(_("No more new messages"),
1492 _("No new message found. "
1493 "Go to next folder?"),
1494 _("Yes"), _("Search again"), _("No"));
1495 if (val == G_ALERTDEFAULT) {
1496 if (gtk_signal_n_emissions_by_name
1497 (GTK_OBJECT(ctree), "key_press_event") > 0)
1498 gtk_signal_emit_stop_by_name
1501 folderview_select_next_unread(summaryview->folderview);
1503 } else if (val == G_ALERTALTERNATE)
1510 summary_select_node(summaryview, node, TRUE, FALSE);
1513 void summary_select_prev_marked(SummaryView *summaryview)
1517 node = summary_find_prev_flagged_msg
1518 (summaryview, summaryview->selected, MSG_MARKED, TRUE);
1523 val = alertpanel(_("No more marked messages"),
1524 _("No marked message found. "
1525 "Search from the end?"),
1526 _("Yes"), _("No"), NULL);
1527 if (val != G_ALERTDEFAULT) return;
1528 node = summary_find_prev_flagged_msg(summaryview, NULL,
1533 alertpanel_notice(_("No marked messages."));
1535 summary_select_node(summaryview, node, TRUE, FALSE);
1538 void summary_select_next_marked(SummaryView *summaryview)
1542 node = summary_find_next_flagged_msg
1543 (summaryview, summaryview->selected, MSG_MARKED, TRUE);
1548 val = alertpanel(_("No more marked messages"),
1549 _("No marked message found. "
1550 "Search from the beginning?"),
1551 _("Yes"), _("No"), NULL);
1552 if (val != G_ALERTDEFAULT) return;
1553 node = summary_find_next_flagged_msg(summaryview, NULL,
1558 alertpanel_notice(_("No marked messages."));
1560 summary_select_node(summaryview, node, TRUE, FALSE);
1563 void summary_select_prev_labeled(SummaryView *summaryview)
1567 node = summary_find_prev_flagged_msg
1568 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
1573 val = alertpanel(_("No more labeled messages"),
1574 _("No labeled message found. "
1575 "Search from the end?"),
1576 _("Yes"), _("No"), NULL);
1577 if (val != G_ALERTDEFAULT) return;
1578 node = summary_find_prev_flagged_msg(summaryview, NULL,
1579 MSG_CLABEL_FLAG_MASK, TRUE);
1583 alertpanel_notice(_("No labeled messages."));
1585 summary_select_node(summaryview, node, TRUE, FALSE);
1588 void summary_select_next_labeled(SummaryView *summaryview)
1592 node = summary_find_next_flagged_msg
1593 (summaryview, summaryview->selected, MSG_CLABEL_FLAG_MASK, TRUE);
1598 val = alertpanel(_("No more labeled messages"),
1599 _("No labeled message found. "
1600 "Search from the beginning?"),
1601 _("Yes"), _("No"), NULL);
1602 if (val != G_ALERTDEFAULT) return;
1603 node = summary_find_next_flagged_msg(summaryview, NULL,
1604 MSG_CLABEL_FLAG_MASK, TRUE);
1608 alertpanel_notice(_("No labeled messages."));
1610 summary_select_node(summaryview, node, TRUE, FALSE);
1613 void summary_select_by_msgnum(SummaryView *summaryview, guint msgnum)
1617 node = summary_find_msg_by_msgnum(summaryview, msgnum);
1618 summary_select_node(summaryview, node, FALSE, TRUE);
1622 * summary_select_node:
1623 * @summaryview: Summary view.
1624 * @node: Summary tree node.
1625 * @display_msg: TRUE to display the selected message.
1626 * @do_refresh: TRUE to refresh the widget.
1628 * Select @node (bringing it into view by scrolling and expanding its
1629 * thread, if necessary) and unselect all others. If @display_msg is
1630 * TRUE, display the corresponding message in the message view.
1631 * If @do_refresh is TRUE, the widget is refreshed.
1633 void summary_select_node(SummaryView *summaryview, GtkCTreeNode *node,
1634 gboolean display_msg, gboolean do_refresh)
1636 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1639 gtkut_ctree_expand_parent_all(ctree, node);
1642 gtk_widget_grab_focus(GTK_WIDGET(ctree));
1643 gtk_ctree_node_moveto(ctree, node, -1, 0.5, 0);
1645 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
1646 if (display_msg && summaryview->displayed == node)
1647 summaryview->displayed = NULL;
1648 summaryview->display_msg = display_msg;
1649 gtk_sctree_select(GTK_SCTREE(ctree), node);
1653 static guint summary_get_msgnum(SummaryView *summaryview, GtkCTreeNode *node)
1655 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1660 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1661 return msginfo->msgnum;
1664 static GtkCTreeNode *summary_find_prev_msg(SummaryView *summaryview,
1665 GtkCTreeNode *current_node)
1667 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1672 node = current_node;
1674 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
1676 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1677 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1678 if (msginfo && !MSG_IS_DELETED(msginfo->flags)) break;
1684 static GtkCTreeNode *summary_find_next_msg(SummaryView *summaryview,
1685 GtkCTreeNode *current_node)
1687 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1692 node = current_node;
1694 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1696 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1697 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1698 if (msginfo && !MSG_IS_DELETED(msginfo->flags)) break;
1704 static GtkCTreeNode *summary_find_prev_flagged_msg(SummaryView *summaryview,
1705 GtkCTreeNode *current_node,
1707 gboolean start_from_prev)
1709 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1714 if (start_from_prev)
1715 node = GTK_CTREE_NODE_PREV(current_node);
1717 node = current_node;
1719 node = gtk_ctree_node_nth(ctree, GTK_CLIST(ctree)->rows - 1);
1721 for (; node != NULL; node = GTK_CTREE_NODE_PREV(node)) {
1722 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1723 if (msginfo && (msginfo->flags.perm_flags & flags) != 0) break;
1729 static GtkCTreeNode *summary_find_next_flagged_msg(SummaryView *summaryview,
1730 GtkCTreeNode *current_node,
1732 gboolean start_from_next)
1734 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1739 if (start_from_next)
1740 node = gtkut_ctree_node_next(ctree, current_node);
1742 node = current_node;
1744 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1746 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1747 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1748 /* Find msg with matching flags but ignore messages with
1749 ignore flags, if searching for new or unread messages */
1750 if (!(((flags & (MSG_NEW | MSG_UNREAD)) != 0) && MSG_IS_IGNORE_THREAD(msginfo->flags)) &&
1751 (msginfo && (msginfo->flags.perm_flags & flags) != 0))
1758 static GtkCTreeNode *summary_find_msg_by_msgnum(SummaryView *summaryview,
1761 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1765 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1767 for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1768 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1769 if (msginfo && msginfo->msgnum == msgnum) break;
1775 static guint attract_hash_func(gconstpointer key)
1781 Xstrdup_a(str, (const gchar *)key, return 0);
1788 for (p += 1; *p != '\0'; p++)
1789 h = (h << 5) - h + *p;
1795 static gint attract_compare_func(gconstpointer a, gconstpointer b)
1797 return subject_compare((const gchar *)a, (const gchar *)b) == 0;
1800 void summary_attract_by_subject(SummaryView *summaryview)
1802 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1803 GtkCList *clist = GTK_CLIST(ctree);
1804 GtkCTreeNode *src_node;
1805 GtkCTreeNode *dst_node, *sibling;
1807 MsgInfo *src_msginfo, *dst_msginfo;
1808 GHashTable *subject_table;
1810 debug_print("Attracting messages by subject...");
1811 STATUSBAR_PUSH(summaryview->mainwin,
1812 _("Attracting messages by subject..."));
1814 main_window_cursor_wait(summaryview->mainwin);
1815 gtk_clist_freeze(clist);
1817 subject_table = g_hash_table_new(attract_hash_func,
1818 attract_compare_func);
1820 for (src_node = GTK_CTREE_NODE(clist->row_list);
1823 tmp = GTK_CTREE_ROW(src_node)->sibling;
1824 src_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(src_node);
1825 if (!src_msginfo) continue;
1826 if (!src_msginfo->subject) continue;
1828 /* find attracting node */
1829 dst_node = g_hash_table_lookup(subject_table,
1830 src_msginfo->subject);
1833 dst_msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(dst_node);
1835 /* if the time difference is more than 20 days,
1837 if (ABS(src_msginfo->date_t - dst_msginfo->date_t)
1838 > 60 * 60 * 24 * 20)
1841 sibling = GTK_CTREE_ROW(dst_node)->sibling;
1842 if (src_node != sibling)
1843 gtk_ctree_move(ctree, src_node, NULL, sibling);
1846 g_hash_table_insert(subject_table,
1847 src_msginfo->subject, src_node);
1850 g_hash_table_destroy(subject_table);
1852 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
1854 gtk_clist_thaw(clist);
1856 debug_print("done.\n");
1857 STATUSBAR_POP(summaryview->mainwin);
1859 main_window_cursor_normal(summaryview->mainwin);
1862 static void summary_free_msginfo_func(GtkCTree *ctree, GtkCTreeNode *node,
1865 MsgInfo *msginfo = gtk_ctree_node_get_row_data(ctree, node);
1868 procmsg_msginfo_free(msginfo);
1871 static void summary_set_marks_func(GtkCTree *ctree, GtkCTreeNode *node,
1874 SummaryView *summaryview = data;
1877 msginfo = gtk_ctree_node_get_row_data(ctree, node);
1879 if (msginfo->folder && msginfo->folder->folder &&
1880 msginfo->folder->folder->klass->type == F_NEWS)
1881 news_flag_crosspost(msginfo);
1883 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)
1884 && procmsg_msg_has_marked_parent(msginfo))
1885 summaryview->unreadmarked++;
1887 if (MSG_IS_DELETED(msginfo->flags))
1888 summaryview->deleted++;
1890 summaryview->total_size += msginfo->size;
1892 summary_set_row_marks(summaryview, node);
1895 static void summary_update_status(SummaryView *summaryview)
1897 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
1901 summaryview->total_size =
1902 summaryview->deleted = summaryview->moved = summaryview->copied = 0;
1904 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1905 node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
1906 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
1908 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)
1909 && procmsg_msg_has_marked_parent(msginfo))
1910 summaryview->unreadmarked++;
1911 if (MSG_IS_DELETED(msginfo->flags))
1912 summaryview->deleted++;
1913 if (MSG_IS_MOVE(msginfo->flags))
1914 summaryview->moved++;
1915 if (MSG_IS_COPY(msginfo->flags))
1916 summaryview->copied++;
1917 summaryview->total_size += msginfo->size;
1921 static void summary_status_show(SummaryView *summaryview)
1924 gchar *del, *mv, *cp;
1928 GList *rowlist, *cur;
1929 guint n_selected = 0;
1934 if (!summaryview->folder_item) {
1935 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder), "");
1936 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), "");
1937 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), "");
1941 rowlist = GTK_CLIST(summaryview->ctree)->selection;
1942 for (cur = rowlist; cur != NULL; cur = cur->next) {
1943 msginfo = gtk_ctree_node_get_row_data
1944 (GTK_CTREE(summaryview->ctree),
1945 GTK_CTREE_NODE(cur->data));
1947 g_warning("summary_status_show(): msginfo == NULL\n");
1949 sel_size += msginfo->size;
1954 name = folder_item_get_name(summaryview->folder_item);
1955 gtk_label_set(GTK_LABEL(summaryview->statlabel_folder), name);
1958 if (summaryview->deleted)
1959 del = g_strdup_printf(_("%d deleted"), summaryview->deleted);
1962 if (summaryview->moved)
1963 mv = g_strdup_printf(_("%s%d moved"),
1964 summaryview->deleted ? _(", ") : "",
1965 summaryview->moved);
1968 if (summaryview->copied)
1969 cp = g_strdup_printf(_("%s%d copied"),
1970 summaryview->deleted ||
1971 summaryview->moved ? _(", ") : "",
1972 summaryview->copied);
1976 if (summaryview->deleted || summaryview->moved || summaryview->copied)
1982 sel = g_strdup_printf(" (%s)", to_human_readable(sel_size));
1983 if (n_selected == 1)
1984 itstr = g_strdup(_(" item selected"));
1986 itstr = g_strdup(_(" items selected"));
1989 itstr = g_strdup("");
1992 str = g_strconcat(n_selected ? itos(n_selected) : "",
1993 itstr, sel, spc, del, mv, cp, NULL);
1994 gtk_label_set(GTK_LABEL(summaryview->statlabel_select), str);
2002 str = g_strdup_printf(_("%d new, %d unread, %d total (%s)"),
2004 summaryview->folder_item->new_msgs,
2005 summaryview->folder_item->unread_msgs,
2006 summaryview->folder_item->total_msgs,
2007 to_human_readable(summaryview->total_size));
2008 gtk_label_set(GTK_LABEL(summaryview->statlabel_msgs), str);
2012 static void summary_set_column_titles(SummaryView *summaryview)
2014 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2020 SummaryColumnType type;
2021 gboolean single_char;
2022 GtkJustification justify;
2024 static FolderSortKey sort_by[N_SUMMARY_COLS] = {
2037 for (pos = 0; pos < N_SUMMARY_COLS; pos++) {
2038 type = summaryview->col_state[pos].type;
2040 /* CLAWS: mime and unread are single char headers */
2041 single_char = (type == S_COL_MIME || type == S_COL_STATUS);
2042 justify = (type == S_COL_NUMBER || type == S_COL_SIZE)
2043 ? GTK_JUSTIFY_RIGHT : GTK_JUSTIFY_LEFT;
2050 if (prefs_common.trans_hdr)
2051 title = gettext(col_label[type]);
2053 title = col_label[type];
2055 /* CLAWS: dummies for mark and locked headers */
2061 title = gettext(col_label[type]);
2064 if (type == S_COL_MIME) {
2065 label = gtk_pixmap_new(clipxpm, clipxpmmask);
2066 gtk_widget_show(label);
2067 gtk_clist_set_column_widget(clist, pos, label);
2071 gtk_clist_set_column_title(clist, pos, title);
2075 /* CLAWS: changed so that locked and mark headers
2076 * show a pixmap instead of single character */
2077 hbox = gtk_hbox_new(FALSE, 4);
2079 if (type == S_COL_LOCKED)
2080 label = gtk_pixmap_new(lockedxpm, lockedxpmmask);
2081 else if (type == S_COL_MARK)
2082 label = gtk_pixmap_new(markxpm, markxpmmask);
2084 label = gtk_label_new(title);
2086 if (justify == GTK_JUSTIFY_RIGHT)
2087 gtk_box_pack_end(GTK_BOX(hbox), label,
2090 gtk_box_pack_start(GTK_BOX(hbox), label,
2093 if (summaryview->sort_key == sort_by[type]) {
2094 arrow = gtk_arrow_new
2095 (summaryview->sort_type == SORT_ASCENDING
2096 ? GTK_ARROW_DOWN : GTK_ARROW_UP,
2098 if (justify == GTK_JUSTIFY_RIGHT)
2099 gtk_box_pack_start(GTK_BOX(hbox), arrow,
2102 gtk_box_pack_end(GTK_BOX(hbox), arrow,
2106 gtk_widget_show_all(hbox);
2107 gtk_clist_set_column_widget(clist, pos, hbox);
2111 void summary_sort(SummaryView *summaryview,
2112 FolderSortKey sort_key, FolderSortType sort_type)
2114 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2115 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2116 GtkCListCompareFunc cmp_func = NULL;
2120 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mark;
2122 case SORT_BY_STATUS:
2123 cmp_func = (GtkCListCompareFunc)summary_cmp_by_status;
2126 cmp_func = (GtkCListCompareFunc)summary_cmp_by_mime;
2128 case SORT_BY_NUMBER:
2129 cmp_func = (GtkCListCompareFunc)summary_cmp_by_num;
2132 cmp_func = (GtkCListCompareFunc)summary_cmp_by_size;
2135 cmp_func = (GtkCListCompareFunc)summary_cmp_by_date;
2138 cmp_func = (GtkCListCompareFunc)summary_cmp_by_from;
2140 case SORT_BY_SUBJECT:
2141 if (summaryview->simplify_subject_preg)
2142 cmp_func = (GtkCListCompareFunc)summary_cmp_by_simplified_subject;
2144 cmp_func = (GtkCListCompareFunc)summary_cmp_by_subject;
2147 cmp_func = (GtkCListCompareFunc)summary_cmp_by_score;
2150 cmp_func = (GtkCListCompareFunc)summary_cmp_by_label;
2153 cmp_func = (GtkCListCompareFunc)summary_cmp_by_to;
2155 case SORT_BY_LOCKED:
2156 cmp_func = (GtkCListCompareFunc)summary_cmp_by_locked;
2165 summaryview->sort_key = sort_key;
2166 summaryview->sort_type = sort_type;
2168 summary_set_column_titles(summaryview);
2169 summary_set_menu_sensitive(summaryview);
2170 if(cmp_func != NULL) {
2171 debug_print("Sorting summary...");
2172 STATUSBAR_PUSH(summaryview->mainwin, _("Sorting summary..."));
2174 main_window_cursor_wait(summaryview->mainwin);
2176 gtk_clist_set_compare_func(clist, cmp_func);
2178 gtk_clist_set_sort_type(clist, (GtkSortType)sort_type);
2180 gtk_sctree_sort_recursive(ctree, NULL);
2182 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
2184 main_window_cursor_normal(summaryview->mainwin);
2186 debug_print("done.\n");
2187 STATUSBAR_POP(summaryview->mainwin);
2191 gboolean summary_insert_gnode_func(GtkCTree *ctree, guint depth, GNode *gnode,
2192 GtkCTreeNode *cnode, gpointer data)
2194 SummaryView *summaryview = (SummaryView *)data;
2195 MsgInfo *msginfo = (MsgInfo *)gnode->data;
2196 gchar *text[N_SUMMARY_COLS];
2197 gint *col_pos = summaryview->col_pos;
2198 const gchar *msgid = msginfo->msgid;
2199 GHashTable *msgid_table = summaryview->msgid_table;
2201 summary_set_header(summaryview, text, msginfo);
2203 gtk_ctree_set_node_info(ctree, cnode, text[col_pos[S_COL_SUBJECT]], 2,
2204 NULL, NULL, NULL, NULL, FALSE,
2205 gnode->parent->parent ? TRUE : FALSE);
2206 #define SET_TEXT(col) \
2207 gtk_ctree_node_set_text(ctree, cnode, col_pos[col], \
2210 SET_TEXT(S_COL_NUMBER);
2211 SET_TEXT(S_COL_SCORE);
2212 SET_TEXT(S_COL_SIZE);
2213 SET_TEXT(S_COL_DATE);
2214 SET_TEXT(S_COL_FROM);
2215 SET_TEXT(S_COL_SUBJECT);
2219 GTKUT_CTREE_NODE_SET_ROW_DATA(cnode, msginfo);
2220 summary_set_marks_func(ctree, cnode, summaryview);
2222 if (msgid && msgid[0] != '\0')
2223 g_hash_table_insert(msgid_table, (gchar *)msgid, cnode);
2228 static void summary_set_ctree_from_list(SummaryView *summaryview,
2231 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2233 GtkCTreeNode *node = NULL;
2234 GHashTable *msgid_table;
2235 GHashTable *subject_table;
2240 debug_print("\tSetting summary from message data...");
2241 STATUSBAR_PUSH(summaryview->mainwin,
2242 _("Setting summary from message data..."));
2245 msgid_table = g_hash_table_new(g_str_hash, g_str_equal);
2246 summaryview->msgid_table = msgid_table;
2247 subject_table = g_hash_table_new(g_str_hash, g_str_equal);
2248 summaryview->subject_table = subject_table;
2250 if (prefs_common.use_addr_book)
2251 start_address_completion();
2253 for (cur = mlist ; cur != NULL; cur = cur->next) {
2254 msginfo = (MsgInfo *)cur->data;
2255 msginfo->threadscore = msginfo->score;
2258 if (summaryview->threaded) {
2259 GNode *root, *gnode;
2261 root = procmsg_get_thread_tree(mlist);
2263 for (gnode = root->children; gnode != NULL;
2264 gnode = gnode->next) {
2265 node = gtk_ctree_insert_gnode
2266 (ctree, NULL, node, gnode,
2267 summary_insert_gnode_func, summaryview);
2270 g_node_destroy(root);
2272 summary_thread_init(summaryview);
2274 gchar *text[N_SUMMARY_COLS];
2276 for (; mlist != NULL; mlist = mlist->next) {
2277 msginfo = (MsgInfo *)mlist->data;
2279 summary_set_header(summaryview, text, msginfo);
2281 node = gtk_ctree_insert_node
2282 (ctree, NULL, node, text, 2,
2283 NULL, NULL, NULL, NULL, FALSE, FALSE);
2284 GTKUT_CTREE_NODE_SET_ROW_DATA(node, msginfo);
2285 summary_set_marks_func(ctree, node, summaryview);
2287 if (msginfo->msgid && msginfo->msgid[0] != '\0')
2288 g_hash_table_insert(msgid_table,
2289 msginfo->msgid, node);
2291 subject_table_insert(subject_table,
2298 if (prefs_common.enable_hscrollbar &&
2299 summaryview->col_pos[S_COL_SUBJECT] == N_SUMMARY_COLS - 1) {
2302 optimal_width = gtk_clist_optimal_column_width
2303 (GTK_CLIST(ctree), summaryview->col_pos[S_COL_SUBJECT]);
2304 gtk_clist_set_column_width(GTK_CLIST(ctree),
2305 summaryview->col_pos[S_COL_SUBJECT],
2309 if (prefs_common.use_addr_book)
2310 end_address_completion();
2312 debug_print("done.\n");
2313 STATUSBAR_POP(summaryview->mainwin);
2314 if (debug_get_mode()) {
2315 debug_print("\tmsgid hash table size = %d\n",
2316 g_hash_table_size(msgid_table));
2317 debug_print("\tsubject hash table size = %d\n",
2318 g_hash_table_size(subject_table));
2322 static gchar *summary_complete_address(const gchar *addr)
2325 gchar *res, *tmp, *email_addr;
2327 Xstrdup_a(email_addr, addr, return NULL);
2328 extract_address(email_addr);
2329 g_return_val_if_fail(*email_addr, NULL);
2332 * completion stuff must be already initialized
2335 if (1 < (count = complete_address(email_addr))) {
2336 tmp = get_complete_address(1);
2337 /* tmp = addressbook_lookup_name( email_addr );
2339 res = procheader_get_fromname(tmp);
2346 static void summary_set_header(SummaryView *summaryview, gchar *text[],
2349 static gchar date_modified[80];
2350 static gchar *to = NULL;
2351 static gchar col_score[11];
2352 static gchar buf[BUFFSIZE];
2353 gint *col_pos = summaryview->col_pos;
2355 text[col_pos[S_COL_MARK]] = NULL;
2356 text[col_pos[S_COL_STATUS]] = NULL;
2357 text[col_pos[S_COL_MIME]] = NULL;
2358 text[col_pos[S_COL_LOCKED]] = NULL;
2359 text[col_pos[S_COL_NUMBER]] = itos(msginfo->msgnum);
2360 text[col_pos[S_COL_SIZE]] = to_human_readable(msginfo->size);
2361 text[col_pos[S_COL_SCORE]] = itos_buf(col_score, msginfo->score);
2363 if (msginfo->date_t) {
2364 procheader_date_get_localtime(date_modified,
2365 sizeof(date_modified),
2367 text[col_pos[S_COL_DATE]] = date_modified;
2368 } else if (msginfo->date)
2369 text[col_pos[S_COL_DATE]] = msginfo->date;
2371 text[col_pos[S_COL_DATE]] = _("(No Date)");
2373 text[col_pos[S_COL_FROM]] = msginfo->fromname ? msginfo->fromname :
2375 if (prefs_common.swap_from && msginfo->from && msginfo->to &&
2376 !MSG_IS_NEWS(msginfo->flags)) {
2379 Xstrdup_a(addr, msginfo->from, return);
2380 extract_address(addr);
2381 if (prefs_common.use_addr_book) {
2382 if (account_find_from_address(addr)) {
2383 addr = summary_complete_address(msginfo->to);
2385 to = g_strconcat("-->", addr == NULL ? msginfo->to : addr, NULL);
2386 text[col_pos[S_COL_FROM]] = to;
2390 if (account_find_from_address(addr)) {
2392 to = g_strconcat("-->", msginfo->to, NULL);
2393 text[col_pos[S_COL_FROM]] = to;
2399 * CLAWS: note that the "text[col_pos[S_COL_FROM]] != to" is really a hack,
2400 * checking whether the above block (which handles the special case of
2401 * the --> in sent boxes) was executed.
2403 if (text[col_pos[S_COL_FROM]] != to && prefs_common.use_addr_book && msginfo->from) {
2404 gchar *from = summary_complete_address(msginfo->from);
2408 text[col_pos[S_COL_FROM]] = to;
2412 if (summaryview->simplify_subject_preg != NULL)
2413 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ?
2414 string_remove_match(buf, BUFFSIZE, msginfo->subject,
2415 summaryview->simplify_subject_preg) :
2418 text[col_pos[S_COL_SUBJECT]] = msginfo->subject ? msginfo->subject :
2422 static void summary_display_msg(SummaryView *summaryview, GtkCTreeNode *row)
2424 summary_display_msg_full(summaryview, row, FALSE, FALSE);
2427 static void summary_display_msg_full(SummaryView *summaryview,
2429 gboolean new_window, gboolean all_headers)
2431 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2436 if (summaryview->displayed == row)
2439 summaryview->messageview->filtered = FALSE;
2442 g_return_if_fail(row != NULL);
2444 if (summary_is_locked(summaryview)) return;
2445 summary_lock(summaryview);
2447 STATUSBAR_POP(summaryview->mainwin);
2450 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2452 if (new_window || !prefs_common.mark_as_read_on_new_window) {
2453 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)
2454 && procmsg_msg_has_marked_parent(msginfo))
2455 summaryview->unreadmarked--;
2456 if (MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)) {
2457 procmsg_msginfo_unset_flags
2458 (msginfo, MSG_NEW | MSG_UNREAD, 0);
2459 summary_set_row_marks(summaryview, row);
2460 gtk_clist_thaw(GTK_CLIST(ctree));
2461 summary_status_show(summaryview);
2463 flags = msginfo->flags;
2468 MessageView *msgview;
2470 msgview = messageview_create_with_new_window(summaryview->mainwin);
2471 messageview_show(msgview, msginfo, all_headers);
2473 MessageView *msgview;
2475 msgview = summaryview->messageview;
2477 summaryview->displayed = row;
2478 if (!messageview_is_visible(msgview))
2479 main_window_toggle_message_view(summaryview->mainwin);
2480 messageview_show(msgview, msginfo, all_headers);
2481 if (GTK_CLIST(msgview->mimeview->ctree)->row_list == NULL)
2482 gtk_widget_grab_focus(summaryview->ctree);
2484 gtkut_ctree_node_move_if_on_the_edge(ctree, row);
2487 summary_set_menu_sensitive(summaryview);
2488 toolbar_main_set_sensitive(summaryview->mainwin);
2490 summary_unlock(summaryview);
2493 void summary_display_msg_selected(SummaryView *summaryview,
2494 gboolean all_headers)
2496 if (summary_is_locked(summaryview)) return;
2497 summaryview->displayed = NULL;
2498 summary_display_msg_full(summaryview, summaryview->selected, FALSE,
2502 void summary_redisplay_msg(SummaryView *summaryview)
2506 if (summaryview->displayed) {
2507 node = summaryview->displayed;
2508 summaryview->displayed = NULL;
2509 summary_display_msg(summaryview, node);
2513 void summary_open_msg(SummaryView *summaryview)
2515 if (!summaryview->selected) return;
2517 /* CLAWS: if separate message view, don't open a new window
2518 * but rather use the current separated message view */
2519 summary_display_msg_full(summaryview, summaryview->selected,
2520 prefs_common.sep_msg ? FALSE : TRUE,
2524 void summary_view_source(SummaryView * summaryview)
2526 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2528 SourceWindow *srcwin;
2530 if (!summaryview->selected) return;
2532 srcwin = source_window_create();
2533 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
2534 source_window_show_msg(srcwin, msginfo);
2535 source_window_show(srcwin);
2538 void summary_reedit(SummaryView *summaryview)
2542 if (!summaryview->selected) return;
2543 if (!summaryview->folder_item) return;
2544 if (summaryview->folder_item->stype != F_OUTBOX &&
2545 summaryview->folder_item->stype != F_DRAFT &&
2546 summaryview->folder_item->stype != F_QUEUE) return;
2548 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
2549 summaryview->selected);
2550 if (!msginfo) return;
2552 compose_reedit(msginfo);
2555 void summary_step(SummaryView *summaryview, GtkScrollType type)
2557 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2560 if (summary_is_locked(summaryview)) return;
2562 if (type == GTK_SCROLL_STEP_FORWARD) {
2563 node = gtkut_ctree_node_next(ctree, summaryview->selected);
2565 gtkut_ctree_expand_parent_all(ctree, node);
2569 if (summaryview->selected) {
2570 node = GTK_CTREE_NODE_PREV(summaryview->selected);
2575 if (messageview_is_visible(summaryview->messageview))
2576 summaryview->display_msg = TRUE;
2578 gtk_signal_emit_by_name(GTK_OBJECT(ctree), "scroll_vertical",
2581 if (GTK_CLIST(ctree)->selection)
2582 gtk_sctree_set_anchor_row
2584 GTK_CTREE_NODE(GTK_CLIST(ctree)->selection->data));
2588 void summary_toggle_view(SummaryView *summaryview)
2590 if (!messageview_is_visible(summaryview->messageview) &&
2591 summaryview->selected)
2592 summary_display_msg(summaryview,
2593 summaryview->selected);
2595 main_window_toggle_message_view(summaryview->mainwin);
2598 static gboolean summary_search_unread_recursive(GtkCTree *ctree,
2604 msginfo = gtk_ctree_node_get_row_data(ctree, node);
2605 if (msginfo && MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags))
2607 node = GTK_CTREE_ROW(node)->children;
2609 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
2612 if (summary_search_unread_recursive(ctree, node) == TRUE)
2614 node = GTK_CTREE_ROW(node)->sibling;
2620 static gboolean summary_have_unread_children(SummaryView *summaryview,
2623 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2625 if (!node) return FALSE;
2627 node = GTK_CTREE_ROW(node)->children;
2630 if (summary_search_unread_recursive(ctree, node) == TRUE)
2632 node = GTK_CTREE_ROW(node)->sibling;
2638 static void summary_set_row_marks(SummaryView *summaryview, GtkCTreeNode *row)
2640 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2641 GtkStyle *style = NULL;
2644 gint *col_pos = summaryview->col_pos;
2646 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2647 if (!msginfo) return;
2649 flags = msginfo->flags;
2651 gtk_ctree_node_set_foreground(ctree, row, NULL);
2653 /* set new/unread column */
2654 if (MSG_IS_IGNORE_THREAD(flags)) {
2655 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
2656 ignorethreadxpm, ignorethreadxpmmask);
2657 } else if (MSG_IS_NEW(flags)) {
2658 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
2659 newxpm, newxpmmask);
2660 } else if (MSG_IS_UNREAD(flags)) {
2661 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
2662 unreadxpm, unreadxpmmask);
2663 } else if (MSG_IS_REPLIED(flags)) {
2664 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
2665 repliedxpm, repliedxpmmask);
2666 } else if (MSG_IS_FORWARDED(flags)) {
2667 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_STATUS],
2668 forwardedxpm, forwardedxpmmask);
2670 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_STATUS],
2674 if (prefs_common.bold_unread &&
2675 ((MSG_IS_UNREAD(flags) && !MSG_IS_IGNORE_THREAD(flags)) ||
2676 (!GTK_CTREE_ROW(row)->expanded &&
2677 GTK_CTREE_ROW(row)->children &&
2678 summary_have_unread_children(summaryview, row))))
2681 /* set mark column */
2682 if (MSG_IS_DELETED(flags)) {
2683 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
2684 deletedxpm, deletedxpmmask);
2686 style = bold_deleted_style;
2688 style = small_deleted_style;
2690 gtk_ctree_node_set_foreground
2691 (ctree, row, &summaryview->color_dim);
2692 } else if (MSG_IS_MARKED(flags)) {
2693 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MARK],
2694 markxpm, markxpmmask);
2695 } else if (MSG_IS_MOVE(flags)) {
2696 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], "o");
2698 style = bold_marked_style;
2700 style = small_marked_style;
2702 gtk_ctree_node_set_foreground
2703 (ctree, row, &summaryview->color_marked);
2704 } else if (MSG_IS_COPY(flags)) {
2705 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], "O");
2707 style = bold_marked_style;
2709 style = small_marked_style;
2711 gtk_ctree_node_set_foreground
2712 (ctree, row, &summaryview->color_marked);
2713 } else if ((global_scoring ||
2714 summaryview->folder_item->prefs->scoring) &&
2715 (msginfo->score >= summaryview->important_score) &&
2716 (MSG_IS_MARKED(msginfo->flags) || MSG_IS_MOVE(msginfo->flags) || MSG_IS_COPY(msginfo->flags))) {
2717 gtk_ctree_node_set_text(ctree, row, S_COL_MARK, "!");
2718 gtk_ctree_node_set_foreground(ctree, row,
2719 &summaryview->color_important);
2721 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MARK], NULL);
2724 if (MSG_IS_LOCKED(flags)) {
2725 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_LOCKED],
2726 lockedxpm, lockedxpmmask);
2729 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_LOCKED], NULL);
2732 if (MSG_IS_SIGNED(flags)) {
2733 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2734 gpgsignedxpm, gpgsignedxpmmask);
2735 } else if (MSG_IS_MIME(flags) && MSG_IS_ENCRYPTED(flags)) {
2736 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2737 clipkeyxpm, clipkeyxpmmask);
2738 } else if (MSG_IS_ENCRYPTED(flags)) {
2739 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2740 keyxpm, keyxpmmask);
2741 } else if (MSG_IS_MIME(flags)) {
2742 gtk_ctree_node_set_pixmap(ctree, row, col_pos[S_COL_MIME],
2743 clipxpm, clipxpmmask);
2745 gtk_ctree_node_set_text(ctree, row, col_pos[S_COL_MIME], NULL);
2748 style = small_style;
2750 gtk_ctree_node_set_row_style(ctree, row, style);
2752 if (MSG_GET_COLORLABEL(flags))
2753 summary_set_colorlabel_color(ctree, row, MSG_GET_COLORLABEL_VALUE(flags));
2756 void summary_set_marks_selected(SummaryView *summaryview)
2760 for (cur = GTK_CLIST(summaryview->ctree)->selection; cur != NULL;
2762 summary_set_row_marks(summaryview, GTK_CTREE_NODE(cur->data));
2765 static void summary_mark_row(SummaryView *summaryview, GtkCTreeNode *row)
2767 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2770 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2771 if (MSG_IS_DELETED(msginfo->flags))
2772 summaryview->deleted--;
2773 if (MSG_IS_MOVE(msginfo->flags))
2774 summaryview->moved--;
2775 if (MSG_IS_COPY(msginfo->flags))
2776 summaryview->copied--;
2777 procmsg_update_unread_children (msginfo, TRUE);
2779 procmsg_msginfo_set_to_folder(msginfo, NULL);
2780 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_MOVE | MSG_COPY);
2781 procmsg_msginfo_set_flags(msginfo, MSG_MARKED, 0);
2782 summary_set_row_marks(summaryview, row);
2783 debug_print("Message %s/%d is marked\n", msginfo->folder->path, msginfo->msgnum);
2786 static void summary_lock_row(SummaryView *summaryview, GtkCTreeNode *row)
2788 gboolean changed = FALSE;
2789 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2792 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2793 if (MSG_IS_DELETED(msginfo->flags))
2794 summaryview->deleted--;
2795 if (MSG_IS_MOVE(msginfo->flags)) {
2796 summaryview->moved--;
2799 if (MSG_IS_COPY(msginfo->flags)) {
2800 summaryview->copied--;
2803 procmsg_msginfo_set_to_folder(msginfo, NULL);
2804 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, MSG_MOVE | MSG_COPY);
2805 procmsg_msginfo_set_flags(msginfo, MSG_LOCKED, 0);
2806 summary_set_row_marks(summaryview, row);
2807 debug_print("Message %d is locked\n", msginfo->msgnum);
2810 static void summary_unlock_row(SummaryView *summaryview, GtkCTreeNode *row)
2812 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2815 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2816 if (!MSG_IS_LOCKED(msginfo->flags))
2818 procmsg_msginfo_set_to_folder(msginfo, NULL);
2819 procmsg_msginfo_unset_flags(msginfo, MSG_LOCKED, 0);
2820 summary_set_row_marks(summaryview, row);
2821 debug_print("Message %d is unlocked\n", msginfo->msgnum);
2824 void summary_mark(SummaryView *summaryview)
2826 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2829 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2830 summary_mark_row(summaryview, GTK_CTREE_NODE(cur->data));
2832 /* summary_step(summaryview, GTK_SCROLL_STEP_FORWARD); */
2833 summary_status_show(summaryview);
2836 static void summary_mark_row_as_read(SummaryView *summaryview,
2839 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2842 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2844 if(!(MSG_IS_NEW(msginfo->flags) || MSG_IS_UNREAD(msginfo->flags)))
2847 if (MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)
2848 && procmsg_msg_has_marked_parent(msginfo))
2849 summaryview->unreadmarked--;
2851 procmsg_msginfo_unset_flags(msginfo, MSG_NEW | MSG_UNREAD, 0);
2852 summary_set_row_marks(summaryview, row);
2853 debug_print("Message %d is marked as read\n",
2857 void summary_mark_as_read(SummaryView *summaryview)
2859 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2862 folder_item_update_freeze();
2863 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2864 summary_mark_row_as_read(summaryview,
2865 GTK_CTREE_NODE(cur->data));
2866 folder_item_update_thaw();
2868 summary_status_show(summaryview);
2871 void summary_msgs_lock(SummaryView *summaryview)
2873 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2876 folder_item_update_freeze();
2877 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2878 summary_lock_row(summaryview,
2879 GTK_CTREE_NODE(cur->data));
2880 folder_item_update_thaw();
2882 summary_status_show(summaryview);
2885 void summary_msgs_unlock(SummaryView *summaryview)
2887 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2890 folder_item_update_freeze();
2891 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2892 summary_unlock_row(summaryview,
2893 GTK_CTREE_NODE(cur->data));
2894 folder_item_update_thaw();
2896 summary_status_show(summaryview);
2899 void summary_mark_all_read(SummaryView *summaryview)
2901 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2902 GtkCList *clist = GTK_CLIST(summaryview->ctree);
2905 gtk_clist_freeze(clist);
2906 folder_item_update_freeze();
2907 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list); node != NULL;
2908 node = gtkut_ctree_node_next(ctree, node))
2909 summary_mark_row_as_read(summaryview, node);
2910 gtk_clist_thaw(clist);
2911 folder_item_update_thaw();
2913 summary_status_show(summaryview);
2916 static void summary_mark_row_as_unread(SummaryView *summaryview,
2919 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2922 msginfo = gtk_ctree_node_get_row_data(ctree, row);
2923 if (MSG_IS_DELETED(msginfo->flags)) {
2924 procmsg_msginfo_set_to_folder(msginfo, NULL);
2925 procmsg_msginfo_unset_flags(msginfo, MSG_DELETED, 0);
2926 summaryview->deleted--;
2929 if (!MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_IGNORE_THREAD(msginfo->flags)
2930 && procmsg_msg_has_marked_parent(msginfo))
2931 summaryview->unreadmarked++;
2933 procmsg_msginfo_unset_flags(msginfo, MSG_REPLIED | MSG_FORWARDED, 0);
2934 procmsg_msginfo_set_flags(msginfo, MSG_UNREAD, 0);
2935 debug_print("Message %d is marked as unread\n",
2938 summary_set_row_marks(summaryview, row);
2941 void summary_mark_as_unread(SummaryView *summaryview)
2943 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
2946 folder_item_update_freeze();
2947 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
2948 summary_mark_row_as_unread(summaryview,
2949 GTK_CTREE_NODE(cur->data));
2950 folder_item_update_thaw();
2952 summary_status_show(summaryview);
2955 static gboolean check_permission(SummaryView *summaryview, MsgInfo * msginfo)
2960 switch (FOLDER_TYPE(summaryview->folder_item->folder)) {
2965 security : checks if one the accounts correspond to
2966 the author of the post
2970 for(cur = account_get_list() ; cur != NULL ; cur = cur->next) {
2971 PrefsAccount * account;
2974 account = cur->data;
2975 if (account->name && *account->name)
2977 g_strdup_printf("%s <%s>",
2982 g_strdup_printf("%s",
2985 if (g_strcasecmp(from_name, msginfo->from) == 0) {
2994 alertpanel_error(_("You're not the author of the article\n"));
3004 static void summary_delete_row(SummaryView *summaryview, GtkCTreeNode *row)
3006 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3009 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3011 if (!check_permission(summaryview, msginfo))
3014 if (MSG_IS_LOCKED(msginfo->flags)) return;
3016 if (MSG_IS_DELETED(msginfo->flags)) return;
3018 if (MSG_IS_MOVE(msginfo->flags))
3019 summaryview->moved--;
3020 if (MSG_IS_COPY(msginfo->flags))
3021 summaryview->copied--;
3022 procmsg_update_unread_children (msginfo, FALSE);
3024 procmsg_msginfo_set_to_folder(msginfo, NULL);
3025 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED, MSG_MOVE | MSG_COPY);
3026 procmsg_msginfo_set_flags(msginfo, MSG_DELETED, 0);
3027 summaryview->deleted++;
3029 if (!prefs_common.immediate_exec &&
3030 summaryview->folder_item->stype != F_TRASH)
3031 summary_set_row_marks(summaryview, row);
3033 debug_print("Message %s/%d is set to delete\n",
3034 msginfo->folder->path, msginfo->msgnum);
3037 void summary_cancel(SummaryView *summaryview)
3040 GtkCList *clist = GTK_CLIST(summaryview->ctree);
3042 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
3043 summaryview->selected);
3044 if (!msginfo) return;
3046 if (!check_permission(summaryview, msginfo))
3049 news_cancel_article(summaryview->folder_item->folder, msginfo);
3051 if (summary_is_locked(summaryview)) return;
3053 summary_lock(summaryview);
3055 gtk_clist_freeze(clist);
3057 summary_update_status(summaryview);
3058 summary_status_show(summaryview);
3060 gtk_clist_thaw(clist);
3062 summary_unlock(summaryview);
3065 void summary_delete(SummaryView *summaryview)
3067 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3068 FolderItem *item = summaryview->folder_item;
3070 GtkCTreeNode *sel_last = NULL;
3075 if (summary_is_locked(summaryview)) return;
3077 /* if current folder is trash, ask for confirmation */
3078 if (item->stype == F_TRASH) {
3081 aval = alertpanel(_("Delete message(s)"),
3082 _("Do you really want to delete message(s) from the trash?"),
3083 _("Yes"), _("No"), NULL);
3084 if (aval != G_ALERTDEFAULT) return;
3087 main_window_cursor_wait(summaryview->mainwin);
3089 /* next code sets current row focus right. We need to find a row
3090 * that is not deleted. */
3091 folder_item_update_freeze();
3092 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next) {
3093 sel_last = GTK_CTREE_NODE(cur->data);
3094 summary_delete_row(summaryview, sel_last);
3096 folder_item_update_thaw();
3098 node = summary_find_next_msg(summaryview, sel_last);
3100 node = summary_find_prev_msg(summaryview, sel_last);
3102 summary_select_node(summaryview, node, FALSE, FALSE);
3104 if (prefs_common.immediate_exec || item->stype == F_TRASH) {
3105 summary_execute(summaryview);
3106 /* after deleting, the anchor may be at an invalid row
3107 * so reset it to the node we found earlier */
3108 gtk_sctree_set_anchor_row(GTK_SCTREE(ctree), node);
3110 summary_status_show(summaryview);
3112 main_window_cursor_normal(summaryview->mainwin);
3115 void summary_delete_duplicated(SummaryView *summaryview)
3117 if (!summaryview->folder_item ||
3118 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
3119 if (summaryview->folder_item->stype == F_TRASH) return;
3121 main_window_cursor_wait(summaryview->mainwin);
3122 debug_print("Deleting duplicated messages...");
3123 STATUSBAR_PUSH(summaryview->mainwin,
3124 _("Deleting duplicated messages..."));
3126 folder_item_update_freeze();
3128 gtk_ctree_pre_recursive(GTK_CTREE(summaryview->ctree), NULL,
3129 GTK_CTREE_FUNC(summary_delete_duplicated_func),
3132 if (prefs_common.immediate_exec)
3133 summary_execute(summaryview);
3135 summary_status_show(summaryview);
3137 folder_item_update_thaw();
3139 debug_print("done.\n");
3140 STATUSBAR_POP(summaryview->mainwin);
3141 main_window_cursor_normal(summaryview->mainwin);
3144 static void summary_delete_duplicated_func(GtkCTree *ctree, GtkCTreeNode *node,
3145 SummaryView *summaryview)
3147 GtkCTreeNode *found;
3149 MsgInfo *dup_msginfo;
3151 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3153 if (!msginfo || !msginfo->msgid || !*msginfo->msgid) return;
3155 found = g_hash_table_lookup(summaryview->msgid_table, msginfo->msgid);
3157 if (found && found != node) {
3158 dup_msginfo = gtk_ctree_node_get_row_data(ctree, found);
3159 /* prefer to delete the unread one */
3160 if ((MSG_IS_UNREAD(msginfo->flags) && !MSG_IS_UNREAD(dup_msginfo->flags))
3161 || (MSG_IS_UNREAD(msginfo->flags) == MSG_IS_UNREAD(dup_msginfo->flags)))
3162 summary_delete_row(summaryview, node);
3166 static void summary_unmark_row(SummaryView *summaryview, GtkCTreeNode *row)
3168 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3171 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3172 if (MSG_IS_DELETED(msginfo->flags))
3173 summaryview->deleted--;
3174 if (MSG_IS_MOVE(msginfo->flags))
3175 summaryview->moved--;
3176 if (MSG_IS_COPY(msginfo->flags))
3177 summaryview->copied--;
3178 procmsg_update_unread_children (msginfo, FALSE);
3180 procmsg_msginfo_set_to_folder(msginfo, NULL);
3181 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_MOVE | MSG_COPY);
3182 summary_set_row_marks(summaryview, row);
3184 debug_print("Message %s/%d is unmarked\n",
3185 msginfo->folder->path, msginfo->msgnum);
3188 void summary_unmark(SummaryView *summaryview)
3190 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3193 for (cur = GTK_CLIST(ctree)->selection; cur != NULL; cur = cur->next)
3194 summary_unmark_row(summaryview, GTK_CTREE_NODE(cur->data));
3196 summary_status_show(summaryview);
3199 static void summary_move_row_to(SummaryView *summaryview, GtkCTreeNode *row,
3200 FolderItem *to_folder)
3202 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3205 g_return_if_fail(to_folder != NULL);
3207 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3208 procmsg_msginfo_set_to_folder(msginfo, to_folder);
3209 if (MSG_IS_DELETED(msginfo->flags))
3210 summaryview->deleted--;
3211 if (MSG_IS_COPY(msginfo->flags)) {
3212 summaryview->copied--;
3214 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_COPY);
3215 if (!MSG_IS_MOVE(msginfo->flags)) {
3216 procmsg_msginfo_set_flags(msginfo, 0, MSG_MOVE);
3217 summaryview->moved++;
3219 if (!prefs_common.immediate_exec) {
3220 summary_set_row_marks(summaryview, row);
3223 debug_print("Message %d is set to move to %s\n",
3224 msginfo->msgnum, to_folder->path);
3227 void summary_move_selected_to(SummaryView *summaryview, FolderItem *to_folder)
3231 if (!to_folder) return;
3232 if (!summaryview->folder_item ||
3233 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
3235 if (summary_is_locked(summaryview)) return;
3237 if (summaryview->folder_item == to_folder) {
3238 alertpanel_notice(_("Destination is same as current folder."));
3242 for (cur = GTK_CLIST(summaryview->ctree)->selection;
3243 cur != NULL; cur = cur->next)
3245 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
3247 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
3249 if (prefs_common.immediate_exec)
3250 summary_execute(summaryview);
3252 summary_status_show(summaryview);
3255 if (!summaryview->selected) { /* this was the last message */
3256 GtkCTreeNode *node = gtk_ctree_node_nth (GTK_CTREE(summaryview->ctree),
3257 GTK_CLIST(summaryview->ctree)->rows - 1);
3259 summary_select_node(summaryview, node, TRUE, TRUE);
3264 void summary_move_to(SummaryView *summaryview)
3266 FolderItem *to_folder;
3268 if (!summaryview->folder_item ||
3269 FOLDER_TYPE(summaryview->folder_item->folder) == F_NEWS) return;
3271 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
3272 FOLDER_SEL_MOVE, NULL);
3273 summary_move_selected_to(summaryview, to_folder);
3276 static void summary_copy_row_to(SummaryView *summaryview, GtkCTreeNode *row,
3277 FolderItem *to_folder)
3279 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3282 g_return_if_fail(to_folder != NULL);
3284 msginfo = gtk_ctree_node_get_row_data(ctree, row);
3285 procmsg_msginfo_set_to_folder(msginfo, to_folder);
3286 if (MSG_IS_DELETED(msginfo->flags))
3287 summaryview->deleted--;
3288 if (MSG_IS_MOVE(msginfo->flags)) {
3289 summaryview->moved--;
3291 procmsg_msginfo_unset_flags(msginfo, MSG_MARKED | MSG_DELETED, MSG_MOVE);
3292 if (!MSG_IS_COPY(msginfo->flags)) {
3293 procmsg_msginfo_set_flags(msginfo, 0, MSG_COPY);
3294 summaryview->copied++;
3296 if (!prefs_common.immediate_exec) {
3297 summary_set_row_marks(summaryview, row);
3300 debug_print("Message %d is set to copy to %s\n",
3301 msginfo->msgnum, to_folder->path);
3304 void summary_copy_selected_to(SummaryView *summaryview, FolderItem *to_folder)
3308 if (!to_folder) return;
3309 if (!summaryview->folder_item) return;
3311 if (summary_is_locked(summaryview)) return;
3313 if (summaryview->folder_item == to_folder) {
3315 (_("Destination to copy is same as current folder."));
3319 for (cur = GTK_CLIST(summaryview->ctree)->selection;
3320 cur != NULL; cur = cur->next)
3322 (summaryview, GTK_CTREE_NODE(cur->data), to_folder);
3324 summary_step(summaryview, GTK_SCROLL_STEP_FORWARD);
3326 if (prefs_common.immediate_exec)
3327 summary_execute(summaryview);
3329 summary_status_show(summaryview);
3333 void summary_copy_to(SummaryView *summaryview)
3335 FolderItem *to_folder;
3337 if (!summaryview->folder_item) return;
3339 to_folder = foldersel_folder_sel(summaryview->folder_item->folder,
3340 FOLDER_SEL_COPY, NULL);
3341 summary_copy_selected_to(summaryview, to_folder);
3344 void summary_add_address(SummaryView *summaryview)
3349 msginfo = gtk_ctree_node_get_row_data(GTK_CTREE(summaryview->ctree),
3350 summaryview->selected);
3351 if (!msginfo) return;
3353 Xstrdup_a(from, msginfo->from, return);
3354 eliminate_address_comment(from);
3355 extract_address(from);
3356 addressbook_add_contact(msginfo->fromname, from, NULL);
3359 void summary_select_all(SummaryView *summaryview)
3361 if (!summaryview->folder_item) return;
3363 if (summaryview->folder_item->total_msgs >= 500) {
3364 STATUSBAR_PUSH(summaryview->mainwin,
3365 _("Selecting all messages..."));
3366 main_window_cursor_wait(summaryview->mainwin);
3369 gtk_clist_select_all(GTK_CLIST(summaryview->ctree));
3371 if (summaryview->folder_item->total_msgs >= 500) {
3372 STATUSBAR_POP(summaryview->mainwin);
3373 main_window_cursor_normal(summaryview->mainwin);
3377 void summary_unselect_all(SummaryView *summaryview)
3379 gtk_sctree_unselect_all(GTK_SCTREE(summaryview->ctree));
3382 void summary_select_thread(SummaryView *summaryview)
3384 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3385 GtkCTreeNode *node = summaryview->selected;
3389 while (GTK_CTREE_ROW(node)->parent != NULL)
3390 node = GTK_CTREE_ROW(node)->parent;
3392 if (node != summaryview->selected)
3395 messageview_is_visible(summaryview->messageview),
3398 gtk_ctree_select_recursive(ctree, node);
3400 summary_status_show(summaryview);
3403 void summary_save_as(SummaryView *summaryview)
3405 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3407 gchar *filename = NULL;
3410 AlertValue aval = 0;
3412 if (!summaryview->selected) return;
3413 msginfo = gtk_ctree_node_get_row_data(ctree, summaryview->selected);
3414 if (!msginfo) return;
3416 if (msginfo->subject) {
3417 Xstrdup_a(filename, msginfo->subject, return);
3418 subst_for_filename(filename);
3420 dest = filesel_select_file(_("Save as"), filename);
3422 if (is_file_exist(dest)) {
3423 aval = alertpanel(_("Append or Overwrite"),
3424 _("Append or overwrite existing file?"),
3425 _("Append"), _("Overwrite"), _("Cancel"));
3426 if (aval != 0 && aval != 1)
3430 src = procmsg_get_message_file(msginfo);
3431 if ( aval==0 ) { /* append */
3432 if (append_file(src, dest, TRUE) < 0)
3433 alertpanel_error(_("Can't save the file `%s'."),
3435 } else { /* overwrite */
3436 if (copy_file(src, dest, TRUE) < 0)
3437 alertpanel_error(_("Can't save the file `%s'."),
3444 * If two or more msgs are selected,
3445 * append them to the output file.
3447 if (GTK_CLIST(ctree)->selection->next) {
3449 for (item = GTK_CLIST(ctree)->selection->next; item != NULL; item=item->next) {
3450 msginfo = gtk_ctree_node_get_row_data(ctree, (GtkCTreeNode*)item->data);
3451 if (!msginfo) break;
3452 src = procmsg_get_message_file(msginfo);
3453 if (append_file(src, dest, TRUE) < 0)
3454 alertpanel_error(_("Can't save the file `%s'."),
3461 void summary_print(SummaryView *summaryview)
3463 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3464 GtkCList *clist = GTK_CLIST(summaryview->ctree);
3470 if (clist->selection == NULL) return;
3472 cmdline = input_dialog(_("Print"),
3473 _("Enter the print command line:\n"
3474 "(`%s' will be replaced with file name)"),
3475 prefs_common.print_cmd);
3476 if (!cmdline) return;
3477 if (!(p = strchr(cmdline, '%')) || *(p + 1) != 's' ||
3478 strchr(p + 2, '%')) {
3479 alertpanel_error(_("Print command line is invalid:\n`%s'"),
3485 for (cur = clist->selection; cur != NULL; cur = cur->next) {
3486 msginfo = gtk_ctree_node_get_row_data
3487 (ctree, GTK_CTREE_NODE(cur->data));
3488 if (msginfo) procmsg_print_message(msginfo, cmdline);
3494 gboolean summary_execute(SummaryView *summaryview)
3496 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3497 GtkCList *clist = GTK_CLIST(summaryview->ctree);
3498 GtkCTreeNode *node, *next;
3499 GtkCTreeNode *new_selected = NULL;
3501 if (!summaryview->folder_item) return FALSE;
3503 if (summary_is_locked(summaryview)) return FALSE;
3504 summary_lock(summaryview);
3506 gtk_clist_freeze(clist);
3508 if (summaryview->threaded)
3509 summary_unthread_for_exec(summaryview);
3511 folder_item_update_freeze();
3512 summary_execute_move(summaryview);
3513 summary_execute_copy(summaryview);
3514 summary_execute_delete(summaryview);
3515 folder_item_update_thaw();
3517 node = GTK_CTREE_NODE(clist->row_list);
3518 for (; node != NULL; node = next) {
3519 next = gtkut_ctree_node_next(ctree, node);
3520 if (gtk_ctree_node_get_row_data(ctree, node) != NULL) continue;
3522 if (node == summaryview->displayed) {
3523 messageview_clear(summaryview->messageview);
3524 summaryview->displayed = NULL;
3526 if (GTK_CTREE_ROW(node)->children != NULL) {
3527 g_warning("summary_execute(): children != NULL\n");
3531 if (!new_selected &&
3532 gtkut_ctree_node_is_selected(ctree, node)) {
3533 gtk_sctree_unselect_all(GTK_SCTREE(ctree));
3534 new_selected = summary_find_next_msg(summaryview, node);
3536 new_selected = summary_find_prev_msg
3537 (summaryview, node);
3540 gtk_ctree_remove_node(ctree, node);
3546 summaryview->displayed ? summaryview->displayed
3550 if (summaryview->threaded)
3551 summary_thread_build(summaryview);
3553 summaryview->selected = clist->selection ?
3554 GTK_CTREE_NODE(clist->selection->data) : NULL;
3556 if (!GTK_CLIST(summaryview->ctree)->row_list) {
3557 menu_set_insensitive_all
3558 (GTK_MENU_SHELL(summaryview->popupmenu));
3559 gtk_widget_grab_focus(summaryview->folderview->ctree);
3561 gtk_widget_grab_focus(summaryview->ctree);
3563 summary_update_status(summaryview);
3564 summary_status_show(summaryview);
3566 gtk_ctree_node_moveto(ctree, summaryview->selected, -1, 0.5, 0);
3568 gtk_clist_thaw(clist);
3570 summary_unlock(summaryview);
3574 static void summary_execute_move(SummaryView *summaryview)
3576 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3579 /* search moving messages and execute */
3580 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_move_func,
3583 if (summaryview->mlist) {
3584 procmsg_move_messages(summaryview->mlist);
3586 for (cur = summaryview->mlist; cur != NULL; cur = cur->next)
3587 procmsg_msginfo_free((MsgInfo *)cur->data);
3588 g_slist_free(summaryview->mlist);
3589 summaryview->mlist = NULL;
3593 static void summary_execute_move_func(GtkCTree *ctree, GtkCTreeNode *node,
3596 SummaryView *summaryview = data;
3599 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3601 if (msginfo && MSG_IS_MOVE(msginfo->flags) && msginfo->to_folder) {
3602 summaryview->mlist =
3603 g_slist_prepend(summaryview->mlist, msginfo);
3604 gtk_ctree_node_set_row_data(ctree, node, NULL);
3606 if (msginfo->msgid && *msginfo->msgid &&
3607 node == g_hash_table_lookup(summaryview->msgid_table,
3609 g_hash_table_remove(summaryview->msgid_table,
3614 static void summary_execute_copy(SummaryView *summaryview)
3616 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3618 /* search copying messages and execute */
3619 gtk_ctree_pre_recursive(ctree, NULL, summary_execute_copy_func,
3622 if (summaryview->mlist) {
3623 summaryview->mlist = g_slist_reverse(summaryview->mlist);
3624 procmsg_copy_messages(summaryview->mlist);
3626 g_slist_free(summaryview->mlist);
3627 summaryview->mlist = NULL;
3631 static void summary_execute_copy_func(GtkCTree *ctree, GtkCTreeNode *node,
3634 SummaryView *summaryview = data;
3637 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3639 if (msginfo && MSG_IS_COPY(msginfo->flags) && msginfo->to_folder) {
3640 summaryview->mlist =
3641 g_slist_prepend(summaryview->mlist, msginfo);
3643 procmsg_msginfo_unset_flags(msginfo, 0, MSG_COPY);
3644 summary_set_row_marks(summaryview, node);
3648 static void summary_execute_delete(SummaryView *summaryview)
3650 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3654 trash = summaryview->folder_item->folder->trash;
3656 /* search deleting messages and execute */
3657 gtk_ctree_pre_recursive
3658 (ctree, NULL, summary_execute_delete_func, summaryview);
3660 if (!summaryview->mlist) return;
3662 if (trash == NULL || summaryview->folder_item == trash)
3663 folder_item_remove_msgs(summaryview->folder_item,
3664 summaryview->mlist);
3666 folder_item_move_msgs(trash, summaryview->mlist);
3668 for (cur = summaryview->mlist; cur != NULL; cur = cur->next)
3669 procmsg_msginfo_free((MsgInfo *)cur->data);
3671 g_slist_free(summaryview->mlist);
3672 summaryview->mlist = NULL;
3675 static void summary_execute_delete_func(GtkCTree *ctree, GtkCTreeNode *node,
3678 SummaryView *summaryview = data;
3681 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3683 if (msginfo && MSG_IS_DELETED(msginfo->flags)) {
3684 summaryview->mlist =
3685 g_slist_append(summaryview->mlist, msginfo);
3686 gtk_ctree_node_set_row_data(ctree, node, NULL);
3688 if (msginfo->msgid && *msginfo->msgid &&
3689 node == g_hash_table_lookup(summaryview->msgid_table,
3691 g_hash_table_remove(summaryview->msgid_table,
3694 if (msginfo->subject && *msginfo->subject &&
3695 node == subject_table_lookup(summaryview->subject_table,
3696 msginfo->subject)) {
3697 subject_table_remove(summaryview->subject_table,
3703 /* thread functions */
3705 void summary_thread_build(SummaryView *summaryview)
3707 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3710 GtkCTreeNode *parent;
3713 summary_lock(summaryview);
3715 debug_print("Building threads...");
3716 STATUSBAR_PUSH(summaryview->mainwin, _("Building threads..."));
3717 main_window_cursor_wait(summaryview->mainwin);
3719 gtk_signal_handler_block_by_func(GTK_OBJECT(ctree),
3720 summary_tree_expanded, summaryview);
3721 gtk_clist_freeze(GTK_CLIST(ctree));
3723 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3725 next = GTK_CTREE_ROW(node)->sibling;
3727 msginfo = GTKUT_CTREE_NODE_GET_ROW_DATA(node);
3731 /* alfons - claws seems to prefer subject threading before
3732 * inreplyto threading. we should look more deeply in this,
3733 * because inreplyto should have precedence... */
3734 if (msginfo && msginfo->inreplyto) {
3735 parent = g_hash_table_lookup(summaryview->msgid_table,
3736 msginfo->inreplyto);
3739 if (prefs_common.thread_by_subject && parent == NULL) {
3740 parent = subject_table_lookup
3741 (summaryview->subject_table,
3745 if (parent && parent != node) {
3746 gtk_ctree_move(ctree, node, parent, NULL);
3747 gtk_ctree_expand(ctree, node);
3753 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3756 next = GTK_CTREE_NODE_NEXT(node);
3757 if (!summaryview->thread_collapsed)
3758 gtk_ctree_expand(ctree, node);
3759 if (prefs_common.bold_unread &&
3760 GTK_CTREE_ROW(node)->children)
3761 summary_set_row_marks(summaryview, node);
3765 gtkut_ctree_set_focus_row(ctree, summaryview->selected);
3767 gtk_clist_thaw(GTK_CLIST(ctree));
3768 gtk_signal_handler_unblock_by_func(GTK_OBJECT(ctree),
3769 summary_tree_expanded, summaryview);
3771 debug_print("done.\n");
3772 STATUSBAR_POP(summaryview->mainwin);
3773 main_window_cursor_normal(summaryview->mainwin);
3775 summaryview->threaded = TRUE;
3777 summary_unlock(summaryview);
3780 static void summary_thread_init(SummaryView *summaryview)
3782 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3783 GtkCTreeNode *node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3786 if (!summaryview->thread_collapsed) {
3788 next = GTK_CTREE_ROW(node)->sibling;
3789 if (GTK_CTREE_ROW(node)->children)
3790 gtk_ctree_expand(ctree, node);
3793 } else if (prefs_common.bold_unread) {
3795 next = GTK_CTREE_ROW(node)->sibling;
3796 if (GTK_CTREE_ROW(node)->children)
3797 summary_set_row_marks(summaryview, node);
3803 void summary_unthread(SummaryView *summaryview)
3805 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);
3807 GtkCTreeNode *child;
3808 GtkCTreeNode *sibling;
3809 GtkCTreeNode *next_child;
3811 summary_lock(summaryview);
3813 debug_print("Unthreading...");
3814 STATUSBAR_PUSH(summaryview->mainwin, _("Unthreading..."));
3815 main_window_cursor_wait(summaryview->mainwin);
3817 gtk_signal_handler_block_by_func(GTK_OBJECT(ctree),
3818 summary_tree_collapsed, summaryview);
3819 gtk_clist_freeze(GTK_CLIST(ctree));
3821 for (node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
3822 node != NULL; node = GTK_CTREE_NODE_NEXT(node)) {
3823 child = GTK_CTREE_ROW(node)->children;
3824 sibling = GTK_CTREE_ROW(node)->sibling;
3826 while (child != NULL) {
3827 next_child = GTK_CTREE_ROW(child)->sibling;
3828 gtk_ctree_move(ctree, child, NULL, sibling);
3833 /* CLAWS: and sort it */
3834 gtk_sctree_sort_recursive(ctree, NULL);
3836 gtk_clist_thaw(GTK_CLIST(ctree));
3837 gtk_signal_handler_unblock_by_func(GTK_OBJECT(ctree),
3838 summary_tree_collapsed, summaryview);
3840 debug_print("done.\n");
3841 STATUSBAR_POP(summaryview->mainwin);
3842 main_window_cursor_normal(summaryview->mainwin);
3844 summaryview->threaded = FALSE;
3846 summary_unlock(summaryview);
3849 static void summary_unthread_for_exec(SummaryView *summaryview)
3852 GtkCTree *ctree = GTK_CTREE(summaryview->ctree);