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