e23ac4caefcc302cb2444383f2976af80f807194
[claws.git] / src / folderview.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2001 Hiroyuki Yamamoto
4  *
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.
9  *
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.
14  *
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.
18  */
19
20 #include "defs.h"
21
22 #include <glib.h>
23 #include <gdk/gdkkeysyms.h>
24 #include <gtk/gtkwidget.h>
25 #include <gtk/gtkscrolledwindow.h>
26 #include <gtk/gtkctree.h>
27 #include <gtk/gtkcontainer.h>
28 #include <gtk/gtkclist.h>
29 #include <gtk/gtkstyle.h>
30 #include <gtk/gtksignal.h>
31 #include <gtk/gtkmain.h>
32 #include <gtk/gtkstatusbar.h>
33 #include <gtk/gtkmenu.h>
34 #include <gtk/gtkmenuitem.h>
35 #include <gtk/gtkitemfactory.h>
36 #include <stdio.h>
37 #include <string.h>
38 #include <stdlib.h>
39
40 #include "intl.h"
41 #include "main.h"
42 #include "mainwindow.h"
43 #include "folderview.h"
44 #include "summaryview.h"
45 #include "inputdialog.h"
46 #include "manage_window.h"
47 #include "alertpanel.h"
48 #include "menu.h"
49 #include "procmsg.h"
50 #include "utils.h"
51 #include "gtkutils.h"
52 #include "prefs_common.h"
53 #include "prefs_account.h"
54 #include "account.h"
55 #include "folder.h"
56 #include "grouplist_dialog.h"
57
58 #include "pixmaps/inbox.xpm"
59 #include "pixmaps/outbox.xpm"
60 #include "pixmaps/dir-close.xpm"
61 #include "pixmaps/dir-open.xpm"
62 #include "pixmaps/trash.xpm"
63
64 typedef enum
65 {
66         COL_FOLDER      = 0,
67         COL_NEW         = 1,
68         COL_UNREAD      = 2,
69         COL_TOTAL       = 3
70 } FolderColumnPos;
71
72 #define N_FOLDER_COLS           4
73 #define COL_FOLDER_WIDTH        150
74 #define COL_NUM_WIDTH           32
75
76 #define STATUSBAR_PUSH(mainwin, str) \
77 { \
78         gtk_statusbar_push(GTK_STATUSBAR(mainwin->statusbar), \
79                            mainwin->folderview_cid, str); \
80         gtkut_widget_wait_for_draw(mainwin->hbox_stat); \
81 }
82
83 #define STATUSBAR_POP(mainwin) \
84 { \
85         gtk_statusbar_pop(GTK_STATUSBAR(mainwin->statusbar), \
86                           mainwin->folderview_cid); \
87 }
88
89 static GList *folderview_list = NULL;
90
91 static GdkFont *normalfont;
92 static GdkFont *boldfont;
93
94 static GdkPixmap *inboxxpm;
95 static GdkBitmap *inboxxpmmask;
96 static GdkPixmap *outboxxpm;
97 static GdkBitmap *outboxxpmmask;
98 static GdkPixmap *folderxpm;
99 static GdkBitmap *folderxpmmask;
100 static GdkPixmap *folderopenxpm;
101 static GdkBitmap *folderopenxpmmask;
102 static GdkPixmap *trashxpm;
103 static GdkBitmap *trashxpmmask;
104
105 static void folderview_select_node       (FolderView    *folderview,
106                                           GtkCTreeNode  *node);
107 static void folderview_set_folders       (FolderView    *folderview);
108 static void folderview_sort_folders      (FolderView    *folderview,
109                                           GtkCTreeNode  *root,
110                                           Folder        *folder);
111 static void folderview_append_folder     (FolderView    *folderview,
112                                           Folder        *folder);
113 static void folderview_update_node       (FolderView    *folderview,
114                                           GtkCTreeNode  *node);
115
116 static GtkCTreeNode *folderview_find_by_name    (GtkCTree       *ctree,
117                                                  GtkCTreeNode   *node,
118                                                  const gchar    *name);
119
120 static gint folderview_compare_name     (gconstpointer   a,
121                                          gconstpointer   b);
122
123 /* callback functions */
124 static void folderview_button_pressed   (GtkWidget      *ctree,
125                                          GdkEventButton *event,
126                                          FolderView     *folderview);
127 static void folderview_button_released  (GtkWidget      *ctree,
128                                          GdkEventButton *event,
129                                          FolderView     *folderview);
130 static void folderview_key_pressed      (GtkWidget      *widget,
131                                          GdkEventKey    *event,
132                                          FolderView     *folderview);
133 static void folderview_selected         (GtkCTree       *ctree,
134                                          GtkCTreeNode   *row,
135                                          gint            column,
136                                          FolderView     *folderview);
137 static void folderview_tree_expanded    (GtkCTree       *ctree,
138                                          GtkCTreeNode   *node,
139                                          FolderView     *folderview);
140 static void folderview_tree_collapsed   (GtkCTree       *ctree,
141                                          GtkCTreeNode   *node,
142                                          FolderView     *folderview);
143 static void folderview_popup_close      (GtkMenuShell   *menu_shell,
144                                          FolderView     *folderview);
145 static void folderview_col_resized      (GtkCList       *clist,
146                                          gint            column,
147                                          gint            width,
148                                          FolderView     *folderview);
149
150 static void folderview_new_folder_cb    (FolderView     *folderview,
151                                          guint           action,
152                                          GtkWidget      *widget);
153 static void folderview_new_mbox_folder_cb(FolderView *folderview,
154                                           guint action,
155                                           GtkWidget *widget);
156 static void folderview_rename_folder_cb (FolderView     *folderview,
157                                          guint           action,
158                                          GtkWidget      *widget);
159 static void folderview_rename_mbox_folder_cb(FolderView *folderview,
160                                              guint action,
161                                              GtkWidget *widget);
162 static void folderview_delete_folder_cb (FolderView     *folderview,
163                                          guint           action,
164                                          GtkWidget      *widget);
165 static void folderview_remove_mailbox_cb(FolderView     *folderview,
166                                          guint           action,
167                                          GtkWidget      *widget);
168
169 static void folderview_new_imap_folder_cb(FolderView    *folderview,
170                                           guint          action,
171                                           GtkWidget     *widget);
172 static void folderview_rm_imap_folder_cb (FolderView    *folderview,
173                                           guint          action,
174                                           GtkWidget     *widget);
175 static void folderview_rm_imap_server_cb (FolderView    *folderview,
176                                           guint          action,
177                                           GtkWidget     *widget);
178
179 static void folderview_new_news_group_cb(FolderView     *folderview,
180                                          guint           action,
181                                          GtkWidget      *widget);
182 static void folderview_rm_news_group_cb (FolderView     *folderview,
183                                          guint           action,
184                                          GtkWidget      *widget);
185 static void folderview_rm_news_server_cb(FolderView     *folderview,
186                                          guint           action,
187                                          GtkWidget      *widget);
188
189 static gboolean folderview_drag_motion_cb(GtkWidget      *widget,
190                                           GdkDragContext *context,
191                                           gint            x,
192                                           gint            y,
193                                           guint           time,
194                                           FolderView     *folderview);
195 static void folderview_drag_leave_cb     (GtkWidget        *widget,
196                                           GdkDragContext   *context,
197                                           guint             time,
198                                           FolderView       *folderview);
199 static void folderview_drag_received_cb  (GtkWidget        *widget,
200                                           GdkDragContext   *drag_context,
201                                           gint              x,
202                                           gint              y,
203                                           GtkSelectionData *data,
204                                           guint             info,
205                                           guint             time,
206                                           FolderView       *folderview);
207 static void folderview_scoring_cb(FolderView *folderview, guint action,
208                                   GtkWidget *widget);
209
210 static GtkItemFactoryEntry folderview_mbox_popup_entries[] =
211 {
212         {N_("/Create _new folder..."),  NULL, folderview_new_mbox_folder_cb,    0, NULL},
213         {N_("/_Rename folder..."),      NULL, folderview_rename_mbox_folder_cb, 0, NULL},
214         {N_("/_Delete folder"),         NULL, folderview_delete_folder_cb, 0, NULL},
215         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
216         {N_("/Remove _mailbox"),        NULL, folderview_remove_mailbox_cb, 0, NULL},
217         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
218         {N_("/_Property..."),           NULL, NULL, 0, NULL},
219         {N_("/_Scoring..."),            NULL, folderview_scoring_cb, 0, NULL}
220 };
221
222 static GtkItemFactoryEntry folderview_mail_popup_entries[] =
223 {
224         {N_("/Create _new folder..."),  NULL, folderview_new_folder_cb,    0, NULL},
225         {N_("/_Rename folder..."),      NULL, folderview_rename_folder_cb, 0, NULL},
226         {N_("/_Delete folder"),         NULL, folderview_delete_folder_cb, 0, NULL},
227         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
228         {N_("/Remove _mailbox"),        NULL, folderview_remove_mailbox_cb, 0, NULL},
229         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
230         {N_("/_Property..."),           NULL, NULL, 0, NULL},
231         {N_("/_Scoring..."),            NULL, folderview_scoring_cb, 0, NULL}
232 };
233
234 static GtkItemFactoryEntry folderview_imap_popup_entries[] =
235 {
236         {N_("/Create _new folder..."),  NULL, folderview_new_imap_folder_cb, 0, NULL},
237         {N_("/_Rename folder..."),      NULL, NULL, 0, NULL},
238         {N_("/_Delete folder"),         NULL, folderview_rm_imap_folder_cb, 0, NULL},
239         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
240         {N_("/Remove _IMAP4 server"),   NULL, folderview_rm_imap_server_cb, 0, NULL},
241         {N_("/---"),                    NULL, NULL, 0, "<Separator>"},
242         {N_("/_Property..."),           NULL, NULL, 0, NULL},
243         {N_("/_Scoring..."),            NULL, folderview_scoring_cb, 0, NULL}
244 };
245
246 static GtkItemFactoryEntry folderview_news_popup_entries[] =
247 {
248         {N_("/_Subscribe to newsgroup..."),
249                                          NULL, folderview_new_news_group_cb, 0, NULL},
250         {N_("/_Remove newsgroup"),       NULL, folderview_rm_news_group_cb, 0, NULL},
251         {N_("/---"),                     NULL, NULL, 0, "<Separator>"},
252         {N_("/Remove _news server"),     NULL, folderview_rm_news_server_cb, 0, NULL},
253         {N_("/---"),                     NULL, NULL, 0, "<Separator>"},
254         {N_("/_Property..."),            NULL, NULL, 0, NULL},
255         {N_("/_Scoring..."),            NULL, folderview_scoring_cb, 0, NULL}
256 };
257
258
259 FolderView *folderview_create(void)
260 {
261         FolderView *folderview;
262         GtkWidget *scrolledwin;
263         GtkWidget *ctree;
264         gchar *titles[N_FOLDER_COLS] = {_("Folder"), _("New"),
265                                         _("Unread"), _("#")};
266         GtkWidget *mail_popup;
267         GtkWidget *news_popup;
268         GtkWidget *imap_popup;
269         GtkWidget *mbox_popup;
270         GtkItemFactory *mail_factory;
271         GtkItemFactory *news_factory;
272         GtkItemFactory *imap_factory;
273         GtkItemFactory *mbox_factory;
274         gint n_entries;
275         gint i;
276
277         debug_print(_("Creating folder view...\n"));
278         folderview = g_new0(FolderView, 1);
279
280         scrolledwin = gtk_scrolled_window_new(NULL, NULL);
281         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
282                                        GTK_POLICY_AUTOMATIC,
283                                        GTK_POLICY_ALWAYS);
284         gtk_widget_set_usize(scrolledwin,
285                              prefs_common.folderview_width,
286                              prefs_common.folderview_height);
287
288         ctree = gtk_ctree_new_with_titles(N_FOLDER_COLS, COL_FOLDER, titles);
289         gtk_container_add(GTK_CONTAINER(scrolledwin), ctree);
290         gtk_clist_set_selection_mode(GTK_CLIST(ctree), GTK_SELECTION_BROWSE);
291         gtk_clist_set_column_justification(GTK_CLIST(ctree), COL_NEW,
292                                            GTK_JUSTIFY_RIGHT);
293         gtk_clist_set_column_justification(GTK_CLIST(ctree), COL_UNREAD,
294                                            GTK_JUSTIFY_RIGHT);
295         gtk_clist_set_column_justification(GTK_CLIST(ctree), COL_TOTAL,
296                                            GTK_JUSTIFY_RIGHT);
297         gtk_clist_set_column_width(GTK_CLIST(ctree), COL_FOLDER,
298                                    prefs_common.folder_col_folder);
299         gtk_clist_set_column_width(GTK_CLIST(ctree), COL_NEW,
300                                    prefs_common.folder_col_new);
301         gtk_clist_set_column_width(GTK_CLIST(ctree), COL_UNREAD,        
302                                    prefs_common.folder_col_unread);
303         gtk_clist_set_column_width(GTK_CLIST(ctree), COL_TOTAL,
304                                    prefs_common.folder_col_total);
305         gtk_ctree_set_line_style(GTK_CTREE(ctree), GTK_CTREE_LINES_DOTTED);
306         gtk_ctree_set_expander_style(GTK_CTREE(ctree),
307                                      GTK_CTREE_EXPANDER_SQUARE);
308         gtk_ctree_set_indent(GTK_CTREE(ctree), CTREE_INDENT);
309         
310         /* don't let title buttons take key focus */
311         for (i = 0; i < N_FOLDER_COLS; i++)
312                 GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(ctree)->column[i].button,
313                                        GTK_CAN_FOCUS);
314
315         /* popup menu */
316         n_entries = sizeof(folderview_mail_popup_entries) /
317                 sizeof(folderview_mail_popup_entries[0]);
318         mail_popup = menu_create_items(folderview_mail_popup_entries,
319                                        n_entries,
320                                        "<MailFolder>", &mail_factory,
321                                        folderview);
322         n_entries = sizeof(folderview_imap_popup_entries) /
323                 sizeof(folderview_imap_popup_entries[0]);
324         imap_popup = menu_create_items(folderview_imap_popup_entries,
325                                        n_entries,
326                                        "<IMAPFolder>", &imap_factory,
327                                        folderview);
328         n_entries = sizeof(folderview_news_popup_entries) /
329                 sizeof(folderview_news_popup_entries[0]);
330         news_popup = menu_create_items(folderview_news_popup_entries,
331                                        n_entries,
332                                        "<NewsFolder>", &news_factory,
333                                        folderview);
334         n_entries = sizeof(folderview_mbox_popup_entries) /
335                 sizeof(folderview_mbox_popup_entries[0]);
336         mbox_popup = menu_create_items(folderview_mbox_popup_entries,
337                                        n_entries,
338                                        "<MailFolder>", &mbox_factory,
339                                        folderview);
340
341         gtk_signal_connect(GTK_OBJECT(ctree), "key_press_event",
342                            GTK_SIGNAL_FUNC(folderview_key_pressed),
343                            folderview);
344         gtk_signal_connect(GTK_OBJECT(ctree), "button_press_event",
345                            GTK_SIGNAL_FUNC(folderview_button_pressed),
346                            folderview);
347         gtk_signal_connect(GTK_OBJECT(ctree), "button_release_event",
348                            GTK_SIGNAL_FUNC(folderview_button_released),
349                            folderview);
350         gtk_signal_connect(GTK_OBJECT(ctree), "tree_select_row",
351                            GTK_SIGNAL_FUNC(folderview_selected), folderview);
352
353
354         gtk_signal_connect_after(GTK_OBJECT(ctree), "tree_expand",
355                                  GTK_SIGNAL_FUNC(folderview_tree_expanded),
356                                  folderview);
357         gtk_signal_connect_after(GTK_OBJECT(ctree), "tree_collapse",
358                                  GTK_SIGNAL_FUNC(folderview_tree_collapsed),
359                                  folderview);
360
361         gtk_signal_connect(GTK_OBJECT(ctree), "resize_column",
362                            GTK_SIGNAL_FUNC(folderview_col_resized),
363                            folderview);
364
365         gtk_signal_connect(GTK_OBJECT(mail_popup), "selection_done",
366                            GTK_SIGNAL_FUNC(folderview_popup_close),
367                            folderview);
368         gtk_signal_connect(GTK_OBJECT(imap_popup), "selection_done",
369                            GTK_SIGNAL_FUNC(folderview_popup_close),
370                            folderview);
371         gtk_signal_connect(GTK_OBJECT(news_popup), "selection_done",
372                            GTK_SIGNAL_FUNC(folderview_popup_close),
373                            folderview);
374         gtk_signal_connect(GTK_OBJECT(mbox_popup), "selection_done",
375                            GTK_SIGNAL_FUNC(folderview_popup_close),
376                            folderview);
377
378         /* drop callback */
379         gtk_drag_dest_set(ctree, GTK_DEST_DEFAULT_ALL &
380                           ~GTK_DEST_DEFAULT_HIGHLIGHT,
381                           summary_drag_types, 1,
382                           GDK_ACTION_MOVE);
383         gtk_signal_connect(GTK_OBJECT(ctree), "drag_motion",
384                            GTK_SIGNAL_FUNC(folderview_drag_motion_cb),
385                            folderview);
386         gtk_signal_connect(GTK_OBJECT(ctree), "drag_leave",
387                            GTK_SIGNAL_FUNC(folderview_drag_leave_cb),
388                            folderview);
389         gtk_signal_connect(GTK_OBJECT(ctree), "drag_data_received",
390                            GTK_SIGNAL_FUNC(folderview_drag_received_cb),
391                            folderview);
392
393         folderview->scrolledwin  = scrolledwin;
394         folderview->ctree        = ctree;
395         folderview->mail_popup   = mail_popup;
396         folderview->mail_factory = mail_factory;
397         folderview->imap_popup   = imap_popup;
398         folderview->imap_factory = imap_factory;
399         folderview->news_popup   = news_popup;
400         folderview->news_factory = news_factory;
401         folderview->mbox_popup   = mbox_popup;
402         folderview->mbox_factory = mbox_factory;
403
404         gtk_widget_show_all(scrolledwin);
405
406         folderview_list = g_list_append(folderview_list, folderview);
407
408         return folderview;
409 }
410
411 void folderview_init(FolderView *folderview)
412 {
413         GtkWidget *ctree = folderview->ctree;
414
415         PIXMAP_CREATE(ctree, inboxxpm, inboxxpmmask, inbox_xpm);
416         PIXMAP_CREATE(ctree, outboxxpm, outboxxpmmask, outbox_xpm);
417         PIXMAP_CREATE(ctree, folderxpm, folderxpmmask, DIRECTORY_CLOSE_XPM);
418         PIXMAP_CREATE(ctree, folderopenxpm, folderopenxpmmask,
419                       DIRECTORY_OPEN_XPM);
420         PIXMAP_CREATE(ctree, trashxpm, trashxpmmask, trash_xpm);
421
422         if (!normalfont)
423                 normalfont = gdk_fontset_load(NORMAL_FONT);
424         if (!boldfont)
425                 boldfont = gdk_fontset_load(BOLD_FONT);
426 }
427
428 void folderview_set(FolderView *folderview)
429 {
430         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
431         MainWindow *mainwin = folderview->mainwin;
432
433         debug_print(_("Setting folder info...\n"));
434         STATUSBAR_PUSH(mainwin, _("Setting folder info..."));
435
436         main_window_cursor_wait(mainwin);
437
438         folderview->selected = NULL;
439         folderview->opened = NULL;
440
441         gtk_clist_freeze(GTK_CLIST(ctree));
442         gtk_clist_clear(GTK_CLIST(ctree));
443         gtk_clist_thaw(GTK_CLIST(ctree));
444         gtk_clist_freeze(GTK_CLIST(ctree));
445
446         folderview_set_folders(folderview);
447
448         gtk_clist_thaw(GTK_CLIST(ctree));
449         main_window_cursor_normal(mainwin);
450         STATUSBAR_POP(mainwin);
451 }
452
453 void folderview_select(FolderView *folderview, FolderItem *item)
454 {
455         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
456         GtkCTreeNode *node;
457
458         if (!item) return;
459
460         node = gtk_ctree_find_by_row_data(ctree, NULL, item);
461         if (node) folderview_select_node(folderview, node);
462 }
463
464 static void folderview_select_node(FolderView *folderview, GtkCTreeNode *node)
465 {
466         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
467
468         g_return_if_fail(node != NULL);
469
470         folderview->open_folder = TRUE;
471         gtk_ctree_select(ctree, node);
472         gtkut_ctree_set_focus_row(ctree, node);
473         if (folderview->summaryview->messages > 0)
474                 gtk_widget_grab_focus(folderview->summaryview->ctree);
475         else
476                 gtk_widget_grab_focus(folderview->ctree);
477
478         while ((node = gtkut_ctree_find_collapsed_parent(ctree, node))
479                != NULL)
480                 gtk_ctree_expand(ctree, node);
481 }
482
483 void folderview_unselect(FolderView *folderview)
484 {
485         if (folderview->opened && !GTK_CTREE_ROW(folderview->opened)->children)
486                 gtk_ctree_collapse
487                         (GTK_CTREE(folderview->ctree), folderview->opened);
488
489         folderview->selected = folderview->opened = NULL;
490 }
491
492 static GtkCTreeNode *folderview_find_next_unread(GtkCTree *ctree,
493                                                  GtkCTreeNode *node)
494 {
495         FolderItem *item;
496
497         if (node)
498                 node = gtkut_ctree_node_next(ctree, node);
499         else
500                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
501
502         for (; node != NULL; node = gtkut_ctree_node_next(ctree, node)) {
503                 item = gtk_ctree_node_get_row_data(ctree, node);
504                 if (item && item->unread > 0 && item->stype != F_TRASH)
505                         return node;
506         }
507
508         return NULL;
509 }
510
511 void folderview_select_next_unread(FolderView *folderview)
512 {
513         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
514         GtkCTreeNode *node = NULL;
515
516         if ((node = folderview_find_next_unread(ctree, folderview->opened))
517             != NULL) {
518                 folderview_select_node(folderview, node);
519                 return;
520         }
521
522         if (!folderview->opened ||
523             folderview->opened == GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list))
524                 return;
525         /* search again from the first node */
526         if ((node = folderview_find_next_unread(ctree, NULL)) != NULL)
527                 folderview_select_node(folderview, node);
528 }
529
530 void folderview_update_msg_num(FolderView *folderview, GtkCTreeNode *row,
531                                gint new, gint unread, gint total)
532 {
533         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
534         static GtkCTreeNode *prev_row = NULL;
535         FolderItem *item;
536
537         if (!row) return;
538
539         item = gtk_ctree_node_get_row_data(ctree, row);
540         if (!item) return;
541         if (prev_row     == row    &&
542             item->new    == new    &&
543             item->unread == unread &&
544             item->total  == total)
545                 return;
546
547         prev_row = row;
548
549         item->new    = new;
550         item->unread = unread;
551         item->total  = total;
552
553         folderview_update_node(folderview, row);
554 }
555
556 static void folderview_set_folders(FolderView *folderview)
557 {
558         GList *list;
559
560         list = folder_get_list();
561
562         for (; list != NULL; list = list->next)
563                 folderview_append_folder(folderview, FOLDER(list->data));
564 }
565
566 static void folderview_scan_tree_func(Folder *folder, FolderItem *item,
567                                       gpointer data)
568 {
569         GList *list;
570
571         for (list = folderview_list; list != NULL; list = list->next) {
572                 FolderView *folderview = (FolderView *)list->data;
573                 MainWindow *mainwin = folderview->mainwin;
574                 gchar *str;
575
576                 if (item->path)
577                         str = g_strdup_printf(_("Scanning folder %s%c%s ..."),
578                                               LOCAL_FOLDER(folder)->rootpath,
579                                               G_DIR_SEPARATOR,
580                                               item->path);
581                 else
582                         str = g_strdup_printf(_("Scanning folder %s ..."),
583                                               LOCAL_FOLDER(folder)->rootpath);
584
585                 STATUSBAR_PUSH(mainwin, str);
586                 STATUSBAR_POP(mainwin);
587                 g_free(str);
588         }
589 }
590
591 static GtkWidget *label_window_create(const gchar *str)
592 {
593         GtkWidget *window;
594         GtkWidget *label;
595
596         window = gtk_window_new(GTK_WINDOW_DIALOG);
597         gtk_widget_set_usize(window, 380, 60);
598         gtk_container_set_border_width(GTK_CONTAINER(window), 8);
599         gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
600         gtk_window_set_title(GTK_WINDOW(window), str);
601         gtk_window_set_modal(GTK_WINDOW(window), TRUE);
602         gtk_window_set_policy(GTK_WINDOW(window), FALSE, FALSE, FALSE);
603         manage_window_set_transient(GTK_WINDOW(window));
604
605         label = gtk_label_new(str);
606         gtk_container_add(GTK_CONTAINER(window), label);
607         gtk_widget_show(label);
608
609         gtk_widget_show_now(window);
610
611         return window;
612 }
613
614 void folderview_update_all(void)
615 {
616         GList *list;
617         GtkWidget *window;
618
619         window = label_window_create(_("Updating all folders..."));
620
621         list = folder_get_list();
622         for (; list != NULL; list = list->next) {
623                 Folder *folder = list->data;
624
625                 if (!folder->scan_tree) continue;
626                 folder_set_ui_func(folder, folderview_scan_tree_func, NULL);
627                 folder->scan_tree(folder);
628                 folder_set_ui_func(folder, NULL, NULL);
629         }
630
631         folder_write_list();
632
633         for (list = folderview_list; list != NULL; list = list->next) {
634                 FolderView *folderview = (FolderView *)list->data;
635
636                 folderview_set(folderview);
637         }
638
639         gtk_widget_destroy(window);
640 }
641
642 static gboolean folderview_search_new_recursive(GtkCTree *ctree,
643                                                 GtkCTreeNode *node)
644 {
645         FolderItem *item;
646
647         if (node) {
648                 item = gtk_ctree_node_get_row_data(ctree, node);
649                 if (item) {
650                         if (item->new > 0 ||
651                             (item->stype == F_QUEUE && item->total > 0))
652                                 return TRUE;
653                 }
654                 node = GTK_CTREE_ROW(node)->children;
655         } else
656                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
657
658         while (node) {
659                 if (folderview_search_new_recursive(ctree, node) == TRUE)
660                         return TRUE;
661                 node = GTK_CTREE_ROW(node)->sibling;
662         }
663
664         return FALSE;
665 }
666
667 static gboolean folderview_have_new_children(FolderView *folderview,
668                                              GtkCTreeNode *node)
669 {
670         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
671
672         if (!node)
673                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
674         if (!node)
675                 return FALSE;
676
677         node = GTK_CTREE_ROW(node)->children;
678
679         while (node) {
680                 if (folderview_search_new_recursive(ctree, node) == TRUE)
681                         return TRUE;
682                 node = GTK_CTREE_ROW(node)->sibling;
683         }
684
685         return FALSE;
686 }
687
688 static gboolean folderview_search_unread_recursive(GtkCTree *ctree,
689                                                    GtkCTreeNode *node)
690 {
691         FolderItem *item;
692
693         if (node) {
694                 item = gtk_ctree_node_get_row_data(ctree, node);
695                 if (item) {
696                         if (item->unread > 0 ||
697                             (item->stype == F_QUEUE && item->total > 0))
698                                 return TRUE;
699                 }
700                 node = GTK_CTREE_ROW(node)->children;
701         } else
702                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
703
704         while (node) {
705                 if (folderview_search_unread_recursive(ctree, node) == TRUE)
706                         return TRUE;
707                 node = GTK_CTREE_ROW(node)->sibling;
708         }
709
710         return FALSE;
711 }
712
713 static gboolean folderview_have_unread_children(FolderView *folderview,
714                                                 GtkCTreeNode *node)
715 {
716         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
717
718         if (!node)
719                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
720         if (!node)
721                 return FALSE;
722
723         node = GTK_CTREE_ROW(node)->children;
724
725         while (node) {
726                 if (folderview_search_unread_recursive(ctree, node) == TRUE)
727                         return TRUE;
728                 node = GTK_CTREE_ROW(node)->sibling;
729         }
730
731         return FALSE;
732 }
733
734 static void folderview_update_node(FolderView *folderview, GtkCTreeNode *node)
735 {
736         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
737         GtkStyle *style, *prev_style, *ctree_style;
738         GtkCTreeNode *parent;
739         FolderItem *item;
740         GdkPixmap *xpm, *openxpm;
741         GdkBitmap *mask, *openmask;
742         gchar *name;
743         gchar *str;
744         gboolean add_unread_mark;
745         gboolean use_bold, use_color;
746
747         item = gtk_ctree_node_get_row_data(ctree, node);
748         g_return_if_fail(item != NULL);
749
750         switch (item->stype) {
751         case F_INBOX:
752                 xpm = openxpm = inboxxpm;
753                 mask = openmask = inboxxpmmask;
754                 name = g_strdup(_("Inbox"));
755                 break;
756         case F_OUTBOX:
757                 xpm = openxpm = outboxxpm;
758                 mask = openmask = outboxxpmmask;
759                 name = g_strdup(_("Outbox"));
760                 break;
761         case F_QUEUE:
762                 xpm = openxpm = outboxxpm;
763                 mask = openmask = outboxxpmmask;
764                 name = g_strdup(_("Queue"));
765                 break;
766         case F_TRASH:
767                 xpm = openxpm = trashxpm;
768                 mask = openmask = trashxpmmask;
769                 name = g_strdup(_("Trash"));
770                 break;
771         case F_DRAFT:
772                 xpm = folderxpm;
773                 mask = folderxpmmask;
774                 openxpm = folderopenxpm;
775                 openmask = folderopenxpmmask;
776                 name = g_strdup(_("Draft"));
777                 break;
778         default:
779                 xpm = folderxpm;
780                 mask = folderxpmmask;
781                 openxpm = folderopenxpm;
782                 openmask = folderopenxpmmask;
783                 if (!item->parent) {
784                         switch (item->folder->type) {
785                         case F_MH:
786                                 name = " (MH)"; break;
787                         case F_MBOX:
788                                 name = " (mbox)"; break;
789                         case F_IMAP:
790                                 name = " (IMAP4)"; break;
791                         case F_NEWS:
792                                 name = " (News)"; break;
793                         default:
794                                 name = "";
795                         }
796                         name = g_strconcat(item->name, name, NULL);
797                 } else
798                         name = g_strdup(item->name);
799         }
800
801         if (!GTK_CTREE_ROW(node)->expanded &&
802             folderview_have_unread_children(folderview, node))
803                 add_unread_mark = TRUE;
804         else
805                 add_unread_mark = FALSE;
806
807         if (item->stype == F_QUEUE && item->total > 0 &&
808             prefs_common.display_folder_unread) {
809                 str = g_strdup_printf("%s (%d%s)", name, item->total,
810                                       add_unread_mark ? "+" : "");
811                 gtk_ctree_set_node_info(ctree, node, str, FOLDER_SPACING,
812                                         xpm, mask, openxpm, openmask,
813                                         FALSE, GTK_CTREE_ROW(node)->expanded);
814                 g_free(str);
815         } else if ((item->unread > 0 || add_unread_mark) &&
816                  prefs_common.display_folder_unread) {
817
818                 if (item->unread > 0)
819                         str = g_strdup_printf("%s (%d%s)", name, item->unread,
820                                               add_unread_mark ? "+" : "");
821                 else
822                         str = g_strdup_printf("%s (+)", name);
823                 gtk_ctree_set_node_info(ctree, node, str, FOLDER_SPACING,
824                                         xpm, mask, openxpm, openmask,
825                                         FALSE, GTK_CTREE_ROW(node)->expanded);
826                 g_free(str);
827         } else
828                 gtk_ctree_set_node_info(ctree, node, name, FOLDER_SPACING,
829                                         xpm, mask, openxpm, openmask,
830                                         FALSE, GTK_CTREE_ROW(node)->expanded);
831         g_free(name);
832
833         if (!item->parent) {
834                 gtk_ctree_node_set_text(ctree, node, COL_NEW,    "-");
835                 gtk_ctree_node_set_text(ctree, node, COL_UNREAD, "-");
836                 gtk_ctree_node_set_text(ctree, node, COL_TOTAL,  "-");
837         } else {
838                 gtk_ctree_node_set_text(ctree, node, COL_NEW,    itos(item->new));
839                 gtk_ctree_node_set_text(ctree, node, COL_UNREAD, itos(item->unread));
840                 gtk_ctree_node_set_text(ctree, node, COL_TOTAL,  itos(item->total));
841         }
842
843         if (item->stype == F_TRASH) return;
844
845         ctree_style = gtk_widget_get_style(GTK_WIDGET(ctree));
846         prev_style = gtk_ctree_node_get_row_style(ctree, node);
847         if (!prev_style)
848                 prev_style = ctree_style;
849         style = gtk_style_copy(prev_style);
850         if (!style) return;
851
852         if (item->stype == F_QUEUE) {
853                 /* highlight queue folder if there are any messages */
854                 use_bold = use_color = (item->total > 0);
855         } else {
856                 /* if unread messages exist, print with bold font */
857                 use_bold = (item->unread > 0) || add_unread_mark;
858                 /* if new messages exist, print with colored letter */
859                 use_color =
860                         (item->new > 0) ||
861                         (add_unread_mark &&
862                          folderview_have_new_children(folderview, node));
863         }
864
865         if (use_bold && boldfont)
866                 style->font = boldfont;
867         else
868                 style->font = ctree_style->font;
869
870         if (use_color) {
871                 style->fg[GTK_STATE_NORMAL]   = folderview->color_new;
872                 style->fg[GTK_STATE_SELECTED] = folderview->color_new;
873         } else {
874                 style->fg[GTK_STATE_NORMAL] =
875                         ctree_style->fg[GTK_STATE_NORMAL];
876                 style->fg[GTK_STATE_SELECTED] =
877                         ctree_style->fg[GTK_STATE_SELECTED];
878         }
879
880         gtk_ctree_node_set_row_style(ctree, node, style);
881
882         parent = node;
883         while ((parent = gtkut_ctree_find_collapsed_parent(ctree, parent))
884                != NULL)
885                 folderview_update_node(folderview, parent);
886 }
887
888 void folderview_update_item(FolderItem *item, gboolean update_summary)
889 {
890         GList *list;
891         FolderView *folderview;
892         GtkCTree *ctree;
893         GtkCTreeNode *node;
894
895         g_return_if_fail(item != NULL);
896
897         for (list = folderview_list; list != NULL; list = list->next) {
898                 folderview = (FolderView *)list->data;
899                 ctree = GTK_CTREE(folderview->ctree);
900
901                 node = gtk_ctree_find_by_row_data(ctree, NULL, item);
902                 if (node) {
903                         folderview_update_node(folderview, node);
904                         if (update_summary && folderview->opened == node)
905                                 summary_show(folderview->summaryview,
906                                              item, FALSE);
907                 }
908         }
909 }
910
911 static void folderview_update_item_foreach_func(gpointer key, gpointer val,
912                                                 gpointer data)
913 {
914         folderview_update_item((FolderItem *)key, FALSE);
915 }
916
917 void folderview_update_item_foreach(GHashTable *table)
918 {
919         g_hash_table_foreach(table, folderview_update_item_foreach_func, NULL);
920 }
921
922 static gboolean folderview_gnode_func(GtkCTree *ctree, guint depth,
923                                       GNode *gnode, GtkCTreeNode *cnode,
924                                       gpointer data)
925 {
926         FolderView *folderview = (FolderView *)data;
927         FolderItem *item = FOLDER_ITEM(gnode->data);
928
929         g_return_val_if_fail(item != NULL, FALSE);
930
931         gtk_ctree_node_set_row_data(ctree, cnode, item);
932         folderview_update_node(folderview, cnode);
933
934         return TRUE;
935 }
936
937 static void folderview_expand_func(GtkCTree *ctree, GtkCTreeNode *node,
938                                    gpointer data)
939 {
940         if (GTK_CTREE_ROW(node)->children)
941                 gtk_ctree_expand(ctree, node);
942 }
943
944 #define SET_SPECIAL_FOLDER(ctree, item) \
945 { \
946         if (item) { \
947                 GtkCTreeNode *node, *sibling; \
948  \
949                 node = gtk_ctree_find_by_row_data(ctree, root, item); \
950                 if (!node) \
951                         g_warning("%s not found.\n", item->path); \
952                 else { \
953                         if (!prev) \
954                                 sibling = GTK_CTREE_ROW(root)->children; \
955                         else \
956                                 sibling = GTK_CTREE_ROW(prev)->sibling; \
957                         if (node != sibling) \
958                                 gtk_ctree_move(ctree, node, root, sibling); \
959                 } \
960  \
961                 prev = node; \
962         } \
963 }
964
965 static void folderview_sort_folders(FolderView *folderview, GtkCTreeNode *root,
966                                     Folder *folder)
967 {
968         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
969         GtkCTreeNode *prev = NULL;
970
971         gtk_ctree_sort_recursive(ctree, root);
972
973         if (GTK_CTREE_ROW(root)->parent) return;
974
975         SET_SPECIAL_FOLDER(ctree, folder->inbox);
976         SET_SPECIAL_FOLDER(ctree, folder->outbox);
977         SET_SPECIAL_FOLDER(ctree, folder->draft);
978         SET_SPECIAL_FOLDER(ctree, folder->queue);
979         SET_SPECIAL_FOLDER(ctree, folder->trash);
980 }
981
982 static void folderview_append_folder(FolderView *folderview, Folder *folder)
983 {
984         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
985         GtkCTreeNode *root;
986
987         g_return_if_fail(folder != NULL);
988
989         root = gtk_ctree_insert_gnode(ctree, NULL, NULL, folder->node,
990                                       folderview_gnode_func, folderview);
991         gtk_ctree_pre_recursive(ctree, root, folderview_expand_func,
992                                 folderview);
993         folderview_sort_folders(folderview, root, folder);
994 }
995
996 void folderview_new_folder(FolderView *folderview)
997 {
998         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
999         FolderItem *item;
1000
1001         if (!folderview->selected) return;
1002
1003         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1004         g_return_if_fail(item != NULL);
1005         g_return_if_fail(item->folder != NULL);
1006
1007         switch (item->folder->type) {
1008         case F_MBOX:
1009                 folderview_new_mbox_folder_cb(folderview, 0, NULL);
1010                 break;
1011         case F_MH:
1012         case F_MAILDIR:
1013                 folderview_new_folder_cb(folderview, 0, NULL);
1014                 break;
1015         case F_IMAP:
1016                 folderview_new_imap_folder_cb(folderview, 0, NULL);
1017                 break;
1018         case F_NEWS:
1019         default:
1020                 break;
1021         }
1022 }
1023
1024 void folderview_rename_folder(FolderView *folderview)
1025 {
1026         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1027         FolderItem *item;
1028
1029         if (!folderview->selected) return;
1030
1031         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1032         g_return_if_fail(item != NULL);
1033         g_return_if_fail(item->folder != NULL);
1034         if (!item->path) return;
1035         if (item->stype != F_NORMAL) return;
1036
1037         switch (item->folder->type) {
1038         case F_MBOX:
1039                 folderview_rename_mbox_folder_cb(folderview, 0, NULL);
1040         case F_MH:
1041         case F_MAILDIR:
1042                 folderview_rename_folder_cb(folderview, 0, NULL);
1043                 break;
1044         case F_IMAP:
1045         case F_NEWS:
1046         default:
1047                 break;
1048         }
1049 }
1050
1051 void folderview_delete_folder(FolderView *folderview)
1052 {
1053         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1054         FolderItem *item;
1055
1056         if (!folderview->selected) return;
1057
1058         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1059         g_return_if_fail(item != NULL);
1060         g_return_if_fail(item->folder != NULL);
1061         if (!item->path) return;
1062         if (item->stype != F_NORMAL) return;
1063
1064         switch (item->folder->type) {
1065         case F_MH:
1066         case F_MBOX:
1067         case F_MAILDIR:
1068                 folderview_delete_folder_cb(folderview, 0, NULL);
1069                 break;
1070         case F_IMAP:
1071                 folderview_rm_imap_folder_cb(folderview, 0, NULL);
1072         case F_NEWS:
1073         default:
1074                 break;
1075         }
1076 }
1077
1078
1079 /* callback functions */
1080
1081 static void folderview_button_pressed(GtkWidget *ctree, GdkEventButton *event,
1082                                       FolderView *folderview)
1083 {
1084         GtkCList *clist = GTK_CLIST(ctree);
1085         gint prev_row = -1, row = -1, column = -1;
1086         FolderItem *item;
1087         Folder *folder;
1088
1089         if (!event) return;
1090
1091         if (event->button == 1) {
1092                 folderview->open_folder = TRUE;
1093                 return;
1094         }
1095
1096         if (event->button == 2 || event->button == 3) {
1097                 /* right clicked */
1098                 if (clist->selection) {
1099                         GtkCTreeNode *node;
1100
1101                         node = GTK_CTREE_NODE(clist->selection->data);
1102                         if (node)
1103                                 prev_row = gtkut_ctree_get_nth_from_node
1104                                         (GTK_CTREE(ctree), node);
1105                 }
1106
1107                 if (!gtk_clist_get_selection_info(clist, event->x, event->y,
1108                                                   &row, &column))
1109                         return;
1110                 if (prev_row != row) {
1111                         gtk_clist_unselect_all(clist);
1112                         if (event->button == 2)
1113                                 folderview_select_node
1114                                         (folderview,
1115                                          gtk_ctree_node_nth(GTK_CTREE(ctree),
1116                                                             row));
1117                         else
1118                                 gtk_clist_select_row(clist, row, column);
1119                 }
1120         }
1121
1122         if (event->button != 3) return;
1123
1124         item = gtk_clist_get_row_data(clist, row);
1125         g_return_if_fail(item != NULL);
1126         g_return_if_fail(item->folder != NULL);
1127         folder = item->folder;
1128
1129         menu_set_insensitive_all(GTK_MENU_SHELL(folderview->mail_popup));
1130         menu_set_insensitive_all(GTK_MENU_SHELL(folderview->imap_popup));
1131         menu_set_insensitive_all(GTK_MENU_SHELL(folderview->news_popup));
1132         menu_set_insensitive_all(GTK_MENU_SHELL(folderview->mbox_popup));
1133
1134         if (folder->type == F_MH && item->parent == NULL) {
1135                 menu_set_sensitive(folderview->mail_factory,
1136                                    "/Create new folder...", TRUE);
1137                 menu_set_sensitive(folderview->mail_factory,
1138                                    "/Remove mailbox", TRUE);
1139         } else if (folder->type == F_MH && item->stype != F_NORMAL) {
1140                 menu_set_sensitive(folderview->mail_factory,
1141                                    "/Create new folder...", TRUE);
1142                 menu_set_sensitive(folderview->mail_factory,
1143                                    "/Scoring...", TRUE);
1144         } else if (folder->type == F_MH) {
1145                 menu_set_sensitive(folderview->mail_factory,
1146                                    "/Create new folder...", TRUE);
1147                 menu_set_sensitive(folderview->mail_factory,
1148                                    "/Rename folder...", TRUE);
1149                 menu_set_sensitive(folderview->mail_factory,
1150                                    "/Delete folder", TRUE);
1151                 menu_set_sensitive(folderview->mail_factory,
1152                                    "/Scoring...", TRUE);
1153         } else if (folder->type == F_IMAP && item->parent == NULL) {
1154                 menu_set_sensitive(folderview->imap_factory,
1155                                    "/Create new folder...", TRUE);
1156                 menu_set_sensitive(folderview->imap_factory,
1157                                    "/Remove IMAP4 server", TRUE);
1158         } else if (folder->type == F_IMAP && item->stype != F_NORMAL) {
1159                 menu_set_sensitive(folderview->imap_factory,
1160                                    "/Create new folder...", TRUE);
1161         } else if (folder->type == F_IMAP) {
1162                 menu_set_sensitive(folderview->imap_factory,
1163                                    "/Create new folder...", TRUE);
1164                 menu_set_sensitive(folderview->imap_factory,
1165                                    "/Delete folder", TRUE);
1166                 menu_set_sensitive(folderview->imap_factory,
1167                                    "/Scoring...", TRUE);
1168         } else if (folder->type == F_NEWS && item->parent == NULL) {
1169                 menu_set_sensitive(folderview->news_factory,
1170                                    "/Subscribe to newsgroup...", TRUE);
1171                 menu_set_sensitive(folderview->news_factory,
1172                                    "/Remove news server", TRUE);
1173         } else if (folder->type == F_NEWS) {
1174                 menu_set_sensitive(folderview->news_factory,
1175                                    "/Subscribe to newsgroup...", TRUE);
1176                 menu_set_sensitive(folderview->news_factory,
1177                                    "/Remove newsgroup", TRUE);
1178                 menu_set_sensitive(folderview->news_factory,
1179                                    "/Scoring...", TRUE);
1180         }
1181         if (folder->type == F_MBOX && item->parent == NULL) {
1182                 menu_set_sensitive(folderview->mbox_factory,
1183                                    "/Create new folder...", TRUE);
1184                 menu_set_sensitive(folderview->mbox_factory,
1185                                    "/Remove mailbox", TRUE);
1186         } else if (folder->type == F_MBOX && item->stype != F_NORMAL) {
1187                 menu_set_sensitive(folderview->mbox_factory,
1188                                    "/Create new folder...", TRUE);
1189                 menu_set_sensitive(folderview->mbox_factory,
1190                                    "/Scoring...", TRUE);
1191         } else if (folder->type == F_MBOX) {
1192                 menu_set_sensitive(folderview->mbox_factory,
1193                                    "/Create new folder...", TRUE);
1194                 menu_set_sensitive(folderview->mbox_factory,
1195                                    "/Rename folder...", TRUE);
1196                 menu_set_sensitive(folderview->mbox_factory,
1197                                    "/Delete folder", TRUE);
1198                 menu_set_sensitive(folderview->mbox_factory,
1199                                    "/Scoring...", TRUE);
1200         }
1201
1202         if (folder->type == F_MH)
1203                 gtk_menu_popup(GTK_MENU(folderview->mail_popup), NULL, NULL,
1204                                NULL, NULL, event->button, event->time);
1205         else if (folder->type == F_IMAP)
1206                 gtk_menu_popup(GTK_MENU(folderview->imap_popup), NULL, NULL,
1207                                NULL, NULL, event->button, event->time);
1208         else if (folder->type == F_NEWS)
1209                 gtk_menu_popup(GTK_MENU(folderview->news_popup), NULL, NULL,
1210                                NULL, NULL, event->button, event->time);
1211         else if (folder->type == F_MBOX)
1212                 gtk_menu_popup(GTK_MENU(folderview->mbox_popup), NULL, NULL,
1213                                NULL, NULL, event->button, event->time);
1214 }
1215
1216 static void folderview_button_released(GtkWidget *ctree, GdkEventButton *event,
1217                                        FolderView *folderview)
1218 {
1219         if (!event) return;
1220
1221         if (event->button == 1 && folderview->open_folder == FALSE &&
1222             folderview->opened != NULL) {
1223                 gtk_ctree_select(GTK_CTREE(ctree), folderview->opened);
1224                 gtkut_ctree_set_focus_row(GTK_CTREE(ctree),
1225                                           folderview->opened);
1226         }
1227 }
1228
1229 #define BREAK_ON_MODIFIER_KEY() \
1230         if ((event->state & (GDK_MOD1_MASK|GDK_CONTROL_MASK)) != 0) break
1231
1232 static void folderview_key_pressed(GtkWidget *widget, GdkEventKey *event,
1233                                    FolderView *folderview)
1234 {
1235         if (!event) return;
1236
1237         switch (event->keyval) {
1238         case GDK_Return:
1239         case GDK_space:
1240                 if (folderview->selected) {
1241                         folderview_select_node(folderview,
1242                                                folderview->selected);
1243                 }
1244                 break;
1245         case GDK_v:
1246         case GDK_V:
1247         case GDK_g:
1248         case GDK_G:
1249         case GDK_x:
1250         case GDK_X:
1251         case GDK_w:
1252         case GDK_D:
1253         case GDK_Q:
1254                 BREAK_ON_MODIFIER_KEY();
1255                 summary_pass_key_press_event(folderview->summaryview, event);
1256         default:
1257         }
1258 }
1259
1260 static void folderview_selected(GtkCTree *ctree, GtkCTreeNode *row,
1261                                 gint column, FolderView *folderview)
1262 {
1263         static gboolean can_select = TRUE;      /* exclusive lock */
1264         gboolean opened;
1265         FolderItem *item;
1266
1267         folderview->selected = row;
1268
1269         if (folderview->opened == row) {
1270                 folderview->open_folder = FALSE;
1271                 return;
1272         }
1273
1274         if (!can_select) {
1275                 gtk_ctree_select(ctree, folderview->opened);
1276                 gtkut_ctree_set_focus_row(ctree, folderview->opened);
1277                 return;
1278         }
1279
1280         if (!folderview->open_folder) return;
1281
1282         item = gtk_ctree_node_get_row_data(ctree, row);
1283         if (!item) return;
1284
1285         can_select = FALSE;
1286
1287         if (item->path)
1288                 debug_print(_("Folder %s is selected\n"), item->path);
1289
1290         if (!GTK_CTREE_ROW(row)->children)
1291                 gtk_ctree_expand(ctree, row);
1292         if (folderview->opened &&
1293             !GTK_CTREE_ROW(folderview->opened)->children)
1294                 gtk_ctree_collapse(ctree, folderview->opened);
1295
1296         /* ungrab the mouse event */
1297         if (GTK_WIDGET_HAS_GRAB(ctree)) {
1298                 gtk_grab_remove(GTK_WIDGET(ctree));
1299                 if (gdk_pointer_is_grabbed())
1300                         gdk_pointer_ungrab(GDK_CURRENT_TIME);
1301         }
1302
1303         opened = summary_show(folderview->summaryview, item, FALSE);
1304
1305         if (!opened) {
1306                 gtk_ctree_select(ctree, folderview->opened);
1307                 gtkut_ctree_set_focus_row(ctree, folderview->opened);
1308         } else
1309                 folderview->opened = row;
1310
1311         folderview->open_folder = FALSE;
1312         can_select = TRUE;
1313 }
1314
1315 static void folderview_tree_expanded(GtkCTree *ctree, GtkCTreeNode *node,
1316                                      FolderView *folderview)
1317 {
1318         folderview_update_node(folderview, node);
1319 }
1320
1321 static void folderview_tree_collapsed(GtkCTree *ctree, GtkCTreeNode *node,
1322                                       FolderView *folderview)
1323 {
1324         folderview_update_node(folderview, node);
1325 }
1326
1327 static void folderview_popup_close(GtkMenuShell *menu_shell,
1328                                    FolderView *folderview)
1329 {
1330         if (!folderview->opened) return;
1331
1332         gtk_ctree_select(GTK_CTREE(folderview->ctree), folderview->opened);
1333         gtkut_ctree_set_focus_row(GTK_CTREE(folderview->ctree),
1334                                   folderview->opened);
1335 }
1336
1337 static void folderview_col_resized(GtkCList *clist, gint column, gint width,
1338                                    FolderView *folderview)
1339 {
1340         switch (column) {
1341         case COL_FOLDER:
1342                 prefs_common.folder_col_folder = width;
1343                 break;
1344         case COL_NEW:
1345                 prefs_common.folder_col_new = width;
1346                 break;
1347         case COL_UNREAD:
1348                 prefs_common.folder_col_unread = width;
1349                 break;
1350         case COL_TOTAL:
1351                 prefs_common.folder_col_total = width;
1352                 break;
1353         default:
1354         }
1355 }
1356
1357 static GtkCTreeNode *folderview_find_by_name(GtkCTree *ctree,
1358                                              GtkCTreeNode *node,
1359                                              const gchar *name)
1360 {
1361         FolderItem *item;
1362
1363         if (!node)
1364                 node = GTK_CTREE_NODE(GTK_CLIST(ctree)->row_list);
1365         if (!node)
1366                 return NULL;
1367
1368         node = GTK_CTREE_ROW(node)->children;
1369
1370         while (node) {
1371                 item = gtk_ctree_node_get_row_data(ctree, node);
1372                 if (!folderview_compare_name(item, name))
1373                         return node;
1374                 node = GTK_CTREE_ROW(node)->sibling;
1375         }
1376
1377         return NULL;
1378 }
1379
1380 static void folderview_new_folder_cb(FolderView *folderview, guint action,
1381                                      GtkWidget *widget)
1382 {
1383         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1384         gchar *text[N_FOLDER_COLS] = {NULL, "0", "0", "0"};
1385         FolderItem *item;
1386         FolderItem *new_item;
1387         gchar *new_folder;
1388         GtkCTreeNode *node;
1389
1390         if (!folderview->selected) return;
1391
1392         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1393         g_return_if_fail(item != NULL);
1394         g_return_if_fail(item->folder != NULL);
1395
1396         new_folder = input_dialog(_("New folder"),
1397                                   _("Input the name of new folder:"),
1398                                   _("NewFolder"));
1399         if (!new_folder) return;
1400
1401         if (strchr(new_folder, G_DIR_SEPARATOR) != NULL) {
1402                 alertpanel_error(_("`%c' can't be included in folder name."),
1403                                  G_DIR_SEPARATOR);
1404                 g_free(new_folder);
1405                 return;
1406         }
1407
1408         /* find whether the directory already exists */
1409         if (folderview_find_by_name(ctree, folderview->selected, new_folder)) {
1410                 alertpanel_error(_("The folder `%s' already exists."),
1411                                  new_folder);
1412                 g_free(new_folder);
1413                 return;
1414         }
1415
1416         new_item = item->folder->create_folder(item->folder, item, new_folder);
1417         g_free(new_folder);
1418         if (!new_item) return;
1419
1420         gtk_clist_freeze(GTK_CLIST(ctree));
1421
1422         text[COL_FOLDER] = new_item->name;
1423         node = gtk_ctree_insert_node(ctree, folderview->selected, NULL, text,
1424                                      FOLDER_SPACING,
1425                                      folderxpm, folderxpmmask,
1426                                      folderopenxpm, folderopenxpmmask,
1427                                      FALSE, FALSE);
1428         gtk_ctree_expand(ctree, folderview->selected);
1429         gtk_ctree_node_set_row_data(ctree, node, new_item);
1430         folderview_sort_folders(folderview, folderview->selected, item->folder);
1431
1432         gtk_clist_thaw(GTK_CLIST(ctree));
1433
1434         folder_write_list();
1435 }
1436
1437 static void folderview_new_mbox_folder_cb(FolderView *folderview, guint action,
1438                                           GtkWidget *widget)
1439 {
1440         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1441         gchar *text[N_FOLDER_COLS] = {NULL, "0", "0", "0"};
1442         FolderItem *item;
1443         FolderItem *new_item;
1444         gchar *new_folder;
1445         GtkCTreeNode *node;
1446
1447         if (!folderview->selected) return;
1448
1449         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1450         g_return_if_fail(item != NULL);
1451         g_return_if_fail(item->folder != NULL);
1452
1453         new_folder = input_dialog(_("New folder"),
1454                                   _("Input the name of new folder:"),
1455                                   _("NewFolder"));
1456         if (!new_folder) return;
1457
1458         /* find whether the directory already exists */
1459         if (folderview_find_by_name(ctree, folderview->selected, new_folder)) {
1460                 alertpanel_error(_("The folder `%s' already exists."),
1461                                  new_folder);
1462                 g_free(new_folder);
1463                 return;
1464         }
1465
1466         new_item = item->folder->create_folder(item->folder, item, new_folder);
1467         g_free(new_folder);
1468         if (!new_item) return;
1469
1470         gtk_clist_freeze(GTK_CLIST(ctree));
1471
1472         text[COL_FOLDER] = new_item->name;
1473         node = gtk_ctree_insert_node(ctree, folderview->selected, NULL, text,
1474                                      FOLDER_SPACING,
1475                                      folderxpm, folderxpmmask,
1476                                      folderopenxpm, folderopenxpmmask,
1477                                      FALSE, FALSE);
1478         gtk_ctree_expand(ctree, folderview->selected);
1479         gtk_ctree_node_set_row_data(ctree, node, new_item);
1480         folderview_sort_folders(folderview, folderview->selected, item->folder);
1481
1482         gtk_clist_thaw(GTK_CLIST(ctree));
1483
1484         folder_write_list();
1485 }
1486
1487 static void folderview_rename_folder_cb(FolderView *folderview, guint action,
1488                                         GtkWidget *widget)
1489 {
1490         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1491         FolderItem *item;
1492         gchar *new_folder;
1493         gchar *message;
1494
1495         if (!folderview->selected) return;
1496
1497         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1498         g_return_if_fail(item != NULL);
1499         g_return_if_fail(item->path != NULL);
1500         g_return_if_fail(item->folder != NULL);
1501
1502         message = g_strdup_printf(_("Input new name for `%s':"),
1503                                   g_basename(item->path));
1504         new_folder = input_dialog(_("Rename folder"), message,
1505                                   g_basename(item->path));
1506         g_free(message);
1507         if (!new_folder) return;
1508
1509         if (strchr(new_folder, G_DIR_SEPARATOR) != NULL) {
1510                 alertpanel_error(_("`%c' can't be included in folder name."),
1511                                  G_DIR_SEPARATOR);
1512                 g_free(new_folder);
1513                 return;
1514         }
1515
1516         if (folderview_find_by_name
1517                 (ctree, GTK_CTREE_ROW(folderview->selected)->parent,
1518                  new_folder)) {
1519                 alertpanel_error(_("The folder `%s' already exists."),
1520                                  new_folder);
1521                 g_free(new_folder);
1522                 return;
1523         }
1524
1525         if (item->folder->rename_folder(item->folder, item, new_folder) < 0) {
1526                 g_free(new_folder);
1527                 return;
1528         }
1529         g_free(new_folder);
1530
1531         gtk_clist_freeze(GTK_CLIST(ctree));
1532
1533         folderview_update_node(folderview, folderview->selected);
1534         folderview_sort_folders(folderview,
1535                                 GTK_CTREE_ROW(folderview->selected)->parent,
1536                                 item->folder);
1537         if (folderview->opened == folderview->selected) {
1538                 if (!GTK_CTREE_ROW(folderview->opened)->children)
1539                         gtk_ctree_expand(ctree, folderview->opened);
1540                 summary_show(folderview->summaryview, item, FALSE);
1541         }
1542
1543         gtk_clist_thaw(GTK_CLIST(ctree));
1544
1545         folder_write_list();
1546 }
1547
1548 static void folderview_rename_mbox_folder_cb(FolderView *folderview,
1549                                              guint action,
1550                                              GtkWidget *widget)
1551 {
1552         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1553         FolderItem *item;
1554         gchar *new_folder;
1555         gchar *message;
1556
1557         if (!folderview->selected) return;
1558
1559         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1560         g_return_if_fail(item != NULL);
1561         g_return_if_fail(item->path != NULL);
1562         g_return_if_fail(item->folder != NULL);
1563
1564         message = g_strdup_printf(_("Input new name for `%s':"),
1565                                   g_basename(item->path));
1566         new_folder = input_dialog(_("Rename folder"), message,
1567                                   g_basename(item->path));
1568         g_free(message);
1569         if (!new_folder) return;
1570
1571         if (folderview_find_by_name
1572                 (ctree, GTK_CTREE_ROW(folderview->selected)->parent,
1573                  new_folder)) {
1574                 alertpanel_error(_("The folder `%s' already exists."),
1575                                  new_folder);
1576                 g_free(new_folder);
1577                 return;
1578         }
1579
1580         if (item->folder->rename_folder(item->folder, item, new_folder) < 0) {
1581                 g_free(new_folder);
1582                 return;
1583         }
1584         g_free(new_folder);
1585
1586         gtk_clist_freeze(GTK_CLIST(ctree));
1587
1588         folderview_update_node(folderview, folderview->selected);
1589         folderview_sort_folders(folderview,
1590                                 GTK_CTREE_ROW(folderview->selected)->parent,
1591                                 item->folder);
1592         if (folderview->opened == folderview->selected) {
1593                 if (!GTK_CTREE_ROW(folderview->opened)->children)
1594                         gtk_ctree_expand(ctree, folderview->opened);
1595                 summary_show(folderview->summaryview, item, FALSE);
1596         }
1597
1598         gtk_clist_thaw(GTK_CLIST(ctree));
1599
1600         folder_write_list();
1601 }
1602
1603 static void folderview_delete_folder_cb(FolderView *folderview, guint action,
1604                                         GtkWidget *widget)
1605 {
1606         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1607         FolderItem *item;
1608         gchar *message;
1609         AlertValue avalue;
1610
1611         if (!folderview->selected) return;
1612
1613         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1614         g_return_if_fail(item != NULL);
1615         g_return_if_fail(item->path != NULL);
1616         g_return_if_fail(item->folder != NULL);
1617
1618         message = g_strdup_printf
1619                 (_("All folder(s) and message(s) under `%s' will be deleted.\n"
1620                    "Do you really want to delete?"),
1621                  g_basename(item->path));
1622         avalue = alertpanel(_("Delete folder"), message,
1623                             _("Yes"), _("+No"), NULL);
1624         g_free(message);
1625         if (avalue != G_ALERTDEFAULT) return;
1626
1627         if (item->folder->remove_folder(item->folder, item) < 0) {
1628                 alertpanel_error(_("Can't remove the folder `%s'."),
1629                                  item->path);
1630                 return;
1631         }
1632
1633         if (folderview->opened == folderview->selected ||
1634             gtk_ctree_is_ancestor(ctree,
1635                                   folderview->selected,
1636                                   folderview->opened)) {
1637                 summary_clear_all(folderview->summaryview);
1638                 folderview->opened = NULL;
1639         }
1640
1641         gtk_ctree_remove_node(ctree, folderview->selected);
1642         folder_write_list();
1643 }
1644
1645 static void folderview_remove_mailbox_cb(FolderView *folderview, guint action,
1646                                          GtkWidget *widget)
1647 {
1648         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1649         GtkCTreeNode *node;
1650         FolderItem *item;
1651         gchar *message;
1652         AlertValue avalue;
1653
1654         if (!folderview->selected) return;
1655         node = folderview->selected;
1656         item = gtk_ctree_node_get_row_data(ctree, node);
1657         g_return_if_fail(item != NULL);
1658         g_return_if_fail(item->folder != NULL);
1659         if (item->parent) return;
1660
1661         message = g_strdup_printf
1662                 (_("Really remove the mailbox `%s' ?\n"
1663                    "(The messages are NOT deleted from disk)"),
1664                  item->folder->name);
1665         avalue = alertpanel(_("Remove folder"), message,
1666                             _("Yes"), _("+No"), NULL);
1667         g_free(message);
1668         if (avalue != G_ALERTDEFAULT) return;
1669
1670         folder_destroy(item->folder);
1671         summary_clear_all(folderview->summaryview);
1672         folderview_unselect(folderview);
1673         gtk_ctree_remove_node(ctree, node);
1674         folder_write_list();
1675 }
1676
1677 static void folderview_new_imap_folder_cb(FolderView *folderview, guint action,
1678                                           GtkWidget *widget)
1679 {
1680         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1681         gchar *text[N_FOLDER_COLS] = {NULL, "0", "0", "0"};
1682         GtkCTreeNode *node;
1683         FolderItem *item;
1684         FolderItem *new_item;
1685         gchar *new_folder;
1686         gchar *p;
1687
1688         if (!folderview->selected) return;
1689
1690         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1691         g_return_if_fail(item != NULL);
1692         g_return_if_fail(item->folder != NULL);
1693         g_return_if_fail(item->folder->type == F_IMAP);
1694         g_return_if_fail(item->folder->account != NULL);
1695
1696         new_folder = input_dialog
1697                 (_("New folder"),
1698                  _("Input the name of new folder:\n"
1699                    "(if you want to create a folder to store subfolders,\n"
1700                    " append `/' at the end of the name)"),
1701                  _("NewFolder"));
1702         if (!new_folder) return;
1703
1704         if ((p = strchr(new_folder, G_DIR_SEPARATOR)) != NULL &&
1705             *(p + 1) != '\0') {
1706                 alertpanel_error(_("`%c' can't be included in folder name."),
1707                                  G_DIR_SEPARATOR);
1708                 g_free(new_folder);
1709                 return;
1710         }
1711
1712         /* find whether the directory already exists */
1713         if (folderview_find_by_name(ctree, folderview->selected, new_folder)) {
1714                 alertpanel_error(_("The folder `%s' already exists."),
1715                                  new_folder);
1716                 g_free(new_folder);
1717                 return;
1718         }
1719
1720         new_item = item->folder->create_folder(item->folder, item, new_folder);
1721         g_free(new_folder);
1722         if (!new_item) return;
1723
1724         gtk_clist_freeze(GTK_CLIST(ctree));
1725
1726         text[COL_FOLDER] = new_item->name;
1727         node = gtk_ctree_insert_node(ctree, folderview->selected, NULL, text,
1728                                      FOLDER_SPACING,
1729                                      folderxpm, folderxpmmask,
1730                                      folderopenxpm, folderopenxpmmask,
1731                                      FALSE, FALSE);
1732         gtk_ctree_expand(ctree, folderview->selected);
1733         gtk_ctree_node_set_row_data(ctree, node, new_item);
1734         folderview_sort_folders(folderview, folderview->selected, item->folder);
1735
1736         gtk_clist_thaw(GTK_CLIST(ctree));
1737
1738         folder_write_list();
1739 }
1740
1741 static void folderview_rm_imap_folder_cb(FolderView *folderview, guint action,
1742                                          GtkWidget *widget)
1743 {
1744         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1745         FolderItem *item;
1746         gchar *message;
1747         AlertValue avalue;
1748
1749         if (!folderview->selected) return;
1750
1751         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1752         g_return_if_fail(item != NULL);
1753         g_return_if_fail(item->folder != NULL);
1754         g_return_if_fail(item->folder->type == F_IMAP);
1755         g_return_if_fail(item->folder->account != NULL);
1756
1757         message = g_strdup_printf(_("Really delete folder `%s'?"),
1758                                   g_basename(item->path));
1759         avalue = alertpanel(_("Delete folder"), message,
1760                             _("Yes"), _("+No"), NULL);
1761         g_free(message);
1762         if (avalue != G_ALERTDEFAULT) return;
1763
1764         if (item->folder->remove_folder(item->folder, item) < 0) {
1765                 alertpanel_error(_("Can't remove the folder `%s'."),
1766                                  item->path);
1767                 return;
1768         }
1769
1770         if (folderview->opened == folderview->selected ||
1771             gtk_ctree_is_ancestor(ctree,
1772                                   folderview->selected,
1773                                   folderview->opened)) {
1774                 summary_clear_all(folderview->summaryview);
1775                 folderview->opened = NULL;
1776         }
1777
1778         gtk_ctree_remove_node(ctree, folderview->selected);
1779         folder_write_list();
1780 }
1781
1782 static void folderview_rm_imap_server_cb(FolderView *folderview, guint action,
1783                                          GtkWidget *widget)
1784 {
1785         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1786         FolderItem *item;
1787         gchar *message;
1788         AlertValue avalue;
1789
1790         if (!folderview->selected) return;
1791
1792         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1793         g_return_if_fail(item != NULL);
1794         g_return_if_fail(item->folder != NULL);
1795         g_return_if_fail(item->folder->type == F_IMAP);
1796         g_return_if_fail(item->folder->account != NULL);
1797
1798         message = g_strdup_printf(_("Really delete IMAP4 server `%s'?"),
1799                                   item->folder->name);
1800         avalue = alertpanel(_("Delete IMAP4 server"), message,
1801                             _("Yes"), _("+No"), NULL);
1802         g_free(message);
1803
1804         if (avalue != G_ALERTDEFAULT) return;
1805
1806         if (folderview->opened == folderview->selected ||
1807             gtk_ctree_is_ancestor(ctree,
1808                                   folderview->selected,
1809                                   folderview->opened)) {
1810                 summary_clear_all(folderview->summaryview);
1811                 folderview->opened = NULL;
1812         }
1813
1814         account_destroy(item->folder->account);
1815         folder_destroy(item->folder);
1816         gtk_ctree_remove_node(ctree, folderview->selected);
1817         account_set_menu();
1818         main_window_reflect_prefs_all();
1819         folder_write_list();
1820 }
1821
1822 static void folderview_new_news_group_cb(FolderView *folderview, guint action,
1823                                          GtkWidget *widget)
1824 {
1825         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1826         gchar *text[N_FOLDER_COLS] = {NULL, "0", "0", "0"};
1827         GtkCTreeNode *servernode, *node;
1828         FolderItem *item;
1829         FolderItem *newitem;
1830         gchar *new_group;
1831         const gchar *server;
1832
1833         if (!folderview->selected) return;
1834
1835         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1836         g_return_if_fail(item != NULL);
1837         g_return_if_fail(item->folder != NULL);
1838         g_return_if_fail(item->folder->type == F_NEWS);
1839         g_return_if_fail(item->folder->account != NULL);
1840
1841         new_group = grouplist_dialog(item);
1842         if (!new_group) return;
1843
1844         if (GTK_CTREE_ROW(folderview->selected)->parent != NULL)
1845                 servernode = GTK_CTREE_ROW(folderview->selected)->parent;
1846         else
1847                 servernode = folderview->selected;
1848
1849         if (folderview_find_by_name(ctree, servernode, new_group)) {
1850                 alertpanel_error(_("The newsgroup `%s' already exists."),
1851                                  new_group);
1852                 g_free(new_group);
1853                 return;
1854         }
1855
1856         gtk_clist_freeze(GTK_CLIST(ctree));
1857
1858         text[COL_FOLDER] = new_group;
1859         node = gtk_ctree_insert_node(ctree, servernode, NULL, text,
1860                                      FOLDER_SPACING,
1861                                      folderxpm, folderxpmmask,
1862                                      folderopenxpm, folderopenxpmmask,
1863                                      FALSE, FALSE);
1864         gtk_ctree_expand(ctree, servernode);
1865
1866         item = gtk_ctree_node_get_row_data(ctree, servernode);
1867         server = item->folder->account->nntp_server;
1868
1869         newitem = folder_item_new(new_group, new_group);
1870         g_free(new_group);
1871         folder_item_append(item, newitem);
1872         gtk_ctree_node_set_row_data(ctree, node, newitem);
1873         gtk_ctree_sort_node(ctree, servernode);
1874
1875         gtk_clist_thaw(GTK_CLIST(ctree));
1876
1877         folder_write_list();
1878 }
1879
1880 static void folderview_rm_news_group_cb(FolderView *folderview, guint action,
1881                                         GtkWidget *widget)
1882 {
1883         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1884         FolderItem *item;
1885         gchar *message;
1886         AlertValue avalue;
1887
1888         if (!folderview->selected) return;
1889
1890         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1891         g_return_if_fail(item != NULL);
1892         g_return_if_fail(item->folder != NULL);
1893         g_return_if_fail(item->folder->type == F_NEWS);
1894         g_return_if_fail(item->folder->account != NULL);
1895
1896         message = g_strdup_printf(_("Really delete newsgroup `%s'?"),
1897                                   g_basename(item->path));
1898         avalue = alertpanel(_("Delete newsgroup"), message,
1899                             _("Yes"), _("+No"), NULL);
1900         g_free(message);
1901         if (avalue != G_ALERTDEFAULT) return;
1902
1903         if (folderview->opened == folderview->selected) {
1904                 summary_clear_all(folderview->summaryview);
1905                 folderview->opened = NULL;
1906         }
1907
1908         folder_item_remove(item);
1909         gtk_ctree_remove_node(ctree, folderview->selected);
1910         folder_write_list();
1911 }
1912
1913 static void folderview_rm_news_server_cb(FolderView *folderview, guint action,
1914                                          GtkWidget *widget)
1915 {
1916         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
1917         FolderItem *item;
1918         gchar *message;
1919         AlertValue avalue;
1920
1921         if (!folderview->selected) return;
1922
1923         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
1924         g_return_if_fail(item != NULL);
1925         g_return_if_fail(item->folder != NULL);
1926         g_return_if_fail(item->folder->type == F_NEWS);
1927         g_return_if_fail(item->folder->account != NULL);
1928
1929         message = g_strdup_printf(_("Really delete news server `%s'?"),
1930                                   item->folder->name);
1931         avalue = alertpanel(_("Delete news server"), message,
1932                             _("Yes"), _("+No"), NULL);
1933         g_free(message);
1934
1935         if (avalue != G_ALERTDEFAULT) return;
1936
1937         if (folderview->opened == folderview->selected ||
1938             gtk_ctree_is_ancestor(ctree,
1939                                   folderview->selected,
1940                                   folderview->opened)) {
1941                 summary_clear_all(folderview->summaryview);
1942                 folderview->opened = NULL;
1943         }
1944
1945         account_destroy(item->folder->account);
1946         folder_destroy(item->folder);
1947         gtk_ctree_remove_node(ctree, folderview->selected);
1948         account_set_menu();
1949         main_window_reflect_prefs_all();
1950         folder_write_list();
1951 }
1952
1953 static gboolean folderview_drag_motion_cb(GtkWidget      *widget,
1954                                           GdkDragContext *context,
1955                                           gint            x,
1956                                           gint            y,
1957                                           guint           time,
1958                                           FolderView     *folderview)
1959 {
1960         gint row, column;
1961         FolderItem *item;
1962         GtkCTreeNode *node;
1963
1964         if (gtk_clist_get_selection_info(GTK_CLIST(widget),
1965                                          x - 24, y - 24, &row, &column)) {
1966                 node = gtk_ctree_node_nth(GTK_CTREE(widget), row);
1967                 item = gtk_ctree_node_get_row_data(GTK_CTREE(widget), node);
1968                 if (item != NULL &&
1969                     item->path != NULL &&
1970                     folderview->summaryview->folder_item != item) {
1971                         if (item->folder->type != F_NEWS) {
1972                                 gtk_ctree_select(GTK_CTREE(widget), node);
1973                                 gdk_drag_status(context,
1974                                                 context->suggested_action,
1975                                                 time);
1976                                 return TRUE;
1977                         }
1978                 }
1979         }
1980
1981         gtk_ctree_select(GTK_CTREE(widget), folderview->opened);
1982         gdk_drag_status(context, 0, time);
1983
1984         return FALSE;
1985 }
1986
1987 static void folderview_drag_leave_cb(GtkWidget      *widget,
1988                                      GdkDragContext *context,
1989                                      guint           time,
1990                                      FolderView     *folderview)
1991 {
1992         gtk_ctree_select(GTK_CTREE(widget), folderview->opened);
1993 }
1994
1995 static void folderview_drag_received_cb(GtkWidget        *widget,
1996                                         GdkDragContext   *drag_context,
1997                                         gint              x,
1998                                         gint              y,
1999                                         GtkSelectionData *data,
2000                                         guint             info,
2001                                         guint             time,
2002                                         FolderView       *folderview)
2003 {
2004         gint row, column;
2005         FolderItem *item;
2006         GtkCTreeNode *node;
2007
2008         if (gtk_clist_get_selection_info(GTK_CLIST(widget),
2009                                          x - 24, y - 24, &row, &column) == 0)
2010                 return;
2011
2012         node = gtk_ctree_node_nth(GTK_CTREE(widget), row);
2013         item = gtk_ctree_node_get_row_data(GTK_CTREE(widget), node);
2014         if (item != NULL) {
2015                 summary_move_selected_to(folderview->summaryview, item);
2016                 gtk_drag_finish(drag_context, TRUE, TRUE, time);
2017         } else
2018                 gtk_drag_finish(drag_context, FALSE, FALSE, time);
2019 }
2020
2021 static gint folderview_compare_name(gconstpointer a, gconstpointer b)
2022 {
2023         const FolderItem *item = a;
2024         const gchar *name = b;
2025
2026         if (!item->path) return -1;
2027         return strcmp2(g_basename(item->path), name);
2028 }
2029
2030 static void folderview_scoring_cb(FolderView *folderview, guint action,
2031                                    GtkWidget *widget)
2032 {
2033         GtkCTree *ctree = GTK_CTREE(folderview->ctree);
2034         FolderItem *item;
2035
2036         if (!folderview->selected) return;
2037
2038         item = gtk_ctree_node_get_row_data(ctree, folderview->selected);
2039         g_return_if_fail(item != NULL);
2040         g_return_if_fail(item->folder != NULL);
2041
2042         prefs_scoring_open(item);
2043 }