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.
36 #include "mbox_folder.h"
42 #include "prefs_account.h"
43 #include "mbox_folder.h"
45 static GList *folder_list = NULL;
47 static void folder_init (Folder *folder,
51 static void local_folder_destroy (LocalFolder *lfolder);
52 static void remote_folder_destroy (RemoteFolder *rfolder);
53 static void mh_folder_destroy (MHFolder *folder);
54 static void mbox_folder_destroy (MboxFolder *folder);
55 static void imap_folder_destroy (IMAPFolder *folder);
56 static void news_folder_destroy (NewsFolder *folder);
58 static gboolean folder_read_folder_func (GNode *node,
60 static gchar *folder_get_list_path (void);
61 static void folder_write_list_recursive (GNode *node,
65 Folder *folder_new(FolderType type, const gchar *name, const gchar *path)
67 Folder *folder = NULL;
69 name = name ? name : path;
72 folder = mbox_folder_new(name, path);
75 folder = mh_folder_new(name, path);
78 folder = imap_folder_new(name, path);
81 folder = news_folder_new(name, path);
90 Folder *mh_folder_new(const gchar *name, const gchar *path)
94 folder = (Folder *)g_new0(MHFolder, 1);
95 folder_init(folder, F_MH, name);
96 LOCAL_FOLDER(folder)->rootpath = g_strdup(path);
101 Folder *mbox_folder_new(const gchar *name, const gchar *path)
106 folder = (Folder *)g_new0(MboxFolder, 1);
107 folder_init(folder, F_MBOX, name);
108 LOCAL_FOLDER(folder)->rootpath = g_strdup(path);
113 Folder *maildir_folder_new(const gchar *name, const gchar *path)
115 /* not yet implemented */
119 Folder *imap_folder_new(const gchar *name, const gchar *path)
123 folder = (Folder *)g_new0(IMAPFolder, 1);
124 folder_init(folder, F_IMAP, name);
129 Folder *news_folder_new(const gchar *name, const gchar *path)
133 folder = (Folder *)g_new0(NewsFolder, 1);
134 folder_init(folder, F_NEWS, name);
139 FolderItem *folder_item_new(const gchar *name, const gchar *path)
143 item = g_new0(FolderItem, 1);
145 item->stype = F_NORMAL;
146 item->name = g_strdup(name);
147 item->path = g_strdup(path);
148 item->account = NULL;
158 item->prefs = prefs_folder_item_new();
163 void folder_item_append(FolderItem *parent, FolderItem *item)
167 g_return_if_fail(parent != NULL);
168 g_return_if_fail(parent->folder != NULL);
169 g_return_if_fail(item != NULL);
171 node = parent->folder->node;
172 node = g_node_find(node, G_PRE_ORDER, G_TRAVERSE_ALL, parent);
173 g_return_if_fail(node != NULL);
175 item->parent = parent;
176 item->folder = parent->folder;
177 g_node_append_data(node, item);
180 void folder_item_remove(FolderItem *item)
184 g_return_if_fail(item != NULL);
185 g_return_if_fail(item->folder != NULL);
187 node = item->folder->node;
188 node = g_node_find(node, G_PRE_ORDER, G_TRAVERSE_ALL, item);
189 g_return_if_fail(node != NULL);
191 /* TODO: free all FolderItem's first */
192 if (item->folder->node == node)
193 item->folder->node = NULL;
194 g_node_destroy(node);
197 void folder_set_ui_func(Folder *folder, FolderUIFunc func, gpointer data)
199 g_return_if_fail(folder != NULL);
201 folder->ui_func = func;
202 folder->ui_func_data = data;
205 void folder_set_name(Folder *folder, const gchar *name)
207 g_return_if_fail(folder != NULL);
209 g_free(folder->name);
210 folder->name = name ? g_strdup(name) : NULL;
211 if (folder->node && folder->node->data) {
212 FolderItem *item = (FolderItem *)folder->node->data;
215 item->name = name ? g_strdup(name) : NULL;
219 void folder_destroy(Folder *folder)
221 g_return_if_fail(folder != NULL);
223 folder_list = g_list_remove(folder_list, folder);
225 switch (folder->type) {
227 mh_folder_destroy(MH_FOLDER(folder));
230 mbox_folder_destroy(MBOX_FOLDER(folder));
233 imap_folder_destroy(IMAP_FOLDER(folder));
236 news_folder_destroy(NEWS_FOLDER(folder));
241 folder_tree_destroy(folder);
242 g_free(folder->name);
246 void folder_tree_destroy(Folder *folder)
248 /* TODO: destroy all FolderItem before */
249 g_node_destroy(folder->node);
252 void folder_add(Folder *folder)
258 g_return_if_fail(folder != NULL);
260 for (i = 0, cur = folder_list; cur != NULL; cur = cur->next, i++) {
261 cur_folder = FOLDER(cur->data);
262 if (folder->type == F_MH) {
263 if (cur_folder->type != F_MH) break;
264 } else if (folder->type == F_MBOX) {
265 if (cur_folder->type != F_MH &&
266 cur_folder->type != F_MBOX) break;
267 } else if (folder->type == F_IMAP) {
268 if (cur_folder->type != F_MH &&
269 cur_folder->type != F_MBOX &&
270 cur_folder->type != F_IMAP) break;
271 } else if (folder->type == F_NEWS) {
272 if (cur_folder->type != F_MH &&
273 cur_folder->type != F_MBOX &&
274 cur_folder->type != F_IMAP &&
275 cur_folder->type != F_NEWS) break;
279 folder_list = g_list_insert(folder_list, folder, i);
282 GList *folder_get_list(void)
287 gint folder_read_list(void)
292 node = xml_parse_file(folder_get_list_path());
293 if (!node) return -1;
295 xmlnode = node->data;
296 if (strcmp2(xmlnode->tag->tag, "folderlist") != 0) {
297 g_warning("wrong folder list\n");
302 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, 2,
303 folder_read_folder_func, NULL);
312 void folder_write_list(void)
319 path = folder_get_list_path();
320 if ((pfile = prefs_write_open(path)) == NULL) return;
322 fprintf(pfile->fp, "<?xml version=\"1.0\" encoding=\"%s\"?>\n",
323 conv_get_current_charset_str());
324 fputs("\n<folderlist>\n", pfile->fp);
326 for (list = folder_list; list != NULL; list = list->next) {
328 folder_write_list_recursive(folder->node, pfile->fp);
331 fputs("</folderlist>\n", pfile->fp);
333 if (prefs_write_close(pfile) < 0)
334 g_warning("failed to write folder list.\n");
337 Folder *folder_find_from_path(const gchar *path)
342 for (list = folder_list; list != NULL; list = list->next) {
344 if ((folder->type == F_MH || folder->type == F_MBOX) &&
345 !path_cmp(LOCAL_FOLDER(folder)->rootpath, path))
352 static gboolean folder_item_find_func(GNode *node, gpointer data)
354 FolderItem *item = node->data;
356 const gchar *path = d[0];
358 if (path_cmp(path, item->path) != 0)
366 FolderItem *folder_find_item_from_path(const gchar *path)
371 folder = folder_get_default_folder();
372 g_return_val_if_fail(folder != NULL, NULL);
374 d[0] = (gpointer)path;
376 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
377 folder_item_find_func, d);
381 Folder *folder_get_default_folder(void)
383 return FOLDER(folder_list->data);
386 FolderItem *folder_get_default_inbox(void)
390 folder = FOLDER(folder_list->data);
391 g_return_val_if_fail(folder != NULL, NULL);
392 return folder->inbox;
395 FolderItem *folder_get_default_outbox(void)
399 folder = FOLDER(folder_list->data);
400 g_return_val_if_fail(folder != NULL, NULL);
401 return folder->outbox;
404 FolderItem *folder_get_default_draft(void)
408 folder = FOLDER(folder_list->data);
409 g_return_val_if_fail(folder != NULL, NULL);
410 return folder->draft;
413 FolderItem *folder_get_default_queue(void)
417 folder = FOLDER(folder_list->data);
418 g_return_val_if_fail(folder != NULL, NULL);
419 return folder->queue;
422 FolderItem *folder_get_default_trash(void)
426 folder = FOLDER(folder_list->data);
427 g_return_val_if_fail(folder != NULL, NULL);
428 return folder->trash;
431 gchar *folder_item_get_path(FolderItem *item)
436 g_return_val_if_fail(item != NULL, NULL);
438 if (FOLDER_TYPE(item->folder) == F_MH)
439 folder_path = g_strdup(LOCAL_FOLDER(item->folder)->rootpath);
440 else if (FOLDER_TYPE(item->folder) == F_MBOX) {
441 path = mbox_get_virtual_path(item);
444 folder_path = g_strconcat(get_mbox_cache_dir(),
445 G_DIR_SEPARATOR_S, path, NULL);
450 else if (FOLDER_TYPE(item->folder) == F_IMAP) {
451 g_return_val_if_fail(item->folder->account != NULL, NULL);
452 folder_path = g_strconcat(get_imap_cache_dir(),
454 item->folder->account->recv_server,
456 item->folder->account->userid,
458 } else if (FOLDER_TYPE(item->folder) == F_NEWS) {
459 g_return_val_if_fail(item->folder->account != NULL, NULL);
460 folder_path = g_strconcat(get_news_cache_dir(),
462 item->folder->account->nntp_server,
467 g_return_val_if_fail(folder_path != NULL, NULL);
469 if (folder_path[0] == G_DIR_SEPARATOR) {
471 path = g_strconcat(folder_path, G_DIR_SEPARATOR_S,
474 path = g_strdup(folder_path);
477 path = g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S,
478 folder_path, G_DIR_SEPARATOR_S,
481 path = g_strconcat(get_home_dir(), G_DIR_SEPARATOR_S,
489 void folder_item_scan(FolderItem *item)
493 g_return_if_fail(item != NULL);
495 folder = item->folder;
497 g_return_if_fail(folder->scan != NULL);
499 folder->scan(folder, item);
502 static void folder_item_scan_foreach_func(gpointer key, gpointer val,
505 folder_item_scan(FOLDER_ITEM(key));
508 void folder_item_scan_foreach(GHashTable *table)
510 g_hash_table_foreach(table, folder_item_scan_foreach_func, NULL);
513 gchar *folder_item_fetch_msg(FolderItem *item, gint num)
517 g_return_val_if_fail(item != NULL, NULL);
519 folder = item->folder;
521 g_return_val_if_fail(folder->scan != NULL, NULL);
522 g_return_val_if_fail(folder->fetch_msg != NULL, NULL);
524 if (item->last_num < 0) folder->scan(folder, item);
526 return folder->fetch_msg(folder, item, num);
529 gint folder_item_add_msg(FolderItem *dest, const gchar *file,
530 gboolean remove_source)
535 g_return_val_if_fail(dest != NULL, -1);
536 g_return_val_if_fail(file != NULL, -1);
538 folder = dest->folder;
540 g_return_val_if_fail(folder->scan != NULL, -1);
541 g_return_val_if_fail(folder->add_msg != NULL, -1);
543 if (dest->last_num < 0) folder->scan(folder, dest);
545 num = folder->add_msg(folder, dest, file, remove_source);
546 if (num > 0) dest->last_num = num;
552 gint folder_item_move_msg(FolderItem *dest, MsgInfo *msginfo)
557 g_return_val_if_fail(dest != NULL, -1);
558 g_return_val_if_fail(msginfo != NULL, -1);
560 folder = dest->folder;
562 g_return_val_if_fail(folder->scan != NULL, -1);
563 g_return_val_if_fail(folder->move_msg != NULL, -1);
565 if (dest->last_num < 0) folder->scan(folder, dest);
567 num = folder->move_msg(folder, dest, msginfo);
568 if (num > 0) dest->last_num = num;
574 gint folder_item_move_msg(FolderItem *dest, MsgInfo *msginfo)
581 g_return_val_if_fail(dest != NULL, -1);
582 g_return_val_if_fail(msginfo != NULL, -1);
584 folder = dest->folder;
586 g_return_val_if_fail(folder->scan != NULL, -1);
587 g_return_val_if_fail(folder->remove_msg != NULL, -1);
588 g_return_val_if_fail(folder->copy_msg != NULL, -1);
590 if (dest->last_num < 0) folder->scan(folder, dest);
592 src_folder = msginfo->folder->folder;
594 num = folder->copy_msg(folder, dest, msginfo);
596 src_folder->remove_msg(src_folder,
600 if (folder->finished_copy)
601 folder->finished_copy(folder, dest);
603 src_folder = msginfo->folder->folder;
605 if (msginfo->folder && src_folder->scan)
606 src_folder->scan(src_folder, msginfo->folder);
607 folder->scan(folder, dest);
613 gint folder_item_move_msgs_with_dest(FolderItem *dest, GSList *msglist)
618 g_return_val_if_fail(dest != NULL, -1);
619 g_return_val_if_fail(msglist != NULL, -1);
621 folder = dest->folder;
623 g_return_val_if_fail(folder->scan != NULL, -1);
624 g_return_val_if_fail(folder->move_msgs_with_dest != NULL, -1);
626 if (dest->last_num < 0) folder->scan(folder, dest);
628 num = folder->move_msgs_with_dest(folder, dest, msglist);
629 if (num > 0) dest->last_num = num;
635 gint folder_item_move_msgs_with_dest(FolderItem *dest, GSList *msglist)
642 g_return_val_if_fail(dest != NULL, -1);
643 g_return_val_if_fail(msglist != NULL, -1);
645 folder = dest->folder;
647 g_return_val_if_fail(folder->scan != NULL, -1);
648 g_return_val_if_fail(folder->copy_msg != NULL, -1);
649 g_return_val_if_fail(folder->remove_msg != NULL, -1);
651 if (dest->last_num < 0) folder->scan(folder, dest);
654 for(l = msglist ; l != NULL ; l = g_slist_next(l)) {
655 MsgInfo * msginfo = (MsgInfo *) l->data;
657 if (!item && msginfo->folder != NULL)
658 item = msginfo->folder;
660 if (folder->copy_msg(folder, dest, msginfo) != -1)
661 item->folder->remove_msg(item->folder,
666 if (folder->finished_copy)
667 folder->finished_copy(folder, dest);
669 if (item && item->folder->scan)
670 item->folder->scan(item->folder, item);
671 folder->scan(folder, dest);
673 return dest->last_num;
677 gint folder_item_copy_msg(FolderItem *dest, MsgInfo *msginfo)
682 g_return_val_if_fail(dest != NULL, -1);
683 g_return_val_if_fail(msginfo != NULL, -1);
685 folder = dest->folder;
687 g_return_val_if_fail(folder->scan != NULL, -1);
688 g_return_val_if_fail(folder->copy_msg != NULL, -1);
690 if (dest->last_num < 0) folder->scan(folder, dest);
692 num = folder->copy_msg(folder, dest, msginfo);
693 if (num > 0) dest->last_num = num;
699 gint folder_item_copy_msg(FolderItem *dest, MsgInfo *msginfo)
706 g_return_val_if_fail(dest != NULL, -1);
707 g_return_val_if_fail(msginfo != NULL, -1);
709 folder = dest->folder;
711 g_return_val_if_fail(folder->scan != NULL, -1);
712 g_return_val_if_fail(folder->copy_msg != NULL, -1);
714 if (dest->last_num < 0) folder->scan(folder, dest);
716 num = folder->copy_msg(folder, dest, msginfo);
718 if (folder->finished_copy)
719 folder->finished_copy(folder, dest);
721 folder->scan(folder, dest);
727 gint folder_item_copy_msgs_with_dest(FolderItem *dest, GSList *msglist)
732 g_return_val_if_fail(dest != NULL, -1);
733 g_return_val_if_fail(msglist != NULL, -1);
735 folder = dest->folder;
737 g_return_val_if_fail(folder->scan != NULL, -1);
738 g_return_val_if_fail(folder->copy_msgs_with_dest != NULL, -1);
740 if (dest->last_num < 0) folder->scan(folder, dest);
742 num = folder->copy_msgs_with_dest(folder, dest, msglist);
743 if (num > 0) dest->last_num = num;
749 gint folder_item_copy_msgs_with_dest(FolderItem *dest, GSList *msglist)
756 g_return_val_if_fail(dest != NULL, -1);
757 g_return_val_if_fail(msglist != NULL, -1);
759 folder = dest->folder;
761 g_return_val_if_fail(folder->scan != NULL, -1);
762 g_return_val_if_fail(folder->copy_msg != NULL, -1);
764 if (dest->last_num < 0) folder->scan(folder, dest);
766 for(l = msglist ; l != NULL ; l = g_slist_next(l)) {
767 MsgInfo * msginfo = (MsgInfo *) l->data;
769 folder->copy_msg(folder, dest, msginfo);
772 if (folder->finished_copy)
773 folder->finished_copy(folder, dest);
775 folder->scan(folder, dest);
777 return dest->last_num;
780 gint folder_item_remove_msg(FolderItem *item, gint num)
785 g_return_val_if_fail(item != NULL, -1);
787 folder = item->folder;
789 g_return_val_if_fail(folder->scan != NULL, -1);
790 g_return_val_if_fail(folder->remove_msg != NULL, -1);
792 if (folder->finished_remove)
793 folder->finished_remove(folder, item);
795 result = folder->remove_msg(folder, item, num);
797 if (item->last_num < 0) folder->scan(folder, item);
800 if (folder->finished_remove)
801 folder->finished_remove(folder, item);
807 gint folder_item_remove_all_msg(FolderItem *item)
812 g_return_val_if_fail(item != NULL, -1);
814 folder = item->folder;
816 g_return_val_if_fail(folder->scan != NULL, -1);
817 g_return_val_if_fail(folder->remove_all_msg != NULL, -1);
819 if (item->last_num < 0) folder->scan(folder, item);
821 result = folder->remove_all_msg(folder, item);
824 if (folder->finished_remove)
825 folder->finished_remove(folder, item);
831 gboolean folder_item_is_msg_changed(FolderItem *item, MsgInfo *msginfo)
835 g_return_val_if_fail(item != NULL, FALSE);
837 folder = item->folder;
839 g_return_val_if_fail(folder->is_msg_changed != NULL, -1);
841 return folder->is_msg_changed(folder, item, msginfo);
844 gchar *folder_item_get_cache_file(FolderItem *item)
849 g_return_val_if_fail(item != NULL, NULL);
850 g_return_val_if_fail(item->path != NULL, NULL);
852 path = folder_item_get_path(item);
853 g_return_val_if_fail(path != NULL, NULL);
854 if (!is_dir_exist(path))
856 file = g_strconcat(path, G_DIR_SEPARATOR_S, CACHE_FILE, NULL);
862 gchar *folder_item_get_mark_file(FolderItem *item)
867 g_return_val_if_fail(item != NULL, NULL);
868 g_return_val_if_fail(item->path != NULL, NULL);
870 path = folder_item_get_path(item);
871 g_return_val_if_fail(path != NULL, NULL);
872 if (!is_dir_exist(path))
874 file = g_strconcat(path, G_DIR_SEPARATOR_S, MARK_FILE, NULL);
881 static void folder_init(Folder *folder, FolderType type, const gchar *name)
885 g_return_if_fail(folder != NULL);
888 folder_set_name(folder, name);
889 folder->account = NULL;
890 folder->inbox = NULL;
891 folder->outbox = NULL;
892 folder->draft = NULL;
893 folder->queue = NULL;
894 folder->trash = NULL;
895 folder->ui_func = NULL;
896 folder->ui_func_data = NULL;
897 item = folder_item_new(name, NULL);
898 item->folder = folder;
899 folder->node = g_node_new(item);
904 folder->get_msg_list = mh_get_msg_list;
905 folder->fetch_msg = mh_fetch_msg;
906 folder->add_msg = mh_add_msg;
908 folder->move_msg = mh_move_msg;
909 folder->move_msgs_with_dest = mh_move_msgs_with_dest;
910 folder->copy_msg = mh_copy_msg;
911 folder->copy_msgs_with_dest = mh_copy_msgs_with_dest;
913 folder->copy_msg = mh_copy_msg;
914 folder->remove_msg = mh_remove_msg;
915 folder->remove_all_msg = mh_remove_all_msg;
916 folder->is_msg_changed = mh_is_msg_changed;
917 folder->scan = mh_scan_folder;
918 folder->scan_tree = mh_scan_tree;
919 folder->create_tree = mh_create_tree;
920 folder->create_folder = mh_create_folder;
921 folder->rename_folder = mh_rename_folder;
922 folder->remove_folder = mh_remove_folder;
925 folder->get_msg_list = imap_get_msg_list;
926 folder->fetch_msg = imap_fetch_msg;
927 folder->add_msg = imap_add_msg;
928 folder->move_msg = imap_move_msg;
929 folder->move_msgs_with_dest = imap_move_msgs_with_dest;
930 folder->copy_msg = imap_copy_msg;
931 folder->copy_msgs_with_dest = imap_copy_msgs_with_dest;
932 folder->remove_msg = imap_remove_msg;
933 folder->remove_all_msg = imap_remove_all_msg;
934 folder->scan = imap_scan_folder;
935 folder->create_folder = imap_create_folder;
936 folder->remove_folder = imap_remove_folder;
939 folder->get_msg_list = news_get_article_list;
940 folder->fetch_msg = news_fetch_msg;
941 folder->scan = news_scan_group;
944 folder->get_msg_list = mbox_get_msg_list;
945 folder->fetch_msg = mbox_fetch_msg;
946 folder->scan = mbox_scan_folder;
947 folder->add_msg = mbox_add_msg;
948 folder->remove_all_msg = mbox_remove_all_msg;
949 folder->remove_msg = mbox_remove_msg;
951 folder->move_msg = mbox_move_msg;
952 folder->move_msgs_with_dest = mbox_move_msgs_with_dest;
953 folder->copy_msg = mbox_copy_msg;
954 folder->copy_msgs_with_dest = mbox_copy_msgs_with_dest;
956 folder->copy_msg = mbox_copy_msg;
958 folder->create_tree = mbox_create_tree;
959 folder->create_folder = mbox_create_folder;
960 folder->rename_folder = mbox_rename_folder;
961 folder->remove_folder = mbox_remove_folder;
963 folder->update_mark = mbox_update_mark;
964 folder->change_flags = mbox_change_flags;
965 folder->finished_copy = mbox_finished_copy;
975 LOCAL_FOLDER(folder)->rootpath = NULL;
979 REMOTE_FOLDER(folder)->session = NULL;
985 static void local_folder_destroy(LocalFolder *lfolder)
987 g_return_if_fail(lfolder != NULL);
989 g_free(lfolder->rootpath);
992 static void remote_folder_destroy(RemoteFolder *rfolder)
994 g_return_if_fail(rfolder != NULL);
996 if (rfolder->session)
997 session_destroy(rfolder->session);
1000 static void mh_folder_destroy(MHFolder *folder)
1002 local_folder_destroy(LOCAL_FOLDER(folder));
1005 static void mbox_folder_destroy(MboxFolder *folder)
1007 local_folder_destroy(LOCAL_FOLDER(folder));
1010 static void imap_folder_destroy(IMAPFolder *folder)
1012 remote_folder_destroy(REMOTE_FOLDER(folder));
1015 static void news_folder_destroy(NewsFolder *folder)
1017 remote_folder_destroy(REMOTE_FOLDER(folder));
1020 static gboolean folder_build_tree(GNode *node, gpointer data)
1022 Folder *folder = FOLDER(data);
1026 SpecialFolderItemType stype = F_NORMAL;
1027 const gchar *name = NULL;
1028 const gchar *path = NULL;
1029 PrefsAccount *account = NULL;
1030 gint mtime = 0, new = 0, unread = 0, total = 0;
1032 g_return_val_if_fail(node->data != NULL, FALSE);
1033 if (!node->parent) return FALSE;
1035 xmlnode = node->data;
1036 if (strcmp2(xmlnode->tag->tag, "folderitem") != 0) {
1037 g_warning("tag name != \"folderitem\"\n");
1041 list = xmlnode->tag->attr;
1042 for (; list != NULL; list = list->next) {
1043 XMLAttr *attr = list->data;
1045 if (!attr || !attr->name || !attr->value) continue;
1046 if (!strcmp(attr->name, "type")) {
1047 if (!strcasecmp(attr->value, "normal"))
1049 else if (!strcasecmp(attr->value, "inbox"))
1051 else if (!strcasecmp(attr->value, "outbox"))
1053 else if (!strcasecmp(attr->value, "draft"))
1055 else if (!strcasecmp(attr->value, "queue"))
1057 else if (!strcasecmp(attr->value, "trash"))
1059 } else if (!strcmp(attr->name, "name"))
1061 else if (!strcmp(attr->name, "path"))
1063 else if (!strcmp(attr->name, "account_id")) {
1064 account = account_find_from_id(atoi(attr->value));
1065 if (!account) g_warning("account_id: %s not found\n",
1068 else if (!strcmp(attr->name, "mtime"))
1069 mtime = atoi(attr->value);
1070 else if (!strcmp(attr->name, "new"))
1071 new = atoi(attr->value);
1072 else if (!strcmp(attr->name, "unread"))
1073 unread = atoi(attr->value);
1074 else if (!strcmp(attr->name, "total"))
1075 total = atoi(attr->value);
1078 item = folder_item_new(name, path);
1079 item->stype = stype;
1080 item->account = account;
1081 item->mtime = mtime;
1083 item->unread = unread;
1084 item->total = total;
1085 item->parent = FOLDER_ITEM(node->parent->data);
1086 item->folder = folder;
1088 case F_INBOX: folder->inbox = item; break;
1089 case F_OUTBOX: folder->outbox = item; break;
1090 case F_DRAFT: folder->draft = item; break;
1091 case F_QUEUE: folder->queue = item; break;
1092 case F_TRASH: folder->trash = item; break;
1096 prefs_folder_item_read_config(item);
1099 xml_free_node(xmlnode);
1104 static gboolean folder_read_folder_func(GNode *node, gpointer data)
1109 FolderType type = F_UNKNOWN;
1110 const gchar *name = NULL;
1111 const gchar *path = NULL;
1112 PrefsAccount *account = NULL;
1114 if (g_node_depth(node) != 2) return FALSE;
1115 g_return_val_if_fail(node->data != NULL, FALSE);
1117 xmlnode = node->data;
1118 if (strcmp2(xmlnode->tag->tag, "folder") != 0) {
1119 g_warning("tag name != \"folder\"\n");
1122 g_node_unlink(node);
1123 list = xmlnode->tag->attr;
1124 for (; list != NULL; list = list->next) {
1125 XMLAttr *attr = list->data;
1127 if (!attr || !attr->name || !attr->value) continue;
1128 if (!strcmp(attr->name, "type")) {
1129 if (!strcasecmp(attr->value, "mh"))
1131 else if (!strcasecmp(attr->value, "mbox"))
1133 else if (!strcasecmp(attr->value, "maildir"))
1135 else if (!strcasecmp(attr->value, "imap"))
1137 else if (!strcasecmp(attr->value, "news"))
1139 } else if (!strcmp(attr->name, "name"))
1141 else if (!strcmp(attr->name, "path"))
1143 else if (!strcmp(attr->name, "account_id")) {
1144 account = account_find_from_id(atoi(attr->value));
1145 if (!account) g_warning("account_id: %s not found\n",
1150 folder = folder_new(type, name, path);
1151 g_return_val_if_fail(folder != NULL, FALSE);
1152 folder->account = account;
1153 if (account && (type == F_IMAP || type == F_NEWS))
1154 account->folder = REMOTE_FOLDER(folder);
1155 node->data = folder->node->data;
1156 g_node_destroy(folder->node);
1157 folder->node = node;
1160 g_node_traverse(node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
1161 folder_build_tree, folder);
1166 static gchar *folder_get_list_path(void)
1168 static gchar *filename = NULL;
1171 filename = g_strconcat(get_rc_dir(), G_DIR_SEPARATOR_S,
1177 static void folder_write_list_recursive(GNode *node, gpointer data)
1179 FILE *fp = (FILE *)data;
1180 FolderItem *item = FOLDER_ITEM(node->data);
1182 static gchar *folder_type_str[] = {"mh", "mbox", "maildir", "imap",
1184 static gchar *folder_item_stype_str[] = {"normal", "inbox", "outbox",
1185 "draft", "queue", "trash"};
1187 g_return_if_fail(item != NULL);
1189 depth = g_node_depth(node);
1190 for (i = 0; i < depth; i++)
1193 Folder *folder = item->folder;
1195 fprintf(fp, "<folder type=\"%s\"", folder_type_str[folder->type]);
1197 fputs(" name=\"", fp);
1198 xml_file_put_escape_str(fp, folder->name);
1201 if ((folder->type == F_MH) || (folder->type == F_MBOX)) {
1202 fputs(" path=\"", fp);
1203 xml_file_put_escape_str
1204 (fp, LOCAL_FOLDER(folder)->rootpath);
1207 if (folder->account)
1208 fprintf(fp, " account_id=\"%d\"",
1209 folder->account->account_id);
1211 fprintf(fp, "<folderitem type=\"%s\"",
1212 folder_item_stype_str[item->stype]);
1214 fputs(" name=\"", fp);
1215 xml_file_put_escape_str(fp, item->name);
1219 fputs(" path=\"", fp);
1220 xml_file_put_escape_str(fp, item->path);
1224 fprintf(fp, " account_id = \"%d\"",
1225 item->account->account_id);
1227 " mtime=\"%ld\" new=\"%d\" unread=\"%d\" total=\"%d\"",
1228 item->mtime, item->new, item->unread, item->total);
1231 if (node->children) {
1235 child = node->children;
1241 folder_write_list_recursive(cur, data);
1244 for (i = 0; i < depth; i++)
1246 fprintf(fp, "</%s>\n", depth == 1 ? "folder" : "folderitem");
1251 typedef struct _type_str {
1257 static type_str type_str_table[] =
1261 {"#maildir", F_MAILDIR},
1267 static gchar * folder_get_type_string(gint type)
1271 for(i = 0 ; i < sizeof(type_str_table) / sizeof(type_str) ; i++) {
1272 if (type_str_table[i].type == type)
1273 return type_str_table[i].str;
1278 static gint folder_get_type_from_string(gchar * str)
1282 for(i = 0 ; i < sizeof(type_str_table) / sizeof(type_str) ; i++) {
1283 if (g_strcasecmp(type_str_table[i].str, str))
1284 return type_str_table[i].type;
1289 gchar * folder_get_identifier(Folder * folder)
1292 type_str = folder_get_type_string(folder->type);
1294 return g_strconcat(type_str, "/", folder->name, NULL);
1298 static gchar * folder_item_get_tree_identifier(FolderItem * item)
1300 if (item->parent != NULL) {
1304 path = folder_item_get_tree_identifier(item->parent);
1308 id = g_strconcat(path, "/", item->name, NULL);
1314 return g_strconcat("/", item->name, NULL);
1319 gchar * folder_item_get_identifier(FolderItem * item)
1325 folder_str = folder_get_identifier(item->folder);
1327 if (item->path == NULL) {
1332 id = g_strconcat(folder_str, "/", item->path, NULL);
1337 Folder * folder_find_from_name(const gchar * name)
1342 for (list = g_list_first(folder_list); list != NULL;
1343 list = list->next) {
1344 folder = list->data;
1345 if (strcmp(name, folder->name) == 0)
1351 FolderItem * folder_find_item_from_identifier(const gchar *identifier)
1361 Xalloca(str, strlen(identifier) + 1, return NULL);
1362 strcpy(str, identifier);
1364 /* extract box type */
1366 p = strchr(str, '/');
1373 type = folder_get_type_from_string(str);
1377 /* extract box name */
1379 p = strchr(str, '/');
1388 folder = folder_find_from_name(name);
1394 d[0] = (gpointer)path;
1396 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
1397 folder_item_find_func, d);