2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2001 Hiroyuki Yamamoto
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
32 #include "folderview.h"
37 #include "mbox_folder.h"
43 #include "prefs_account.h"
44 #include "mbox_folder.h"
46 static GList *folder_list = NULL;
48 static void folder_init (Folder *folder,
52 static void local_folder_destroy (LocalFolder *lfolder);
53 static void remote_folder_destroy (RemoteFolder *rfolder);
54 static void mh_folder_destroy (MHFolder *folder);
55 static void mbox_folder_destroy (MboxFolder *folder);
56 static void imap_folder_destroy (IMAPFolder *folder);
57 static void news_folder_destroy (NewsFolder *folder);
59 static gboolean folder_read_folder_func (GNode *node,
61 static gchar *folder_get_list_path (void);
62 static void folder_write_list_recursive (GNode *node,
64 static void folder_update_op_count_rec (GNode *node);
67 Folder *folder_new(FolderType type, const gchar *name, const gchar *path)
69 Folder *folder = NULL;
71 name = name ? name : path;
74 folder = mbox_folder_new(name, path);
77 folder = mh_folder_new(name, path);
80 folder = imap_folder_new(name, path);
83 folder = news_folder_new(name, path);
92 Folder *mh_folder_new(const gchar *name, const gchar *path)
96 folder = (Folder *)g_new0(MHFolder, 1);
97 folder_init(folder, F_MH, name);
98 LOCAL_FOLDER(folder)->rootpath = g_strdup(path);
103 Folder *mbox_folder_new(const gchar *name, const gchar *path)
108 folder = (Folder *)g_new0(MboxFolder, 1);
109 folder_init(folder, F_MBOX, name);
110 LOCAL_FOLDER(folder)->rootpath = g_strdup(path);
115 Folder *maildir_folder_new(const gchar *name, const gchar *path)
117 /* not yet implemented */
121 Folder *imap_folder_new(const gchar *name, const gchar *path)
125 folder = (Folder *)g_new0(IMAPFolder, 1);
126 folder_init(folder, F_IMAP, name);
131 Folder *news_folder_new(const gchar *name, const gchar *path)
135 folder = (Folder *)g_new0(NewsFolder, 1);
136 folder_init(folder, F_NEWS, name);
141 FolderItem *folder_item_new(const gchar *name, const gchar *path)
145 item = g_new0(FolderItem, 1);
147 item->stype = F_NORMAL;
148 item->name = g_strdup(name);
149 item->path = g_strdup(path);
150 item->account = NULL;
156 item->no_sub = FALSE;
157 item->no_select = FALSE;
158 item->collapsed = FALSE;
163 item->prefs = prefs_folder_item_new();
168 void folder_item_append(FolderItem *parent, FolderItem *item)
172 g_return_if_fail(parent != NULL);
173 g_return_if_fail(parent->folder != NULL);
174 g_return_if_fail(item != NULL);
176 node = parent->folder->node;
177 node = g_node_find(node, G_PRE_ORDER, G_TRAVERSE_ALL, parent);
178 g_return_if_fail(node != NULL);
180 item->parent = parent;
181 item->folder = parent->folder;
182 g_node_append_data(node, item);
185 void folder_item_remove(FolderItem *item)
189 g_return_if_fail(item != NULL);
190 g_return_if_fail(item->folder != NULL);
192 node = item->folder->node;
193 node = g_node_find(node, G_PRE_ORDER, G_TRAVERSE_ALL, item);
194 g_return_if_fail(node != NULL);
196 /* TODO: free all FolderItem's first */
197 if (item->folder->node == node)
198 item->folder->node = NULL;
199 g_node_destroy(node);
202 void folder_item_destroy(FolderItem *item)
204 g_return_if_fail(item != NULL);
211 void folder_set_ui_func(Folder *folder, FolderUIFunc func, gpointer data)
213 g_return_if_fail(folder != NULL);
215 folder->ui_func = func;
216 folder->ui_func_data = data;
219 void folder_set_name(Folder *folder, const gchar *name)
221 g_return_if_fail(folder != NULL);
223 g_free(folder->name);
224 folder->name = name ? g_strdup(name) : NULL;
225 if (folder->node && folder->node->data) {
226 FolderItem *item = (FolderItem *)folder->node->data;
229 item->name = name ? g_strdup(name) : NULL;
233 void folder_destroy(Folder *folder)
235 g_return_if_fail(folder != NULL);
237 folder_list = g_list_remove(folder_list, folder);
239 switch (folder->type) {
241 mh_folder_destroy(MH_FOLDER(folder));
244 mbox_folder_destroy(MBOX_FOLDER(folder));
247 imap_folder_destroy(IMAP_FOLDER(folder));
250 news_folder_destroy(NEWS_FOLDER(folder));
256 folder_tree_destroy(folder);
257 g_free(folder->name);
261 void folder_tree_destroy(Folder *folder)
263 /* TODO: destroy all FolderItem before */
264 g_node_destroy(folder->node);
266 folder->inbox = NULL;
267 folder->outbox = NULL;
268 folder->draft = NULL;
269 folder->queue = NULL;
270 folder->trash = NULL;
274 void folder_add(Folder *folder)
280 g_return_if_fail(folder != NULL);
282 for (i = 0, cur = folder_list; cur != NULL; cur = cur->next, i++) {
283 cur_folder = FOLDER(cur->data);
284 if (folder->type == F_MH) {
285 if (cur_folder->type != F_MH) break;
286 } else if (folder->type == F_MBOX) {
287 if (cur_folder->type != F_MH &&
288 cur_folder->type != F_MBOX) break;
289 } else if (folder->type == F_IMAP) {
290 if (cur_folder->type != F_MH &&
291 cur_folder->type != F_MBOX &&
292 cur_folder->type != F_IMAP) break;
293 } else if (folder->type == F_NEWS) {
294 if (cur_folder->type != F_MH &&
295 cur_folder->type != F_MBOX &&
296 cur_folder->type != F_IMAP &&
297 cur_folder->type != F_NEWS) break;
301 folder_list = g_list_insert(folder_list, folder, i);
304 GList *folder_get_list(void)
309 gint folder_read_list(void)
314 node = xml_parse_file(folder_get_list_path());
315 if (!node) return -1;
317 xmlnode = node->data;
318 if (strcmp2(xmlnode->tag->tag, "folderlist") != 0) {
319 g_warning("wrong folder list\n");
324 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, 2,
325 folder_read_folder_func, NULL);
334 void folder_write_list(void)
341 path = folder_get_list_path();
342 if ((pfile = prefs_write_open(path)) == NULL) return;
344 fprintf(pfile->fp, "<?xml version=\"1.0\" encoding=\"%s\"?>\n",
345 conv_get_current_charset_str());
346 fputs("\n<folderlist>\n", pfile->fp);
348 for (list = folder_list; list != NULL; list = list->next) {
350 folder_write_list_recursive(folder->node, pfile->fp);
353 fputs("</folderlist>\n", pfile->fp);
355 if (prefs_write_close(pfile) < 0)
356 g_warning("failed to write folder list.\n");
359 Folder *folder_find_from_path(const gchar *path)
364 for (list = folder_list; list != NULL; list = list->next) {
366 if ((folder->type == F_MH || folder->type == F_MBOX) &&
367 !path_cmp(LOCAL_FOLDER(folder)->rootpath, path))
374 static gboolean folder_item_find_func(GNode *node, gpointer data)
376 FolderItem *item = node->data;
378 const gchar *path = d[0];
380 if (path_cmp(path, item->path) != 0)
388 FolderItem *folder_find_item_from_path(const gchar *path)
393 folder = folder_get_default_folder();
394 g_return_val_if_fail(folder != NULL, NULL);
396 d[0] = (gpointer)path;
398 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
399 folder_item_find_func, d);
403 Folder *folder_get_default_folder(void)
405 return folder_list ? FOLDER(folder_list->data) : NULL;
408 FolderItem *folder_get_default_inbox(void)
412 if (!folder_list) return NULL;
413 folder = FOLDER(folder_list->data);
414 g_return_val_if_fail(folder != NULL, NULL);
415 return folder->inbox;
418 FolderItem *folder_get_default_outbox(void)
422 if (!folder_list) return NULL;
423 folder = FOLDER(folder_list->data);
424 g_return_val_if_fail(folder != NULL, NULL);
425 return folder->outbox;
428 FolderItem *folder_get_default_draft(void)
432 if (!folder_list) return NULL;
433 folder = FOLDER(folder_list->data);
434 g_return_val_if_fail(folder != NULL, NULL);
435 return folder->draft;
438 FolderItem *folder_get_default_queue(void)
442 if (!folder_list) return NULL;
443 folder = FOLDER(folder_list->data);
444 g_return_val_if_fail(folder != NULL, NULL);
445 return folder->queue;
448 FolderItem *folder_get_default_trash(void)
452 if (!folder_list) return NULL;
453 folder = FOLDER(folder_list->data);
454 g_return_val_if_fail(folder != NULL, NULL);
455 return folder->trash;
458 gchar *folder_item_get_path(FolderItem *item)
463 g_return_val_if_fail(item != NULL, NULL);
465 if (FOLDER_TYPE(item->folder) == F_MH)
466 folder_path = g_strdup(LOCAL_FOLDER(item->folder)->rootpath);
467 else if (FOLDER_TYPE(item->folder) == F_MBOX) {
468 path = mbox_get_virtual_path(item);
471 folder_path = g_strconcat(get_mbox_cache_dir(),
472 G_DIR_SEPARATOR_S, path, NULL);
477 else if (FOLDER_TYPE(item->folder) == F_IMAP) {
478 g_return_val_if_fail(item->folder->account != NULL, NULL);
479 folder_path = g_strconcat(get_imap_cache_dir(),
481 item->folder->account->recv_server,
483 item->folder->account->userid,
485 } else if (FOLDER_TYPE(item->folder) == F_NEWS) {
486 g_return_val_if_fail(item->folder->account != NULL, NULL);
487 folder_path = g_strconcat(get_news_cache_dir(),
489 item->folder->account->nntp_server,
494 g_return_val_if_fail(folder_path != NULL, NULL);
496 if (folder_path[0] == G_DIR_SEPARATOR) {
498 path = g_strconcat(folder_path, G_DIR_SEPARATOR_S,
501 path = g_strdup(folder_path);
504 path = g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S,
505 folder_path, G_DIR_SEPARATOR_S,
508 path = g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S,
516 void folder_item_scan(FolderItem *item)
520 g_return_if_fail(item != NULL);
522 folder = item->folder;
524 g_return_if_fail(folder->scan != NULL);
526 folder->scan(folder, item);
529 static void folder_item_scan_foreach_func(gpointer key, gpointer val,
532 folder_item_scan(FOLDER_ITEM(key));
535 void folder_item_scan_foreach(GHashTable *table)
537 g_hash_table_foreach(table, folder_item_scan_foreach_func, NULL);
540 gchar *folder_item_fetch_msg(FolderItem *item, gint num)
544 g_return_val_if_fail(item != NULL, NULL);
546 folder = item->folder;
548 g_return_val_if_fail(folder->scan != NULL, NULL);
549 g_return_val_if_fail(folder->fetch_msg != NULL, NULL);
551 if (item->last_num < 0) folder->scan(folder, item);
553 return folder->fetch_msg(folder, item, num);
556 gint folder_item_add_msg(FolderItem *dest, const gchar *file,
557 gboolean remove_source)
562 g_return_val_if_fail(dest != NULL, -1);
563 g_return_val_if_fail(file != NULL, -1);
565 folder = dest->folder;
567 g_return_val_if_fail(folder->scan != NULL, -1);
568 g_return_val_if_fail(folder->add_msg != NULL, -1);
570 if (dest->last_num < 0) folder->scan(folder, dest);
572 num = folder->add_msg(folder, dest, file, remove_source);
573 if (num > 0) dest->last_num = num;
579 gint folder_item_move_msg(FolderItem *dest, MsgInfo *msginfo)
584 g_return_val_if_fail(dest != NULL, -1);
585 g_return_val_if_fail(msginfo != NULL, -1);
587 folder = dest->folder;
588 if (dest->last_num < 0) folder->scan(folder, dest);
590 num = folder->move_msg(folder, dest, msginfo);
591 if (num > 0) dest->last_num = num;
597 gint folder_item_move_msg(FolderItem *dest, MsgInfo *msginfo)
604 g_return_val_if_fail(dest != NULL, -1);
605 g_return_val_if_fail(msginfo != NULL, -1);
607 folder = dest->folder;
609 g_return_val_if_fail(folder->scan != NULL, -1);
610 g_return_val_if_fail(folder->remove_msg != NULL, -1);
611 g_return_val_if_fail(folder->copy_msg != NULL, -1);
613 if (dest->last_num < 0) folder->scan(folder, dest);
615 src_folder = msginfo->folder->folder;
617 num = folder->copy_msg(folder, dest, msginfo);
620 src_folder->remove_msg(src_folder,
625 if (folder->finished_copy)
626 folder->finished_copy(folder, dest);
628 src_folder = msginfo->folder->folder;
630 if (msginfo->folder && src_folder->scan)
631 src_folder->scan(src_folder, msginfo->folder);
632 folder->scan(folder, dest);
638 gint folder_item_move_msgs_with_dest(FolderItem *dest, GSList *msglist)
643 g_return_val_if_fail(dest != NULL, -1);
644 g_return_val_if_fail(msglist != NULL, -1);
646 folder = dest->folder;
647 if (dest->last_num < 0) folder->scan(folder, dest);
649 num = folder->move_msgs_with_dest(folder, dest, msglist);
650 if (num > 0) dest->last_num = num;
651 else dest->op_count = 0;
657 gint folder_item_move_msgs_with_dest(FolderItem *dest, GSList *msglist)
664 g_return_val_if_fail(dest != NULL, -1);
665 g_return_val_if_fail(msglist != NULL, -1);
667 folder = dest->folder;
669 g_return_val_if_fail(folder->scan != NULL, -1);
670 g_return_val_if_fail(folder->copy_msg != NULL, -1);
671 g_return_val_if_fail(folder->remove_msg != NULL, -1);
673 if (dest->last_num < 0) folder->scan(folder, dest);
676 for(l = msglist ; l != NULL ; l = g_slist_next(l)) {
677 MsgInfo * msginfo = (MsgInfo *) l->data;
679 if (!item && msginfo->folder != NULL)
680 item = msginfo->folder;
682 if (folder->copy_msg(folder, dest, msginfo) != -1)
683 item->folder->remove_msg(item->folder,
688 if (folder->finished_copy)
689 folder->finished_copy(folder, dest);
691 if (item && item->folder->scan)
692 item->folder->scan(item->folder, item);
693 folder->scan(folder, dest);
695 return dest->last_num;
699 gint folder_item_copy_msg(FolderItem *dest, MsgInfo *msginfo)
704 g_return_val_if_fail(dest != NULL, -1);
705 g_return_val_if_fail(msginfo != NULL, -1);
707 folder = dest->folder;
708 if (dest->last_num < 0) folder->scan(folder, dest);
710 num = folder->copy_msg(folder, dest, msginfo);
711 if (num > 0) dest->last_num = num;
717 gint folder_item_copy_msg(FolderItem *dest, MsgInfo *msginfo)
724 g_return_val_if_fail(dest != NULL, -1);
725 g_return_val_if_fail(msginfo != NULL, -1);
727 folder = dest->folder;
729 g_return_val_if_fail(folder->scan != NULL, -1);
730 g_return_val_if_fail(folder->copy_msg != NULL, -1);
732 if (dest->last_num < 0) folder->scan(folder, dest);
734 num = folder->copy_msg(folder, dest, msginfo);
736 if (folder->finished_copy)
737 folder->finished_copy(folder, dest);
739 folder->scan(folder, dest);
745 gint folder_item_copy_msgs_with_dest(FolderItem *dest, GSList *msglist)
750 g_return_val_if_fail(dest != NULL, -1);
751 g_return_val_if_fail(msglist != NULL, -1);
753 folder = dest->folder;
754 if (dest->last_num < 0) folder->scan(folder, dest);
756 num = folder->copy_msgs_with_dest(folder, dest, msglist);
757 if (num > 0) dest->last_num = num;
758 else dest->op_count = 0;
764 gint folder_item_copy_msgs_with_dest(FolderItem *dest, GSList *msglist)
771 g_return_val_if_fail(dest != NULL, -1);
772 g_return_val_if_fail(msglist != NULL, -1);
774 folder = dest->folder;
776 g_return_val_if_fail(folder->scan != NULL, -1);
777 g_return_val_if_fail(folder->copy_msg != NULL, -1);
779 if (dest->last_num < 0) folder->scan(folder, dest);
781 for(l = msglist ; l != NULL ; l = g_slist_next(l)) {
782 MsgInfo * msginfo = (MsgInfo *) l->data;
784 folder->copy_msg(folder, dest, msginfo);
787 if (folder->finished_copy)
788 folder->finished_copy(folder, dest);
790 folder->scan(folder, dest);
792 return dest->last_num;
795 gint folder_item_remove_msg(FolderItem *item, gint num)
800 g_return_val_if_fail(item != NULL, -1);
802 folder = item->folder;
804 g_return_val_if_fail(folder->scan != NULL, -1);
805 g_return_val_if_fail(folder->remove_msg != NULL, -1);
807 if (folder->finished_remove)
808 folder->finished_remove(folder, item);
810 result = folder->remove_msg(folder, item, num);
812 if (item->last_num < 0) folder->scan(folder, item);
815 if (folder->finished_remove)
816 folder->finished_remove(folder, item);
822 gint folder_item_remove_all_msg(FolderItem *item)
827 g_return_val_if_fail(item != NULL, -1);
829 folder = item->folder;
831 g_return_val_if_fail(folder->scan != NULL, -1);
832 g_return_val_if_fail(folder->remove_all_msg != NULL, -1);
834 if (item->last_num < 0) folder->scan(folder, item);
836 result = folder->remove_all_msg(folder, item);
839 if (folder->finished_remove)
840 folder->finished_remove(folder, item);
846 gboolean folder_item_is_msg_changed(FolderItem *item, MsgInfo *msginfo)
850 g_return_val_if_fail(item != NULL, FALSE);
852 folder = item->folder;
854 g_return_val_if_fail(folder->is_msg_changed != NULL, -1);
856 return folder->is_msg_changed(folder, item, msginfo);
859 gchar *folder_item_get_cache_file(FolderItem *item)
864 g_return_val_if_fail(item != NULL, NULL);
865 g_return_val_if_fail(item->path != NULL, NULL);
867 path = folder_item_get_path(item);
868 g_return_val_if_fail(path != NULL, NULL);
869 if (!is_dir_exist(path))
871 file = g_strconcat(path, G_DIR_SEPARATOR_S, CACHE_FILE, NULL);
877 gchar *folder_item_get_mark_file(FolderItem *item)
882 g_return_val_if_fail(item != NULL, NULL);
883 g_return_val_if_fail(item->path != NULL, NULL);
885 path = folder_item_get_path(item);
886 g_return_val_if_fail(path != NULL, NULL);
887 if (!is_dir_exist(path))
889 file = g_strconcat(path, G_DIR_SEPARATOR_S, MARK_FILE, NULL);
896 static void folder_init(Folder *folder, FolderType type, const gchar *name)
900 g_return_if_fail(folder != NULL);
903 folder_set_name(folder, name);
904 folder->account = NULL;
905 folder->inbox = NULL;
906 folder->outbox = NULL;
907 folder->draft = NULL;
908 folder->queue = NULL;
909 folder->trash = NULL;
910 folder->ui_func = NULL;
911 folder->ui_func_data = NULL;
912 item = folder_item_new(name, NULL);
913 item->folder = folder;
914 folder->node = g_node_new(item);
919 folder->get_msg_list = mh_get_msg_list;
920 folder->fetch_msg = mh_fetch_msg;
921 folder->add_msg = mh_add_msg;
923 folder->move_msg = mh_move_msg;
924 folder->move_msgs_with_dest = mh_move_msgs_with_dest;
925 folder->copy_msg = mh_copy_msg;
926 folder->copy_msgs_with_dest = mh_copy_msgs_with_dest;
928 folder->copy_msg = mh_copy_msg;
929 folder->remove_msg = mh_remove_msg;
930 folder->remove_all_msg = mh_remove_all_msg;
931 folder->is_msg_changed = mh_is_msg_changed;
932 folder->scan = mh_scan_folder;
933 folder->scan_tree = mh_scan_tree;
934 folder->create_tree = mh_create_tree;
935 folder->create_folder = mh_create_folder;
936 folder->rename_folder = mh_rename_folder;
937 folder->remove_folder = mh_remove_folder;
940 folder->get_msg_list = imap_get_msg_list;
941 folder->fetch_msg = imap_fetch_msg;
942 folder->add_msg = imap_add_msg;
943 folder->move_msg = imap_move_msg;
944 folder->move_msgs_with_dest = imap_move_msgs_with_dest;
945 folder->copy_msg = imap_copy_msg;
946 folder->copy_msgs_with_dest = imap_copy_msgs_with_dest;
947 folder->remove_msg = imap_remove_msg;
948 folder->remove_all_msg = imap_remove_all_msg;
949 folder->scan = imap_scan_folder;
950 folder->scan_tree = imap_scan_tree;
951 folder->create_tree = imap_create_tree;
952 folder->create_folder = imap_create_folder;
953 folder->remove_folder = imap_remove_folder;
956 folder->get_msg_list = news_get_article_list;
957 folder->fetch_msg = news_fetch_msg;
958 folder->scan = news_scan_group;
961 folder->get_msg_list = mbox_get_msg_list;
962 folder->fetch_msg = mbox_fetch_msg;
963 folder->scan = mbox_scan_folder;
964 folder->add_msg = mbox_add_msg;
965 folder->remove_all_msg = mbox_remove_all_msg;
966 folder->remove_msg = mbox_remove_msg;
968 folder->move_msg = mbox_move_msg;
969 folder->move_msgs_with_dest = mbox_move_msgs_with_dest;
970 folder->copy_msg = mbox_copy_msg;
971 folder->copy_msgs_with_dest = mbox_copy_msgs_with_dest;
973 folder->copy_msg = mbox_copy_msg;
975 folder->create_tree = mbox_create_tree;
976 folder->create_folder = mbox_create_folder;
977 folder->rename_folder = mbox_rename_folder;
978 folder->remove_folder = mbox_remove_folder;
980 folder->update_mark = mbox_update_mark;
981 folder->change_flags = mbox_change_flags;
982 folder->finished_copy = mbox_finished_copy;
993 LOCAL_FOLDER(folder)->rootpath = NULL;
997 REMOTE_FOLDER(folder)->session = NULL;
1004 static void local_folder_destroy(LocalFolder *lfolder)
1006 g_return_if_fail(lfolder != NULL);
1008 g_free(lfolder->rootpath);
1011 static void remote_folder_destroy(RemoteFolder *rfolder)
1013 g_return_if_fail(rfolder != NULL);
1015 if (rfolder->session)
1016 session_destroy(rfolder->session);
1019 static void mh_folder_destroy(MHFolder *folder)
1021 local_folder_destroy(LOCAL_FOLDER(folder));
1024 static void mbox_folder_destroy(MboxFolder *folder)
1026 local_folder_destroy(LOCAL_FOLDER(folder));
1029 static void imap_folder_destroy(IMAPFolder *folder)
1031 remote_folder_destroy(REMOTE_FOLDER(folder));
1034 static void news_folder_destroy(NewsFolder *folder)
1036 remote_folder_destroy(REMOTE_FOLDER(folder));
1039 static gboolean folder_build_tree(GNode *node, gpointer data)
1041 Folder *folder = FOLDER(data);
1045 SpecialFolderItemType stype = F_NORMAL;
1046 const gchar *name = NULL;
1047 const gchar *path = NULL;
1048 PrefsAccount *account = NULL;
1049 gboolean no_sub = FALSE, no_select = FALSE, collapsed = FALSE;
1050 gint mtime = 0, new = 0, unread = 0, total = 0;
1052 g_return_val_if_fail(node->data != NULL, FALSE);
1053 if (!node->parent) return FALSE;
1055 xmlnode = node->data;
1056 if (strcmp2(xmlnode->tag->tag, "folderitem") != 0) {
1057 g_warning("tag name != \"folderitem\"\n");
1061 list = xmlnode->tag->attr;
1062 for (; list != NULL; list = list->next) {
1063 XMLAttr *attr = list->data;
1065 if (!attr || !attr->name || !attr->value) continue;
1066 if (!strcmp(attr->name, "type")) {
1067 if (!strcasecmp(attr->value, "normal"))
1069 else if (!strcasecmp(attr->value, "inbox"))
1071 else if (!strcasecmp(attr->value, "outbox"))
1073 else if (!strcasecmp(attr->value, "draft"))
1075 else if (!strcasecmp(attr->value, "queue"))
1077 else if (!strcasecmp(attr->value, "trash"))
1079 } else if (!strcmp(attr->name, "name"))
1081 else if (!strcmp(attr->name, "path"))
1083 else if (!strcmp(attr->name, "account_id")) {
1084 account = account_find_from_id(atoi(attr->value));
1085 if (!account) g_warning("account_id: %s not found\n",
1087 } else if (!strcmp(attr->name, "mtime"))
1088 mtime = atoi(attr->value);
1089 else if (!strcmp(attr->name, "new"))
1090 new = atoi(attr->value);
1091 else if (!strcmp(attr->name, "unread"))
1092 unread = atoi(attr->value);
1093 else if (!strcmp(attr->name, "total"))
1094 total = atoi(attr->value);
1095 else if (!strcmp(attr->name, "no_sub"))
1096 no_sub = *attr->value == '1' ? TRUE : FALSE;
1097 else if (!strcmp(attr->name, "no_select"))
1098 no_select = *attr->value == '1' ? TRUE : FALSE;
1099 else if (!strcmp(attr->name, "collapsed"))
1100 collapsed = *attr->value == '1' ? TRUE : FALSE;
1103 item = folder_item_new(name, path);
1104 item->stype = stype;
1105 item->account = account;
1106 item->mtime = mtime;
1108 item->unread = unread;
1109 item->total = total;
1110 item->no_sub = no_sub;
1111 item->no_select = no_select;
1112 item->collapsed = collapsed;
1113 item->parent = FOLDER_ITEM(node->parent->data);
1114 item->folder = folder;
1116 case F_INBOX: folder->inbox = item; break;
1117 case F_OUTBOX: folder->outbox = item; break;
1118 case F_DRAFT: folder->draft = item; break;
1119 case F_QUEUE: folder->queue = item; break;
1120 case F_TRASH: folder->trash = item; break;
1125 prefs_folder_item_read_config(item);
1128 xml_free_node(xmlnode);
1133 static gboolean folder_read_folder_func(GNode *node, gpointer data)
1138 FolderType type = F_UNKNOWN;
1139 const gchar *name = NULL;
1140 const gchar *path = NULL;
1141 PrefsAccount *account = NULL;
1142 gboolean collapsed = FALSE;
1144 if (g_node_depth(node) != 2) return FALSE;
1145 g_return_val_if_fail(node->data != NULL, FALSE);
1147 xmlnode = node->data;
1148 if (strcmp2(xmlnode->tag->tag, "folder") != 0) {
1149 g_warning("tag name != \"folder\"\n");
1152 g_node_unlink(node);
1153 list = xmlnode->tag->attr;
1154 for (; list != NULL; list = list->next) {
1155 XMLAttr *attr = list->data;
1157 if (!attr || !attr->name || !attr->value) continue;
1158 if (!strcmp(attr->name, "type")) {
1159 if (!strcasecmp(attr->value, "mh"))
1161 else if (!strcasecmp(attr->value, "mbox"))
1163 else if (!strcasecmp(attr->value, "maildir"))
1165 else if (!strcasecmp(attr->value, "imap"))
1167 else if (!strcasecmp(attr->value, "news"))
1169 } else if (!strcmp(attr->name, "name"))
1171 else if (!strcmp(attr->name, "path"))
1173 else if (!strcmp(attr->name, "account_id")) {
1174 account = account_find_from_id(atoi(attr->value));
1175 if (!account) g_warning("account_id: %s not found\n",
1177 } else if (!strcmp(attr->name, "collapsed"))
1178 collapsed = *attr->value == '1' ? TRUE : FALSE;
1181 folder = folder_new(type, name, path);
1182 g_return_val_if_fail(folder != NULL, FALSE);
1183 folder->account = account;
1184 if (account && (type == F_IMAP || type == F_NEWS))
1185 account->folder = REMOTE_FOLDER(folder);
1186 node->data = folder->node->data;
1187 g_node_destroy(folder->node);
1188 folder->node = node;
1190 FOLDER_ITEM(node->data)->collapsed = collapsed;
1192 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
1193 folder_build_tree, folder);
1198 static gchar *folder_get_list_path(void)
1200 static gchar *filename = NULL;
1203 filename = g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S,
1209 static void folder_write_list_recursive(GNode *node, gpointer data)
1211 FILE *fp = (FILE *)data;
1212 FolderItem *item = FOLDER_ITEM(node->data);
1214 static gchar *folder_type_str[] = {"mh", "mbox", "maildir", "imap",
1216 static gchar *folder_item_stype_str[] = {"normal", "inbox", "outbox",
1217 "draft", "queue", "trash"};
1219 g_return_if_fail(item != NULL);
1221 depth = g_node_depth(node);
1222 for (i = 0; i < depth; i++)
1225 Folder *folder = item->folder;
1227 fprintf(fp, "<folder type=\"%s\"", folder_type_str[folder->type]);
1229 fputs(" name=\"", fp);
1230 xml_file_put_escape_str(fp, folder->name);
1233 if ((folder->type == F_MH) || (folder->type == F_MBOX)) {
1234 fputs(" path=\"", fp);
1235 xml_file_put_escape_str
1236 (fp, LOCAL_FOLDER(folder)->rootpath);
1239 if (folder->account)
1240 fprintf(fp, " account_id=\"%d\"",
1241 folder->account->account_id);
1242 if (item->collapsed && node->children)
1243 fputs(" collapsed=\"1\"", fp);
1245 fprintf(fp, "<folderitem type=\"%s\"",
1246 folder_item_stype_str[item->stype]);
1248 fputs(" name=\"", fp);
1249 xml_file_put_escape_str(fp, item->name);
1253 fputs(" path=\"", fp);
1254 xml_file_put_escape_str(fp, item->path);
1258 fprintf(fp, " account_id=\"%d\"",
1259 item->account->account_id);
1261 fputs(" no_sub=\"1\"", fp);
1262 if (item->no_select)
1263 fputs(" no_select=\"1\"", fp);
1264 if (item->collapsed && node->children)
1265 fputs(" collapsed=\"1\"", fp);
1267 " mtime=\"%ld\" new=\"%d\" unread=\"%d\" total=\"%d\"",
1268 item->mtime, item->new, item->unread, item->total);
1271 if (node->children) {
1275 child = node->children;
1281 folder_write_list_recursive(cur, data);
1284 for (i = 0; i < depth; i++)
1286 fprintf(fp, "</%s>\n", depth == 1 ? "folder" : "folderitem");
1291 static void folder_update_op_count_rec(GNode *node) {
1292 FolderItem *fitem = FOLDER_ITEM(node->data);
1294 if (g_node_depth(node) > 0) {
1295 if (fitem->op_count > 0) {
1296 fitem->op_count = 0;
1297 folderview_update_item(fitem, 0);
1299 if (node->children) {
1302 child = node->children;
1308 folder_update_op_count_rec(cur);
1314 void folder_update_op_count() {
1318 for (cur = folder_list; cur != NULL; cur = cur->next) {
1320 folder_update_op_count_rec(folder->node);
1324 typedef struct _type_str {
1330 static type_str type_str_table[] =
1334 {"#maildir", F_MAILDIR},
1340 static gchar * folder_get_type_string(gint type)
1344 for(i = 0 ; i < sizeof(type_str_table) / sizeof(type_str) ; i++) {
1345 if (type_str_table[i].type == type)
1346 return type_str_table[i].str;
1351 static gint folder_get_type_from_string(gchar * str)
1355 for(i = 0 ; i < sizeof(type_str_table) / sizeof(type_str) ; i++) {
1356 if (g_strcasecmp(type_str_table[i].str, str))
1357 return type_str_table[i].type;
1362 gchar * folder_get_identifier(Folder * folder)
1365 type_str = folder_get_type_string(folder->type);
1367 return g_strconcat(type_str, "/", folder->name, NULL);
1371 static gchar * folder_item_get_tree_identifier(FolderItem * item)
1373 if (item->parent != NULL) {
1377 path = folder_item_get_tree_identifier(item->parent);
1381 id = g_strconcat(path, "/", item->name, NULL);
1387 return g_strconcat("/", item->name, NULL);
1392 gchar * folder_item_get_identifier(FolderItem * item)
1398 folder_str = folder_get_identifier(item->folder);
1400 if (item->path == NULL) {
1405 id = g_strconcat(folder_str, "/", item->path, NULL);
1410 Folder * folder_find_from_name(const gchar * name)
1415 for (list = g_list_first(folder_list); list != NULL;
1416 list = list->next) {
1417 folder = list->data;
1418 if (strcmp(name, folder->name) == 0)
1424 FolderItem * folder_find_item_from_identifier(const gchar *identifier)
1434 Xalloca(str, strlen(identifier) + 1, return NULL);
1435 strcpy(str, identifier);
1437 /* extract box type */
1439 p = strchr(str, '/');
1446 type = folder_get_type_from_string(str);
1450 /* extract box name */
1452 p = strchr(str, '/');
1461 folder = folder_find_from_name(name);
1467 d[0] = (gpointer)path;
1469 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
1470 folder_item_find_func, d);