managesieve: remove dialog hint
[claws.git] / src / plugins / managesieve / sieve_manager.c
1 /*
2  * Claws Mail -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 2004-2015 the Claws Mail team
4  * Copyright (C) 2014-2015 Charles Lehner
5  *
6  * This program is free software; you can redistribute it and/or modify
7  * it under the terms of the GNU General Public License as published by
8  * the Free Software Foundation; either version 3 of the License, or
9  * (at your option) any later version.
10  *
11  * This program is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  * GNU General Public License for more details.
15  *
16  * You should have received a copy of the GNU General Public License
17  * along with this program. If not, see <http://www.gnu.org/licenses/>.
18  * 
19  */
20
21 #ifdef HAVE_CONFIG_H
22 #include "config.h"
23 #include "claws-features.h"
24 #endif
25
26 #include <gtk/gtk.h>
27 #include <glib.h>
28 #include <glib/gi18n.h>
29
30 #include "defs.h"
31 #include "gtk/gtkutils.h"
32 #include "gtk/combobox.h"
33 #include "gtk/inputdialog.h"
34 #include "gtk/manage_window.h"
35 #include "alertpanel.h"
36 #include "utils.h"
37 #include "prefs.h"
38 #include "account.h"
39 #include "mainwindow.h"
40 #include "managesieve.h"
41 #include "sieve_editor.h"
42 #include "sieve_prefs.h"
43 #include "sieve_manager.h"
44
45 enum {
46         FILTER_NAME,
47         FILTER_ACTIVE,
48         N_FILTER_COLUMNS
49 };
50
51 typedef struct {
52         SieveManagerPage *page;
53         gchar *name_old;
54         gchar *name_new;
55 } CommandDataRename;
56
57 typedef struct {
58         SieveManagerPage *page;
59         gchar *filter_name;
60 } CommandDataName;
61
62 typedef struct {
63         SieveManagerPage *page;
64         gchar *filter_name;
65         SieveEditorPage *editor_page;
66         gboolean first_line;
67 } CommandDataGetScript;
68
69 static void account_changed(GtkWidget *widget, SieveManagerPage *page);
70 void sieve_manager_close(GtkWidget *widget, SieveManagerPage *page);
71 static void filter_set_active(SieveManagerPage *page, gchar *filter_name);
72 gboolean filter_find_by_name (GtkTreeModel *model, GtkTreeIter *iter,
73                 gchar *filter_name);
74 static void got_session_error(SieveSession *session, const gchar *msg,
75                 SieveManagerPage *page);
76
77 static GSList *manager_pages = NULL;
78
79 /*
80  * Perform a command on all manager pages for a given session
81  */
82 #define manager_sessions_foreach(cur, session, page) \
83         for(cur = manager_pages; cur != NULL; cur = cur->next) \
84                 if ((page = (SieveManagerPage *)cur->data) && \
85                         page->active_session == session)
86
87 static void filters_list_clear(SieveManagerPage *page)
88 {
89         GtkListStore *list_store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list)));
90         gtk_list_store_clear(list_store);
91         page->got_list = FALSE;
92 }
93
94 static void filters_list_delete_filter(SieveManagerPage *page, gchar *name)
95 {
96         GtkTreeIter iter;
97         GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list));
98
99         if (!filter_find_by_name(model, &iter, name))
100                 return;
101
102         gtk_list_store_remove(GTK_LIST_STORE(model), &iter);
103 }
104
105
106 static void filters_list_rename_filter(SieveManagerPage *page,
107                 gchar *name_old, char *name_new)
108 {
109         GtkTreeIter iter;
110         GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list));
111
112         if (!filter_find_by_name(model, &iter, name_old))
113                 return;
114
115         gtk_list_store_set(GTK_LIST_STORE(model), &iter,
116                         FILTER_NAME, name_new,
117                         -1);
118 }
119
120 static void filters_list_insert_filter(SieveManagerPage *page,
121                 SieveScript *filter)
122 {
123         GtkListStore *list_store = GTK_LIST_STORE(gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list)));
124         GtkTreeIter iter;
125
126         gtk_list_store_append(list_store, &iter);
127         gtk_list_store_set(list_store, &iter,
128                         FILTER_NAME, filter->name,
129                         FILTER_ACTIVE, filter->active,
130                         -1);
131 }
132
133 static gchar *filters_list_get_selected_filter(GtkWidget *list_view)
134 {
135         GtkTreeSelection *selector;
136         GtkTreeModel *model;
137         GtkTreeIter iter;
138         gchar *res = NULL;
139
140         selector = gtk_tree_view_get_selection(GTK_TREE_VIEW(list_view));
141
142         if (!gtk_tree_selection_get_selected(selector, &model, &iter))
143                 return NULL;
144
145         gtk_tree_model_get(model, &iter, FILTER_NAME, &res, -1);
146
147         return res;
148 }
149
150 static void filter_add(GtkWidget *widget, SieveManagerPage *page)
151 {
152         SieveSession *session = page->active_session;
153         if (!session)
154                 return;
155         gchar *filter_name = input_dialog(_("Add Sieve script"),
156                         _("Enter name for a new Sieve filter script."), "");
157         if (!filter_name || !filter_name[0])
158                 return;
159
160         sieve_editor_new(session, filter_name);
161         /*
162         sieve_session_add_script(session, filter_name
163                         (sieve_session_data_cb_fn)filter_added, (gpointer)page);
164                         */
165 }
166
167 static void filter_got_data(SieveSession *session, gchar *contents,
168                 CommandDataGetScript *cmd_data)
169 {
170         SieveManagerPage *page = cmd_data->page;
171         SieveEditorPage *editor;
172
173         if (!contents) {
174                 g_free(cmd_data);
175                 return;
176         } else if (contents == (void *)-1) {
177                 got_session_error(session, _("Unable to get script contents"), page);
178                 return;
179         }
180
181         if (cmd_data->first_line) {
182                 cmd_data->first_line = FALSE;
183                 editor = sieve_editor_new(session, cmd_data->filter_name);
184                 cmd_data->editor_page = editor;
185         } else {
186                 editor = cmd_data->editor_page;
187                 sieve_editor_append_text(editor, "\n", 1);
188         }
189         sieve_editor_append_text(editor, contents, strlen(contents));
190 }
191
192 static void filter_edit(GtkWidget *widget, SieveManagerPage *page)
193 {
194         SieveEditorPage *editor;
195         CommandDataGetScript *cmd_data;
196         SieveSession *session = page->active_session;
197         if (!session)
198                 return;
199         gchar *filter_name = filters_list_get_selected_filter(page->filters_list);
200         if (!filter_name)
201                 return;
202
203         editor = sieve_editor_get(session, filter_name);
204         if (editor) {
205                 sieve_editor_present(editor);
206         } else {
207                 cmd_data = g_new0(CommandDataGetScript, 1);
208                 cmd_data->first_line = TRUE;
209                 cmd_data->filter_name = filter_name;
210                 cmd_data->page = page;
211
212                 sieve_session_get_script(session, filter_name,
213                         (sieve_session_data_cb_fn)filter_got_data, cmd_data);
214         }
215 }
216
217 static void filter_renamed(SieveSession *session, gboolean success,
218                 CommandDataRename *data)
219 {
220         SieveManagerPage *page = data->page;
221         GSList *cur;
222
223         if (!success) {
224                 got_session_error(session, "Unable to rename script", page);
225                 return;
226         }
227
228         manager_sessions_foreach(cur, session, page) {
229                 filters_list_rename_filter(page, data->name_old, data->name_new);
230         }
231         g_free(data);
232 }
233
234 static void filter_rename(GtkWidget *widget, SieveManagerPage *page)
235 {
236         CommandDataRename *cmd_data;
237         gchar *name_old, *name_new;
238         SieveSession *session;
239
240         name_old = filters_list_get_selected_filter(page->filters_list);
241         if (!name_old)
242                 return;
243
244         session = page->active_session;
245         if (!session)
246                 return;
247
248         name_new = input_dialog(_("Add Sieve script"),
249                         _("Enter new name for the script."), name_old);
250         if (!name_new)
251                 return;
252
253         cmd_data = g_new(CommandDataRename, 1);
254         cmd_data->name_new = name_new;
255         cmd_data->name_old = name_old;
256         cmd_data->page = page;
257         sieve_session_rename_script(session, name_old, name_new,
258                         (sieve_session_data_cb_fn)filter_renamed, (gpointer)cmd_data);
259 }
260
261 static void filter_activated(SieveSession *session, gboolean success,
262                 CommandDataName *cmd_data)
263 {
264         SieveManagerPage *page = cmd_data->page;
265         GSList *cur;
266
267         if (!success) {
268                 got_session_error(session, "Unable to set active script", page);
269                 return;
270         }
271
272         manager_sessions_foreach(cur, session, page) {
273                 filter_set_active(page, cmd_data->filter_name);
274         }
275         g_free(cmd_data);
276 }
277
278 static void sieve_set_active_filter(SieveManagerPage *page, gchar *filter_name)
279 {
280         SieveSession *session;
281         CommandDataName *cmd_data;
282
283         session = page->active_session;
284         cmd_data = g_new(CommandDataName, 1);
285         cmd_data->filter_name = filter_name;
286         cmd_data->page = page;
287
288         sieve_session_set_active_script(session, filter_name,
289                         (sieve_session_data_cb_fn)filter_activated, cmd_data);
290 }
291
292 static void filter_deleted(SieveSession *session, const gchar *err_msg,
293                 CommandDataName *cmd_data)
294 {
295         SieveManagerPage *page = cmd_data->page;
296         GSList *cur;
297
298         if (err_msg) {
299                 got_session_error(session, err_msg, page);
300                 return;
301         }
302
303         manager_sessions_foreach(cur, session, page) {
304                 filters_list_delete_filter(page, cmd_data->filter_name);
305         }
306         g_free(cmd_data);
307 }
308
309
310 static void filter_delete(GtkWidget *widget, SieveManagerPage *page)
311 {
312         gchar buf[256];
313         gchar *filter_name;
314         SieveSession *session;
315         CommandDataName *cmd_data;
316
317         filter_name = filters_list_get_selected_filter(page->filters_list);
318         if (filter_name == NULL)
319                 return;
320
321         session = page->active_session;
322         if (!session)
323                 return;
324
325         g_snprintf(buf, sizeof(buf),
326                    _("Do you really want to delete the filter '%s'?"), filter_name);
327         if (alertpanel_full(_("Delete filter"), buf,
328                                 GTK_STOCK_CANCEL, GTK_STOCK_DELETE, NULL, FALSE,
329                                 NULL, ALERT_WARNING, G_ALERTDEFAULT) != G_ALERTALTERNATE)
330                 return;
331
332         cmd_data = g_new(CommandDataName, 1);
333         cmd_data->filter_name = filter_name;
334         cmd_data->page = page;
335
336         sieve_session_delete_script(session, filter_name,
337                         (sieve_session_data_cb_fn)filter_deleted, cmd_data);
338 }
339
340 /*
341  * select a filter in the list
342  *
343  * return TRUE is successfully selected, FALSE otherwise
344  */
345
346 static gboolean filter_select (GtkWidget *list_view, GtkTreeModel *model,
347                 GtkTreeIter *iter)
348 {
349         GtkTreeSelection *selection;
350         GtkTreePath* path;
351
352         selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(list_view));
353         gtk_tree_selection_select_iter(selection, iter);
354         path = gtk_tree_model_get_path(model, iter);
355         if (path == NULL) return FALSE;
356         gtk_tree_view_set_cursor(GTK_TREE_VIEW(list_view), path, NULL, FALSE);
357         gtk_tree_path_free(path);
358         return TRUE;
359 }
360
361 /*
362  * find matching filter. return FALSE on match
363  */
364 static gboolean filter_search_equal_fn (GtkTreeModel *model, gint column,
365                 const gchar *key, GtkTreeIter *iter, gpointer search_data)
366 {
367         SieveManagerPage *page = (SieveManagerPage *)search_data;
368         gchar *filter_name;
369
370         if (!key) return TRUE;
371
372         gtk_tree_model_get (model, iter, FILTER_NAME, &filter_name, -1);
373
374         if (strncmp (key, filter_name, strlen(key)) != 0) return TRUE;
375         return !filter_select(page->filters_list, model, iter);
376 }
377
378 /*
379  * search for a filter row by its name. return true if found.
380  */
381 gboolean filter_find_by_name (GtkTreeModel *model, GtkTreeIter *iter,
382                 gchar *filter_name)
383 {
384         gchar *name;
385
386         if (!gtk_tree_model_get_iter_first (model, iter))
387                 return FALSE;
388
389         do {
390                 gtk_tree_model_get (model, iter, FILTER_NAME, &name, -1);
391                 if (strcmp(filter_name, name) == 0) {
392                         return TRUE;
393                 }
394         } while (gtk_tree_model_iter_next (model, iter));
395         return FALSE;
396 }
397
398 static gboolean filter_set_inactive(GtkTreeModel *model,
399                 GtkTreePath *path, GtkTreeIter *iter, gpointer data)
400 {
401         gtk_list_store_set(GTK_LIST_STORE(model), iter,
402                         FILTER_ACTIVE, FALSE,
403                         -1);
404         return FALSE;
405 }
406
407 /*
408  * Set the active filter in the table.
409  * @param filter_name The filter to make active (may be null)
410  */
411 static void filter_set_active(SieveManagerPage *page, gchar *filter_name)
412 {
413         GtkTreeIter iter;
414         GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list));
415
416         /* Deactivate all filters */
417         gtk_tree_model_foreach(model, filter_set_inactive, NULL);
418
419         /* Set active filter */
420         if (filter_name) {
421                 if (!filter_find_by_name (model, &iter, filter_name))
422                         return;
423
424                 gtk_list_store_set(GTK_LIST_STORE(model), &iter,
425                                 FILTER_ACTIVE, TRUE,
426                                 -1);
427         }
428 }
429
430 static void filter_active_toggled(GtkCellRendererToggle *widget,
431                                     gchar *path,
432                                     SieveManagerPage *page)
433 {
434         GtkTreeIter iter;
435         gchar *filter_name;
436         gboolean active;
437         GtkTreeModel *model = gtk_tree_view_get_model(GTK_TREE_VIEW(page->filters_list));
438
439         if (!gtk_tree_model_get_iter_from_string(model, &iter, path))
440                 return;
441
442         /* get existing value */
443         gtk_tree_model_get(model, &iter,
444                         FILTER_NAME, &filter_name,
445                         FILTER_ACTIVE, &active,
446                         -1);
447
448         sieve_set_active_filter(page, active ? NULL : filter_name);
449 }
450
451 static void filter_double_clicked(GtkTreeView *list_view,
452                                    GtkTreePath *path, GtkTreeViewColumn *column,
453                                    SieveManagerPage *page)
454 {
455         filter_edit(GTK_WIDGET(list_view), page);
456 }
457
458 static void filters_create_list_view_columns(SieveManagerPage *page,
459                 GtkWidget *list_view)
460 {
461         GtkTreeViewColumn *column;
462         GtkCellRenderer *renderer;
463
464         /* Name */
465         renderer = gtk_cell_renderer_text_new();
466         column = gtk_tree_view_column_new_with_attributes
467                 (_("Name"), renderer,
468                  "text", FILTER_NAME,
469                  NULL);
470         gtk_tree_view_append_column(GTK_TREE_VIEW(list_view), column);
471         gtk_tree_view_column_set_expand(column, TRUE);
472
473         /* Active */
474         renderer = gtk_cell_renderer_toggle_new();
475         g_object_set(renderer,
476                      "radio", TRUE,
477                      "activatable", TRUE,
478                       NULL);
479         column = gtk_tree_view_column_new_with_attributes
480                 (_("Active"), renderer,
481                  "active", FILTER_ACTIVE,
482                  NULL);
483         gtk_tree_view_append_column(GTK_TREE_VIEW(list_view), column);
484         gtk_tree_view_column_set_alignment (column, 0.5);
485         CLAWS_SET_TIP(gtk_tree_view_column_get_widget(column),
486                         _("An account can only have one active script at a time."));
487         g_signal_connect(G_OBJECT(renderer), "toggled",
488                          G_CALLBACK(filter_active_toggled), page);
489
490         gtk_tree_view_set_search_column(GTK_TREE_VIEW(list_view), FILTER_NAME);
491         gtk_tree_view_set_search_equal_func(GTK_TREE_VIEW(list_view),
492                         filter_search_equal_fn, page, NULL);
493 }
494
495
496 static GtkListStore* filters_create_data_store(void)
497 {
498         return gtk_list_store_new(N_FILTER_COLUMNS,
499                         G_TYPE_STRING,  /* FILTER_NAME */
500                         G_TYPE_BOOLEAN, /* FILTER_ACTIVE */
501                         -1);
502 }
503
504 static GtkWidget *filters_list_view_create(SieveManagerPage *page)
505 {
506         GtkTreeView *list_view;
507         GtkTreeSelection *selector;
508         GtkListStore *store = filters_create_data_store();
509
510         list_view = GTK_TREE_VIEW(gtk_tree_view_new_with_model(GTK_TREE_MODEL(store)));
511         g_object_unref(G_OBJECT(store));
512
513         selector = gtk_tree_view_get_selection(list_view);
514         gtk_tree_selection_set_mode(selector, GTK_SELECTION_BROWSE);
515
516         /* create the columns */
517         filters_create_list_view_columns(page, GTK_WIDGET(list_view));
518
519         /* set a double click listener */
520         g_signal_connect(G_OBJECT(list_view), "row_activated",
521                         G_CALLBACK(filter_double_clicked),
522                         page);
523
524         return GTK_WIDGET(list_view);
525 }
526
527 static gboolean manager_key_pressed(GtkWidget *widget, GdkEventKey *event,
528                                     gpointer data)
529 {
530         SieveManagerPage* page = (SieveManagerPage *) data;
531
532         if (event && event->keyval == GDK_KEY_Escape)
533                 sieve_manager_done(page);
534
535         return FALSE;
536 }
537
538 static void size_allocate_cb(GtkWidget *widget, GtkAllocation *allocation)
539 {
540         cm_return_if_fail(allocation != NULL);
541
542         sieve_config.manager_win_width = allocation->width;
543         sieve_config.manager_win_height = allocation->height;
544 }
545
546 static void got_session_error(SieveSession *session, const gchar *msg,
547                 SieveManagerPage *page)
548 {
549         if (page->active_session != session)
550                 return;
551         gtk_label_set_text(GTK_LABEL(page->status_text), msg);
552 }
553
554 static void sieve_manager_on_error(SieveSession *session,
555                 const gchar *msg, gpointer user_data)
556 {
557         SieveManagerPage *page = (SieveManagerPage *)user_data;
558         got_session_error(session, msg, page);
559 }
560
561 static void sieve_manager_on_connected(SieveSession *session,
562                 gboolean connected, gpointer user_data)
563 {
564         SieveManagerPage *page = (SieveManagerPage *)user_data;
565         if (page->active_session != session)
566                 return;
567         if (!connected) {
568                 gtk_widget_set_sensitive(GTK_WIDGET(page->vbox_buttons), FALSE);
569                 gtk_label_set_text(GTK_LABEL(page->status_text),
570                                 _("Unable to connect"));
571         }
572 }
573
574 static void got_filter_listed(SieveSession *session, SieveScript *script,
575                 SieveManagerPage *page)
576 {
577         if (!script) {
578                 got_session_error(session, "Unable to list scripts", page);
579                 return;
580         }
581         if (!script->name) {
582                 /* done receiving list */
583                 page->got_list = TRUE;
584                 gtk_widget_set_sensitive(GTK_WIDGET(page->vbox_buttons), TRUE);
585                 gtk_label_set_text(GTK_LABEL(page->status_text), "");
586                 return;
587         }
588         filters_list_insert_filter(page, script);
589 }
590
591 /*
592  * An account was selected from the menu. Get its list of scripts.
593  */
594 static void account_changed(GtkWidget *widget, SieveManagerPage *page)
595 {
596         gint account_id;
597         PrefsAccount *account;
598         SieveSession *session;
599
600         account_id = combobox_get_active_data(GTK_COMBO_BOX(page->accounts_menu));
601         account = account_find_from_id(account_id);
602         if (!account)
603                 return;
604         session = page->active_session = sieve_session_get_for_account(account);
605         sieve_session_handle_status(session,
606                         sieve_manager_on_error,
607                         sieve_manager_on_connected,
608                         page);
609         filters_list_clear(page);
610         if (session_is_connected(SESSION(session))) {
611                 gtk_label_set_text(GTK_LABEL(page->status_text),
612                                 _("Listing scripts..."));
613         } else {
614                 gtk_label_set_text(GTK_LABEL(page->status_text),
615                                 _("Connecting..."));
616         }
617         sieve_session_list_scripts(session,
618                         (sieve_session_data_cb_fn)got_filter_listed, (gpointer)page);
619 }
620
621 static SieveManagerPage *sieve_manager_page_new()
622 {
623         SieveManagerPage *page;
624         GtkWidget *window;
625         GtkWidget *hbox, *vbox, *vbox_buttons;
626         GtkWidget *accounts_menu;
627         GtkWidget *label;
628         GtkWidget *scrolledwin;
629         GtkWidget *list_view;
630         GtkWidget *btn;
631         GtkWidget *status_text;
632         GtkTreeIter iter;
633         GtkListStore *menu;
634         GList *account_list, *cur;
635         PrefsAccount *ap;
636         SieveAccountConfig *config;
637         PrefsAccount *default_account = NULL;
638
639         static GdkGeometry geometry;
640
641         page = g_new0(SieveManagerPage, 1);
642
643         /* Manage Window */
644
645         window = gtkut_window_new(GTK_WINDOW_TOPLEVEL, "sievemanager");
646         gtk_container_set_border_width (GTK_CONTAINER (window), 8);
647         gtk_window_set_title (GTK_WINDOW (window), _("Manage Sieve Filters"));
648         MANAGE_WINDOW_SIGNALS_CONNECT (window);
649
650         g_signal_connect (G_OBJECT (window), "key_press_event",
651                         G_CALLBACK (manager_key_pressed), page);
652         g_signal_connect (G_OBJECT(window), "size_allocate",
653                          G_CALLBACK (size_allocate_cb), NULL);
654
655         if (!geometry.min_height) {
656                 geometry.min_width = 350;
657                 geometry.min_height = 280;
658         }
659
660         gtk_window_set_geometry_hints(GTK_WINDOW(window), NULL, &geometry,
661                                       GDK_HINT_MIN_SIZE);
662         gtk_widget_set_size_request(window, sieve_config.manager_win_width,
663                         sieve_config.manager_win_height);
664
665         vbox = gtk_vbox_new (FALSE, 10);
666         gtk_container_add (GTK_CONTAINER (window), vbox);
667
668         hbox = gtk_hbox_new (FALSE, 8);
669         gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
670
671         /* Accounts list */
672
673         label = gtk_label_new (_("Account"));
674         gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, FALSE, 0);
675
676         accounts_menu = gtkut_sc_combobox_create(NULL, FALSE);
677         menu = GTK_LIST_STORE(gtk_combo_box_get_model(GTK_COMBO_BOX(accounts_menu)));
678         gtk_box_pack_start (GTK_BOX (hbox), accounts_menu, FALSE, FALSE, 0);
679         g_signal_connect (G_OBJECT(accounts_menu), "changed",
680                           G_CALLBACK (account_changed), page);
681
682         account_list = account_get_list();
683         for (cur = account_list; cur != NULL; cur = cur->next) {
684                 ap = (PrefsAccount *)cur->data;
685                 config = sieve_prefs_account_get_config(ap);
686                 if (config->enable) {
687                         COMBOBOX_ADD (menu, ap->account_name, ap->account_id);
688                         if (!default_account || ap->is_default)
689                                 default_account = ap;
690                 }
691         }
692
693         if (!default_account) {
694                 gtk_widget_destroy(label);
695                 gtk_widget_destroy(accounts_menu);
696         }
697
698         /* status */
699         status_text = gtk_label_new ("");
700         gtk_box_pack_start (GTK_BOX (hbox), status_text, FALSE, FALSE, 0);
701         gtk_label_set_justify (GTK_LABEL (status_text), GTK_JUSTIFY_LEFT);
702
703         /* Filters list */
704
705         hbox = gtk_hbox_new (FALSE, 8);
706         gtk_box_pack_start (GTK_BOX (vbox), hbox, TRUE, TRUE, 0);
707         gtk_container_set_border_width (GTK_CONTAINER (hbox), 2);
708
709         /* Table */
710
711         scrolledwin = gtk_scrolled_window_new (NULL, NULL);
712         gtk_box_pack_start (GTK_BOX (hbox), scrolledwin, TRUE, TRUE, 0);
713         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwin),
714                                         GTK_POLICY_AUTOMATIC,
715                                         GTK_POLICY_AUTOMATIC);
716
717         list_view = filters_list_view_create(page);
718         gtk_container_add(GTK_CONTAINER(scrolledwin), list_view);
719
720         /* Buttons */
721
722         vbox_buttons = gtk_vbox_new (FALSE, 8);
723         gtk_widget_set_sensitive(vbox_buttons, FALSE);
724         gtk_box_pack_start (GTK_BOX (hbox), vbox_buttons, FALSE, FALSE, 0);
725
726         /* new */
727         btn = gtk_button_new_from_stock(GTK_STOCK_NEW);
728         gtk_box_pack_start (GTK_BOX (vbox_buttons), btn, FALSE, FALSE, 0);
729         g_signal_connect (G_OBJECT(btn), "clicked",
730                           G_CALLBACK (filter_add), page);
731
732         /* edit */
733         btn = gtk_button_new_from_stock (GTK_STOCK_EDIT);
734         gtk_box_pack_start (GTK_BOX (vbox_buttons), btn, FALSE, FALSE, 0);
735         g_signal_connect (G_OBJECT(btn), "clicked",
736                         G_CALLBACK (filter_edit), page);
737
738         /* delete */
739         btn = gtk_button_new_from_stock(GTK_STOCK_DELETE);
740         gtk_box_pack_start (GTK_BOX (vbox_buttons), btn, FALSE, FALSE, 0);
741         g_signal_connect (G_OBJECT(btn), "clicked",
742                         G_CALLBACK (filter_delete), page);
743
744         /* rename */
745         btn = gtk_button_new_with_label("Rename");
746         gtk_box_pack_start (GTK_BOX (vbox_buttons), btn, FALSE, FALSE, 0);
747         g_signal_connect (G_OBJECT(btn), "clicked",
748                         G_CALLBACK (filter_rename), page);
749
750         /* refresh */
751         btn = gtk_button_new_from_stock(GTK_STOCK_REFRESH);
752         gtk_box_pack_end (GTK_BOX (vbox_buttons), btn, FALSE, FALSE, 0);
753         g_signal_connect (G_OBJECT(btn), "clicked",
754                         G_CALLBACK (account_changed), page);
755
756         /* bottom area stuff */
757
758         gtkut_stock_button_set_create(&hbox,
759                         &btn, GTK_STOCK_CLOSE,
760                         NULL, NULL, NULL, NULL);
761
762         /* close */
763         gtk_box_pack_end (GTK_BOX (vbox), hbox, FALSE, FALSE, 0);
764         gtk_widget_grab_default (btn);
765         g_signal_connect (G_OBJECT (btn), "clicked",
766                           G_CALLBACK (sieve_manager_close), page);
767
768         page->window = window;
769         page->accounts_menu = accounts_menu;
770         page->filters_list = list_view;
771         page->status_text = status_text;
772         page->vbox_buttons = vbox_buttons;
773
774         /* select default (first) account */
775         if (default_account) {
776                 combobox_select_by_data(GTK_COMBO_BOX(accounts_menu),
777                                 default_account->account_id);
778         } else {
779                 gtk_label_set_text(GTK_LABEL(status_text),
780                                 _("To use Sieve, enable it in an account's preferences."));
781         }
782
783         return page;
784 }
785
786 void sieve_manager_close(GtkWidget *widget, SieveManagerPage *page)
787 {
788         sieve_manager_done(page);
789 }
790
791 void sieve_manager_done(SieveManagerPage *page)
792 {
793         manager_pages = g_slist_remove(manager_pages, page);
794         gtk_widget_destroy(page->window);
795         g_free(page);
796 }
797
798 void sieve_manager_show()
799 {
800         SieveManagerPage *page = sieve_manager_page_new();
801         manager_pages = g_slist_prepend(manager_pages, page);
802         gtk_widget_show_all(page->window);
803 }