fix another unnoticed leak; cleanup code
[claws.git] / src / selective_download.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2002 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 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #endif
23
24 #include "defs.h"
25
26 #include <glib.h>
27 #include <gtk/gtk.h>
28 #include <gtk/gtkoptionmenu.h>
29 #include <gdk/gdkkeysyms.h>
30 #include <stdio.h>
31 #include <stdlib.h>
32
33 #include "intl.h"
34 #include "main.h"
35 #include "prefs.h"
36 #include "prefs_matcher.h"
37 #include "prefs_filtering.h"
38 #include "prefs_account.h"
39 #include "account.h"
40 #include "mainwindow.h"
41 #include "manage_window.h"
42 #include "menu.h"
43 #include "stock_pixmap.h"
44 #include "inc.h"
45 #include "utils.h"
46 #include "gtkutils.h"
47 #include "alertpanel.h"
48 #include "folder.h"
49 #include "inc.h"
50 #include "filtering.h"
51 #include "matcher_parser.h"
52 #include "selective_download.h"
53 #include "procheader.h"
54
55 static struct _SDView {
56         MainWindow *mainwin;
57
58         GtkWidget *window;
59         GtkWidget *clist;
60         GtkWidget *preview_btn;
61         GtkWidget *remove_btn;
62         GtkWidget *download_btn;
63         GtkWidget *ac_button;
64         GtkWidget *ac_label;
65         GtkWidget *ac_menu;
66         GtkWidget *preview_popup;
67         GtkWidget *msgs_label;
68         GtkWidget *show_old_chkbtn;
69
70 }selective;
71
72 static GdkPixmap *checkboxonxpm;
73 static GdkBitmap *checkboxonxpmmask;
74 static GdkPixmap *checkboxoffxpm;
75 static GdkBitmap *checkboxoffxpmmask;
76 static GdkPixmap *markxpm;
77 static GdkBitmap *markxpmmask;
78 static GdkPixmap *deletedxpm;
79 static GdkBitmap *deletedxpmmask;
80 static GdkPixmap *continuexpm;
81 static GdkBitmap *continuexpmmask;
82
83
84 /* local functions */
85 static void sd_clear_msglist();
86 static void sd_remove_header_files();
87 static void sd_toggle_btn();
88 static SD_State sd_header_filter (MsgInfo *msginfo);
89 static MsgInfo *sd_get_msginfo_from_file (const gchar *filename);
90
91 static void sd_clist_set_pixmap(HeaderItems *items, gint row);
92 static void sd_clist_get_items();
93 static void sd_clist_set_items();
94 static void sd_update_msg_num(PrefsAccount *acc);
95
96 enum {
97         PREVIEW_NEW,
98         PREVIEW_ALL,
99         REMOVE,
100         DOWNLOAD,
101         DONE,
102         CHECKBTN,
103 };
104
105 /* callbacks */
106 static void sd_action_cb(GtkWidget *widget, guint action);
107
108 static void sd_select_row_cb (GtkCList *clist, gint row, gint column,
109                               GdkEvent *event, gpointer user_data);
110 static void sd_key_pressed (GtkWidget *widget,
111                             GdkEventKey *event,
112                             gpointer data);
113 /* account menu */
114 static void sd_ac_label_pressed(GtkWidget *widget, 
115                                 GdkEventButton *event,
116                                 gpointer data);
117
118 static void sd_ac_menu_popup_closed(GtkMenuShell *menu_shell);
119 static void sd_ac_menu_cb(GtkMenuItem *menuitem, gpointer data);
120 static void sd_ac_menu_set();
121
122 /* preview popup */
123 static void sd_preview_popup_closed(GtkMenuShell *menu_shell);
124 static void sd_preview_popup_cb(GtkWidget *widget, GdkEventButton *event);
125 static GtkItemFactoryEntry preview_popup_entries[] =
126 {
127         {N_("/Preview _new Messages"), NULL, sd_action_cb, PREVIEW_NEW, NULL},
128         {N_("/Preview _all Messages"), NULL, sd_action_cb, PREVIEW_ALL, NULL}
129 };
130
131 /* create dialog */
132 static void sd_window_create (MainWindow *mainwin);
133
134 void selective_download(MainWindow *mainwin)
135 {
136         summary_write_cache(mainwin->summaryview);
137
138         sd_remove_header_files();
139         
140         stock_pixmap_gdk(mainwin->window, STOCK_PIXMAP_CHECKBOX_OFF,
141                          &checkboxoffxpm, &checkboxoffxpmmask);
142         stock_pixmap_gdk(mainwin->window, STOCK_PIXMAP_CHECKBOX_ON,
143                          &checkboxonxpm, &checkboxonxpmmask);
144         stock_pixmap_gdk(mainwin->window, STOCK_PIXMAP_DELETED,
145                          &deletedxpm, &deletedxpmmask);
146         stock_pixmap_gdk(mainwin->window, STOCK_PIXMAP_CONTINUE,
147                          &continuexpm, &continuexpmmask);
148         stock_pixmap_gdk(mainwin->window, STOCK_PIXMAP_MARK,
149                          &markxpm, &markxpmmask);
150         inc_lock();
151
152         if (!selective.window)
153             sd_window_create(mainwin);
154
155         manage_window_set_transient(GTK_WINDOW(selective.window));
156         gtk_widget_show(selective.window);
157         sd_clear_msglist();
158         sd_ac_menu_set();
159         gtk_clist_clear(GTK_CLIST(selective.clist));
160
161
162 }
163
164 static void sd_clear_msglist()
165 {
166         PrefsAccount *acc = cur_account;
167         while (acc->msg_list != NULL) {
168                 HeaderItems *item = (HeaderItems*)acc->msg_list->data;
169
170                 acc->msg_list = g_slist_remove(acc->msg_list, item);
171                 g_free(item->from);
172                 g_free(item->subject);
173                 g_free(item->date);
174                 g_free(item);
175         }       
176         g_slist_free(acc->msg_list);
177         sd_update_msg_num(acc);
178 }
179
180 /* sd_remove_header_files()
181  * 
182  * - removes any stale header files in HEADER_CACHE_DIR 
183  *
184  */
185 static void sd_remove_header_files()
186 {
187         gchar *path = g_strconcat(get_header_cache_dir(), G_DIR_SEPARATOR_S, NULL);
188         
189         remove_all_files(path);
190         g_free(path);
191 }
192
193 /* sd_toggle_btn()
194  *
195  * - checks whether at least on email is selected 
196  *   if so, untoggle remove / download button
197  */
198 static void sd_toggle_btn()
199 {
200         PrefsAccount *acc = cur_account;
201         GSList *cur;
202         
203         for (cur = acc->msg_list; cur != NULL; cur = cur->next) {
204                 HeaderItems *items = (HeaderItems*)cur->data;
205                 
206                 if (items->state == SD_CHECKED) {
207                         gtk_widget_set_sensitive (selective.remove_btn, TRUE);
208                         gtk_widget_set_sensitive (selective.download_btn, TRUE);
209                         return;
210                 }
211         }
212
213         gtk_widget_set_sensitive (selective.remove_btn, FALSE);
214 }
215
216 /* sd_header_filter(MsgInfo *msginfo)
217  *
218  * - parse header line and look for any applying filtering rules
219  * - if message matches other MATCHACTION --> return
220  *
221  */
222 SD_State sd_header_filter(MsgInfo *msginfo)
223 {
224         GSList *rules;
225
226         /* parse header line for line */
227         for (rules = global_processing; rules != NULL; rules = rules->next) { 
228
229                 FilteringProp *prop = (FilteringProp*) rules->data; 
230
231                 if ( matcherlist_match(prop->matchers, msginfo) ) {
232                         if (prop->action->type == MATCHACTION_DELETE_ON_SERVER) {
233                                 debug_print(_("action matched\n"));
234                                 return SD_CHECKED;
235                         }
236                         else {
237                                 debug_print(_("action not matched\n"));
238                                 return SD_UNCHECKED;
239                         }
240                 }
241         }
242         return SD_UNCHECKED;
243 }
244
245 /* sd_get_msginfo_from_file(const gchar *filename)
246  *
247  * - retrieve msginfo from saved header files
248  */
249 static MsgInfo *sd_get_msginfo_from_file(const gchar *filename)
250 {
251         MsgInfo  *msginfo;
252         MsgFlags  msgflags = { 0, 0 };
253         gchar     date[80];
254
255         msginfo = procheader_parse_file(filename, msgflags, TRUE, FALSE);
256
257         if (msginfo && msginfo->date_t) {
258                 procheader_date_get_localtime(date, sizeof date, msginfo->date_t);
259                 if (msginfo->date) g_free(msginfo->date);
260                 msginfo->date = g_strdup(date);
261         } else 
262                 msginfo = g_new0(MsgInfo, 1);
263
264         if (!msginfo->date) 
265                 msginfo->date = g_strdup(_("(No Date)"));
266         if (!msginfo->from)
267                 msginfo->from = g_strdup(_("(No Sender)"));
268         if (!msginfo->subject)
269                 msginfo->subject = g_strdup(_("(No Subject)"));
270                 
271         return msginfo;
272 }
273
274 static void sd_clist_set_pixmap(HeaderItems *items, gint row)
275 {
276         
277         switch (items->state) {
278         case SD_REMOVED:
279                 gtk_clist_set_pixmap (GTK_CLIST (selective.clist), 
280                                       row, 0,
281                                      deletedxpm, deletedxpmmask);
282                 break;
283         case SD_CHECKED:
284                 gtk_clist_set_pixmap (GTK_CLIST (selective.clist), 
285                                       row, 0,
286                                       checkboxonxpm, checkboxonxpmmask);
287                 break;
288         case SD_DOWNLOADED:
289                 gtk_clist_set_pixmap (GTK_CLIST (selective.clist), 
290                                       row, 0,
291                                       markxpm, markxpmmask);
292                 break;          
293         default:
294                 gtk_clist_set_pixmap (GTK_CLIST (selective.clist), 
295                                       row, 0,
296                                       checkboxoffxpm, checkboxoffxpmmask);
297                 break;
298         }
299 }
300
301 static void sd_clist_set_items()
302 {
303         PrefsAccount *acc = cur_account;
304         GSList *cur;
305         gboolean show_old = 
306                 gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(selective.show_old_chkbtn));
307         
308         gtk_clist_clear(GTK_CLIST(selective.clist));
309         gtk_clist_freeze(GTK_CLIST(selective.clist));
310
311         for (cur = acc->msg_list; cur != NULL; cur = cur->next) {
312                 HeaderItems *items = (HeaderItems*)cur->data;
313                 gchar *row[5];
314                 gint row_num;
315                 
316                 row[0] = _("");
317                 row[1] = items->from;
318                 row[2] = items->subject;
319                 row[3] = items->date;
320                 row[4] = g_strdup_printf("%i KB", items->size/1024);
321                 
322                 switch (items->state) {
323                 case SD_REMOVED:
324                 case SD_DOWNLOADED:
325                         items->del_by_old_session = TRUE;
326                 default:
327                         break;
328                 }
329                 if (show_old) {
330                         if (items->received) {
331                                 row_num = gtk_clist_append(GTK_CLIST(selective.clist), row);
332                                 sd_clist_set_pixmap(items, row_num);
333                         }
334                 }
335                 else {
336                         row_num = gtk_clist_append(GTK_CLIST(selective.clist), row);
337                         sd_clist_set_pixmap(items, row_num);
338                 }
339                 g_free(row[4]);
340         }
341
342         gtk_clist_thaw(GTK_CLIST(selective.clist));
343         sd_toggle_btn();
344 }
345
346 /* sd_update_msg_num(PrefsAccount *acc)
347  * - keep track of msgs still on server
348  * - count UNCHECKED items as well as downloaded but not removed
349  */
350 static void sd_update_msg_num(PrefsAccount *acc)
351 {
352         GSList *cur;
353         gint msg_num = g_slist_length(acc->msg_list);
354         gchar *text;
355
356         for (cur = acc->msg_list; cur != NULL; cur = cur->next) {
357                 HeaderItems *items = (HeaderItems*) cur->data;
358
359                 if (items->state != SD_UNCHECKED)
360                         msg_num--;
361                 if (items->state == SD_DOWNLOADED)
362                         if (!acc->sd_rmmail_on_download)
363                                 msg_num++;
364         }
365
366         text = g_strdup_printf("%i Messages", msg_num);
367         gtk_label_set_text(GTK_LABEL(selective.msgs_label), text);
368
369         g_free(text);
370 }
371
372 /* sd_clist_get_items()
373  *
374  * - get items for clist from Files
375  */
376 static void sd_clist_get_items(void)
377 {
378         GSList *cur;
379         PrefsAccount *acc = cur_account;
380         gchar *path = g_strconcat(get_header_cache_dir(), G_DIR_SEPARATOR_S, NULL);
381         
382         for (cur = acc->msg_list; cur != NULL; cur = cur->next) {
383
384                 HeaderItems *items = (HeaderItems*) cur->data;
385                 gchar *filename    = g_strdup_printf("%s%i", path, items->index);
386                 MsgInfo *msginfo   = sd_get_msginfo_from_file(filename);
387                 
388                 items->from        = g_strdup(msginfo->from);
389                 items->subject     = g_strdup(msginfo->subject);
390                 items->date        = g_strdup(msginfo->date);
391                 
392                 msginfo->folder = folder_get_default_processing();
393
394                 /* move msg file to drop folder */
395                 if ((msginfo->msgnum = folder_item_add_msg(msginfo->folder, 
396                                                            filename, TRUE)) < 0) {
397                         unlink(filename);
398                         return;
399                 }
400                 
401                 if (acc->sd_filter_on_recv)
402                         items->state = sd_header_filter(msginfo);
403
404                 folder_item_remove_msg(msginfo->folder, msginfo->msgnum);
405                 g_free(filename);
406                 procmsg_msginfo_free(msginfo);
407         }
408
409         g_free(path);
410 }
411
412 static gint sd_deleted_cb(GtkWidget *widget, GdkEventAny *event, gpointer data)
413 {
414         sd_remove_header_files();
415         sd_clear_msglist();
416         gtk_widget_destroy(selective.window);   
417         selective.window = NULL;
418         inc_unlock();
419         return TRUE;
420 }
421
422 /* --- Callbacks -- */
423 static void sd_action_cb(GtkWidget *widget, guint action)
424 {
425         PrefsAccount *acc = cur_account;
426         
427         switch(action) {
428         case PREVIEW_NEW:
429         case PREVIEW_ALL:
430                 if ( (acc->protocol != A_APOP) &&
431                      (acc->protocol != A_POP3) ) {
432                         alertpanel_error(
433                                 _("Selected Account \"%s\" is not a POP Mail Server.\nPlease select a different Account"), acc->account_name);
434                         return;
435                 }
436                 sd_clear_msglist();
437                 if (action == PREVIEW_NEW) {
438                         gtk_widget_set_sensitive(selective.show_old_chkbtn, FALSE);
439                         inc_selective_download(selective.mainwin, acc, STYPE_PREVIEW_NEW);
440                 }
441                 else {
442                         gtk_widget_set_sensitive(selective.show_old_chkbtn, TRUE);
443                         inc_selective_download(selective.mainwin, acc, STYPE_PREVIEW_ALL);
444                 }
445                 
446                 gtk_clist_clear(GTK_CLIST(selective.clist));
447                 sd_clist_get_items();   
448                 sd_clist_set_items();
449                 break;
450         case REMOVE:
451                 inc_selective_download(selective.mainwin, acc, STYPE_DELETE);
452                 sd_clist_set_items();
453                 break;
454         case DOWNLOAD:
455                 inc_selective_download(selective.mainwin, acc, STYPE_DOWNLOAD);
456                 sd_clist_set_items();
457                 break;
458         case DONE:
459                 sd_remove_header_files();
460                 sd_clear_msglist();
461                 gtk_widget_hide(selective.window);
462                 inc_unlock();
463                 break;
464         case CHECKBTN:
465                 sd_clist_set_items();
466                 break;
467         default:
468                 break;
469         }
470         
471         sd_update_msg_num(acc);
472 }
473
474 /* Events */
475 static void sd_select_row_cb(GtkCList *clist, gint row, gint column,
476                              GdkEvent *event, gpointer user_data)
477 {
478         if ((row >= 0) && (column >= 0)) {
479                 PrefsAccount *acc  = cur_account;
480                 HeaderItems *items = (HeaderItems*) g_slist_nth_data (acc->msg_list, row);
481
482                 if (!items) return;
483                 if (!gtk_clist_get_selectable(GTK_CLIST(selective.clist), row)) return;
484
485                 if (items->state == SD_UNCHECKED) {
486                         items->state = SD_CHECKED;
487                         gtk_clist_set_pixmap (GTK_CLIST (selective.clist), row, 0,
488                                               checkboxonxpm, checkboxonxpmmask);
489                 } else if (items->state == SD_CHECKED) {
490                         items->state = SD_UNCHECKED;
491                         gtk_clist_set_pixmap (GTK_CLIST (selective.clist), row, 0,
492                                               checkboxoffxpm, checkboxoffxpmmask);
493                 }
494                 sd_toggle_btn();
495         }
496 }
497
498 static void sd_key_pressed(GtkWidget *widget,
499                            GdkEventKey *event,
500                            gpointer data)
501 {
502         if (event && event->keyval == GDK_Escape)
503                 sd_action_cb(widget, DONE);
504 }
505
506 /* account menu */
507 static void sd_ac_label_pressed(GtkWidget *widget, GdkEventButton *event,
508                                     gpointer data)
509 {
510         if (!event) return;
511
512         gtk_button_set_relief(GTK_BUTTON(widget), GTK_RELIEF_NORMAL);
513         gtk_object_set_data(GTK_OBJECT(selective.ac_menu), "menu_button",
514                             widget);
515
516         gtk_menu_popup(GTK_MENU(selective.ac_menu), NULL, NULL,
517                        menu_button_position, widget,
518                        event->button, event->time);
519 }
520
521 static void sd_ac_menu_popup_closed(GtkMenuShell *menu_shell)
522 {
523         GtkWidget *button;
524
525         button = gtk_object_get_data(GTK_OBJECT(menu_shell), "menu_button");
526         if (!button) return;
527         gtk_button_set_relief(GTK_BUTTON(button), GTK_RELIEF_NONE);
528         gtk_clist_clear(GTK_CLIST(selective.clist));
529         sd_clear_msglist();
530         sd_toggle_btn();
531         gtk_object_remove_data(GTK_OBJECT(selective.ac_menu), "menu_button");
532 }
533
534 static void sd_ac_menu_cb(GtkMenuItem *menuitem, gpointer data)
535 {
536         cur_account = (PrefsAccount *)data;
537         gtk_label_set_text(GTK_LABEL(selective.ac_label), cur_account->account_name);
538         gtk_widget_queue_resize(selective.ac_button);
539         main_window_reflect_prefs_all();
540 }
541
542 static void sd_ac_menu_set()
543 {
544         GList *cur_ac, *cur_item;
545         GtkWidget *menuitem;
546         PrefsAccount *ac_prefs;
547         GList *account_list = account_get_list();
548
549         /* destroy all previous menu item */
550         cur_item = GTK_MENU_SHELL(selective.ac_menu)->children;
551         while (cur_item != NULL) {
552                 GList *next = cur_item->next;
553                 gtk_widget_destroy(GTK_WIDGET(cur_item->data));
554                 cur_item = next;
555         }
556
557         gtk_label_set_text(GTK_LABEL(selective.ac_label), cur_account->account_name);
558
559         for (cur_ac = account_list; cur_ac != NULL; cur_ac = cur_ac->next) {
560                 ac_prefs = (PrefsAccount *)cur_ac->data;
561                 
562                 menuitem = gtk_menu_item_new_with_label
563                         (ac_prefs->account_name
564                          ? ac_prefs->account_name : _("Untitled"));
565                 gtk_widget_show(menuitem);
566                 gtk_menu_append(GTK_MENU(selective.ac_menu), menuitem);
567                 gtk_signal_connect(GTK_OBJECT(menuitem), "activate",
568                                            GTK_SIGNAL_FUNC(sd_ac_menu_cb),
569                                            ac_prefs);
570         }
571 }
572
573 /* receive button popup */
574 static void sd_preview_popup_closed(GtkMenuShell *menu_shell)
575 {
576         gtk_button_set_relief(GTK_BUTTON(selective.preview_btn), GTK_RELIEF_NORMAL);
577         manage_window_focus_in(selective.window, NULL, NULL);
578 }
579
580 static void sd_preview_popup_cb(GtkWidget *widget, GdkEventButton *event)
581 {
582         if (!event) return;
583         
584         if (event->button == 1) {
585                 gtk_button_set_relief(GTK_BUTTON(widget), GTK_RELIEF_NORMAL);
586                 gtk_menu_popup(GTK_MENU(selective.preview_popup), NULL, NULL,
587                        menu_button_position, widget,
588                        event->button, event->time);
589         }
590 }
591
592 static void sd_window_create(MainWindow *mainwin)
593 {
594         GtkWidget *window;
595         GtkWidget *table;
596         GtkWidget *msgs_label;
597         GtkWidget *bottom_hbox;
598         GtkWidget *fixed_label;
599         GtkWidget *expand_label;
600         GtkWidget *ac_button;
601         GtkWidget *ac_label;
602         GtkWidget *ac_menu;
603         GtkWidget *show_old_chkbtn;
604         GtkWidget *scrolledwindow;
605         GtkWidget *clist;
606         GtkWidget *state_label;
607         GtkWidget *from_label;
608         GtkWidget *subject_label;
609         GtkWidget *size_label;
610         GtkWidget *date_label;
611         GtkWidget *toolbar_hbox;
612         GtkWidget *toolbar;
613         GtkWidget *tmp_toolbar_icon;
614         GtkWidget *preview_btn;
615         GtkWidget *preview_popup;
616         GtkWidget *remove_btn;
617         GtkWidget *download_btn;
618         GtkWidget *done_btn;
619         gint n_menu_entries;
620
621         window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
622         gtk_object_set_data (GTK_OBJECT (window), "window", window);
623         gtk_window_set_title (GTK_WINDOW (window), _("Selective download"));
624         gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER);
625         gtk_window_set_modal (GTK_WINDOW (window), TRUE);
626         gtk_window_set_policy (GTK_WINDOW (window), TRUE, TRUE, TRUE);
627
628         table = gtk_table_new (2, 2, FALSE);
629         gtk_container_add (GTK_CONTAINER (window), table);
630
631         msgs_label = gtk_label_new (_("0 Messages"));
632         gtk_table_attach (GTK_TABLE (table), msgs_label, 1, 2, 1, 2,
633                           (GtkAttachOptions) (GTK_FILL),
634                           (GtkAttachOptions) (0), 0, 0);
635         gtk_misc_set_alignment (GTK_MISC (msgs_label), 0, 0.5);
636
637         bottom_hbox = gtk_hbox_new (FALSE, 0);
638         gtk_table_attach (GTK_TABLE (table), bottom_hbox, 0, 1, 1, 2,
639                           (GtkAttachOptions) (GTK_FILL),
640                           (GtkAttachOptions) (GTK_FILL), 0, 0);
641         
642         show_old_chkbtn = gtk_check_button_new_with_label("Show only old Messages"); 
643         gtk_box_pack_start(GTK_BOX(bottom_hbox), show_old_chkbtn, FALSE, FALSE, 0); 
644         
645         gtk_signal_connect(GTK_OBJECT(show_old_chkbtn), "toggled", 
646                            GTK_SIGNAL_FUNC(sd_action_cb), GUINT_TO_POINTER(CHECKBTN)); 
647         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(show_old_chkbtn), FALSE);
648         GTK_WIDGET_UNSET_FLAGS(show_old_chkbtn, GTK_CAN_FOCUS);                                                 
649                            
650         expand_label = gtk_label_new (_(" "));
651         gtk_box_pack_start (GTK_BOX (bottom_hbox), expand_label, TRUE, TRUE, 0);
652         
653         fixed_label = gtk_label_new (_(" contains "));
654         gtk_box_pack_end (GTK_BOX (bottom_hbox), fixed_label, FALSE, FALSE, 0);
655
656         ac_menu = gtk_menu_new();
657         gtk_signal_connect(GTK_OBJECT(ac_menu), "selection_done",
658                            GTK_SIGNAL_FUNC(sd_ac_menu_popup_closed), NULL);
659         ac_button = gtk_button_new();
660         gtk_button_set_relief(GTK_BUTTON(ac_button), GTK_RELIEF_NONE);
661         GTK_WIDGET_UNSET_FLAGS(ac_button, GTK_CAN_FOCUS);
662         gtk_widget_set_usize(ac_button, -1, 1);
663         gtk_box_pack_start(GTK_BOX(bottom_hbox), ac_button, FALSE, FALSE, 0);
664         gtk_signal_connect(GTK_OBJECT(ac_button), "button_press_event",
665                            GTK_SIGNAL_FUNC(sd_ac_label_pressed), GTK_OBJECT(ac_menu));
666
667         ac_label = gtk_label_new("");
668         gtk_container_add(GTK_CONTAINER(ac_button), ac_label);
669
670         scrolledwindow = gtk_scrolled_window_new (NULL, NULL);
671         gtk_table_attach (GTK_TABLE (table), scrolledwindow, 0, 1, 0, 1,
672                           (GtkAttachOptions) (GTK_EXPAND | GTK_FILL),
673                           (GtkAttachOptions) (GTK_EXPAND | GTK_FILL), 0, 0);
674         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow), 
675                                         GTK_POLICY_NEVER, 
676                                         GTK_POLICY_AUTOMATIC);
677         
678         clist = gtk_clist_new (5);
679         gtk_container_add (GTK_CONTAINER (scrolledwindow), clist);
680         gtk_container_set_border_width (GTK_CONTAINER (clist), 5);
681         gtk_clist_set_column_width (GTK_CLIST (clist), 0, 20);
682         gtk_clist_set_column_width (GTK_CLIST (clist), 1, 150);
683         gtk_clist_set_column_width (GTK_CLIST (clist), 2, 150);
684         gtk_clist_set_column_width (GTK_CLIST (clist), 3, 100);
685         gtk_clist_set_column_width (GTK_CLIST (clist), 4, 30);
686         gtk_clist_set_selection_mode (GTK_CLIST (clist), GTK_SELECTION_BROWSE);
687         gtk_clist_column_titles_show (GTK_CLIST (clist));
688         
689         state_label = gtk_label_new (_("#"));
690         gtk_clist_set_column_widget (GTK_CLIST (clist), 0, state_label);
691         
692         from_label = gtk_label_new (_("From"));
693         gtk_clist_set_column_widget (GTK_CLIST (clist), 1, from_label);
694         
695         subject_label = gtk_label_new (_("Subject"));
696         gtk_clist_set_column_widget (GTK_CLIST (clist), 2, subject_label);
697         
698         date_label = gtk_label_new (_("Date"));
699         gtk_clist_set_column_widget (GTK_CLIST (clist), 3, date_label);
700
701         size_label = gtk_label_new (_("Size"));
702         gtk_widget_ref (size_label);
703         gtk_clist_set_column_widget (GTK_CLIST (clist), 4, size_label);
704         
705         toolbar_hbox = gtk_hbox_new (FALSE, 0);
706
707         gtk_table_attach (GTK_TABLE (table), toolbar_hbox, 1, 2, 0, 1,
708                           (GtkAttachOptions) (GTK_FILL),
709                           (GtkAttachOptions) (GTK_FILL), 0, 0);
710         
711         toolbar = gtk_toolbar_new (GTK_ORIENTATION_VERTICAL, GTK_TOOLBAR_BOTH);
712
713         gtk_box_pack_end (GTK_BOX (toolbar_hbox), toolbar, FALSE, FALSE, 0);
714         gtk_toolbar_set_space_size (GTK_TOOLBAR (toolbar), 30);
715         gtk_toolbar_set_space_style (GTK_TOOLBAR (toolbar), GTK_TOOLBAR_SPACE_LINE);
716         gtk_container_set_border_width (GTK_CONTAINER (toolbar), 5);
717         
718         tmp_toolbar_icon = stock_pixmap_widget(toolbar_hbox, STOCK_PIXMAP_MAIL_RECEIVE);
719         preview_btn = gtk_toolbar_append_element (GTK_TOOLBAR (toolbar),
720                                                   GTK_TOOLBAR_CHILD_BUTTON,
721                                                   NULL,
722                                                   _("Preview Mail"),
723                                                   _("preview old/new E-Mail on account"), NULL,
724                                                   tmp_toolbar_icon, NULL, NULL);
725
726         n_menu_entries = sizeof(preview_popup_entries)/sizeof(preview_popup_entries[0]);
727         preview_popup = popupmenu_create(preview_btn, preview_popup_entries, n_menu_entries,
728                                       "<SelectiveDownload>", window);
729
730         gtk_signal_connect(GTK_OBJECT(preview_popup), "selection_done",
731                            GTK_SIGNAL_FUNC(sd_preview_popup_closed), NULL);
732
733         gtk_toolbar_append_space (GTK_TOOLBAR (toolbar));
734         
735         tmp_toolbar_icon = stock_pixmap_widget(toolbar_hbox, STOCK_PIXMAP_CLOSE);
736         remove_btn = gtk_toolbar_append_element (GTK_TOOLBAR (toolbar),
737                                                  GTK_TOOLBAR_CHILD_BUTTON,
738                                                  NULL,
739                                                  _("Remove"),
740                                                  _("remove selected E-Mails"), NULL,
741                                                  tmp_toolbar_icon, NULL, NULL);
742
743         gtk_widget_set_sensitive (remove_btn, FALSE);
744
745         tmp_toolbar_icon = stock_pixmap_widget(toolbar_hbox, STOCK_PIXMAP_DOWN_ARROW);
746         download_btn = gtk_toolbar_append_element (GTK_TOOLBAR (toolbar),
747                                                  GTK_TOOLBAR_CHILD_BUTTON,
748                                                  NULL,
749                                                  _("Download"),
750                                                  _("Download selected E-Mails"), NULL,
751                                                  tmp_toolbar_icon, NULL, NULL);
752
753         gtk_widget_set_sensitive (download_btn, FALSE);
754
755         gtk_toolbar_append_space (GTK_TOOLBAR (toolbar));
756
757         tmp_toolbar_icon = stock_pixmap_widget (toolbar_hbox, STOCK_PIXMAP_COMPLETE);
758         done_btn = gtk_toolbar_append_element (GTK_TOOLBAR (toolbar),
759                                                GTK_TOOLBAR_CHILD_BUTTON,
760                                                NULL,
761                                                _("Done"),
762                                                _("Exit Dialog"), NULL,
763                                                tmp_toolbar_icon, NULL, NULL);
764         
765         gtk_signal_connect (GTK_OBJECT (window), "delete_event",
766                             GTK_SIGNAL_FUNC (gtk_widget_hide_on_delete),
767                             NULL);
768         gtk_signal_connect (GTK_OBJECT(window), "key_press_event",
769                             GTK_SIGNAL_FUNC(sd_key_pressed),
770                             NULL);
771         MANAGE_WINDOW_SIGNALS_CONNECT(window);
772         gtk_signal_connect (GTK_OBJECT (preview_btn), "button_press_event",
773                             GTK_SIGNAL_FUNC (sd_preview_popup_cb),
774                             NULL);
775
776         gtk_signal_connect (GTK_OBJECT (remove_btn), "clicked",
777                             GTK_SIGNAL_FUNC (sd_action_cb),
778                             GUINT_TO_POINTER(REMOVE));
779         gtk_signal_connect (GTK_OBJECT (download_btn), "clicked",
780                             GTK_SIGNAL_FUNC (sd_action_cb),
781                             GUINT_TO_POINTER(DOWNLOAD));
782         gtk_signal_connect (GTK_OBJECT (done_btn), "clicked",
783                             GTK_SIGNAL_FUNC (sd_action_cb),
784                             GUINT_TO_POINTER(DONE));
785         gtk_signal_connect (GTK_OBJECT (clist), "select_row",
786                             GTK_SIGNAL_FUNC (sd_select_row_cb),
787                             NULL);
788
789
790         selective.mainwin         = mainwin;
791         selective.window          = window;
792         selective.clist           = clist;
793         selective.preview_btn     = preview_btn;
794         selective.remove_btn      = remove_btn;
795         selective.download_btn    = download_btn;
796         selective.ac_label        = ac_label;
797         selective.ac_button       = ac_button;
798         selective.ac_menu         = ac_menu;
799         selective.preview_popup   = preview_popup;
800         selective.msgs_label      = msgs_label;
801         selective.show_old_chkbtn = show_old_chkbtn;
802
803         gtk_widget_show_all(window);
804 }
805