2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2003 Hiroyuki Yamamoto and the Sylpheed-Claws team
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.
28 #include <gtk/gtkoptionmenu.h>
29 #include <gdk/gdkkeysyms.h>
37 #include "prefs_gtk.h"
38 #include "prefs_matcher.h"
39 #include "prefs_filtering.h"
40 #include "prefs_common.h"
41 #include "mainwindow.h"
42 #include "foldersel.h"
43 #include "manage_window.h"
47 #include "alertpanel.h"
49 #include "filtering.h"
50 #include "addr_compl.h"
51 #include "colorlabel.h"
53 #include "matcher_parser.h"
55 #include "prefs_filtering_action.h"
57 static struct Filtering {
61 GtkWidget *cond_entry;
62 GtkWidget *action_entry;
64 GtkWidget *cond_clist;
67 static GSList ** p_processing_list = NULL;
69 /* widget creating functions */
70 static void prefs_filtering_create (void);
72 static void prefs_filtering_set_dialog (const gchar *header,
74 static void prefs_filtering_set_list (void);
76 /* callback functions */
77 static void prefs_filtering_register_cb (void);
78 static void prefs_filtering_substitute_cb (void);
79 static void prefs_filtering_delete_cb (void);
80 static void prefs_filtering_top (void);
81 static void prefs_filtering_up (void);
82 static void prefs_filtering_down (void);
83 static void prefs_filtering_bottom (void);
84 static void prefs_filtering_select (GtkCList *clist,
89 static gint prefs_filtering_deleted (GtkWidget *widget,
92 static gboolean prefs_filtering_key_pressed(GtkWidget *widget,
95 static void prefs_filtering_cancel (void);
96 static void prefs_filtering_ok (void);
98 static void prefs_filtering_condition_define (void);
99 static void prefs_filtering_action_define(void);
100 static gint prefs_filtering_clist_set_row (gint row, FilteringProp * prop);
102 static void prefs_filtering_reset_dialog (void);
103 static gboolean prefs_filtering_rename_path_func(GNode *node, gpointer data);
104 static gboolean prefs_filtering_delete_path_func(GNode *node, gpointer data);
106 static void delete_path(GSList ** p_filters, const gchar * path);
108 void prefs_filtering_open(GSList ** p_processing,
113 if (prefs_rc_is_readonly(FILTERING_RC))
118 if (!filtering.window) {
119 prefs_filtering_create();
122 manage_window_set_transient(GTK_WINDOW(filtering.window));
123 gtk_widget_grab_focus(filtering.ok_btn);
126 gtk_window_set_title(GTK_WINDOW(filtering.window), title);
128 gtk_window_set_title (GTK_WINDOW(filtering.window),
129 _("Filtering/Processing configuration"));
131 p_processing_list = p_processing;
133 prefs_filtering_set_dialog(header, key);
135 gtk_widget_show(filtering.window);
137 start_address_completion();
140 /* prefs_filtering_close() - just to have one common exit point */
141 static void prefs_filtering_close(void)
143 end_address_completion();
145 gtk_widget_hide(filtering.window);
149 static void prefs_filtering_create(void)
154 GtkWidget *cancel_btn;
155 GtkWidget *confirm_area;
163 GtkWidget *cond_label;
164 GtkWidget *cond_entry;
166 GtkWidget *action_label;
167 GtkWidget *action_entry;
168 GtkWidget *action_btn;
171 GtkWidget *subst_btn;
174 GtkWidget *cond_hbox;
175 GtkWidget *cond_scrolledwin;
176 GtkWidget *cond_clist;
183 GtkWidget *bottom_btn;
187 debug_print("Creating filtering configuration window...\n");
189 window = gtk_window_new (GTK_WINDOW_DIALOG);
190 gtk_container_set_border_width (GTK_CONTAINER (window), 8);
191 gtk_window_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER);
192 gtk_window_set_modal (GTK_WINDOW (window), TRUE);
193 gtk_window_set_policy (GTK_WINDOW (window), FALSE, TRUE, FALSE);
195 vbox = gtk_vbox_new (FALSE, 6);
196 gtk_widget_show (vbox);
197 gtk_container_add (GTK_CONTAINER (window), vbox);
199 gtkut_button_set_create(&confirm_area, &ok_btn, _("OK"),
200 &cancel_btn, _("Cancel"), NULL, NULL);
201 gtk_widget_show (confirm_area);
202 gtk_box_pack_end (GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
203 gtk_widget_grab_default (ok_btn);
205 gtk_window_set_title (GTK_WINDOW(window),
206 _("Filtering/Processing configuration"));
208 gtk_signal_connect (GTK_OBJECT(window), "delete_event",
209 GTK_SIGNAL_FUNC(prefs_filtering_deleted), NULL);
210 gtk_signal_connect (GTK_OBJECT(window), "key_press_event",
211 GTK_SIGNAL_FUNC(prefs_filtering_key_pressed), NULL);
212 MANAGE_WINDOW_SIGNALS_CONNECT (window);
213 gtk_signal_connect (GTK_OBJECT(ok_btn), "clicked",
214 GTK_SIGNAL_FUNC(prefs_filtering_ok), NULL);
215 gtk_signal_connect (GTK_OBJECT(cancel_btn), "clicked",
216 GTK_SIGNAL_FUNC(prefs_filtering_cancel), NULL);
218 vbox1 = gtk_vbox_new (FALSE, VSPACING);
219 gtk_widget_show (vbox1);
220 gtk_box_pack_start (GTK_BOX (vbox), vbox1, TRUE, TRUE, 0);
221 gtk_container_set_border_width (GTK_CONTAINER (vbox1), 2);
223 cond_label = gtk_label_new (_("Condition"));
224 gtk_widget_show (cond_label);
225 gtk_misc_set_alignment (GTK_MISC (cond_label), 0, 0.5);
226 gtk_box_pack_start (GTK_BOX (vbox1), cond_label, FALSE, FALSE, 0);
228 hbox1 = gtk_hbox_new (FALSE, VSPACING);
229 gtk_widget_show (hbox1);
230 gtk_box_pack_start (GTK_BOX (vbox1), hbox1, FALSE, FALSE, 0);
231 gtk_container_set_border_width (GTK_CONTAINER (vbox1), 2);
233 cond_entry = gtk_entry_new ();
234 gtk_widget_show (cond_entry);
235 gtk_box_pack_start (GTK_BOX (hbox1), cond_entry, TRUE, TRUE, 0);
237 cond_btn = gtk_button_new_with_label (_("Define ..."));
238 gtk_widget_show (cond_btn);
239 gtk_box_pack_start (GTK_BOX (hbox1), cond_btn, FALSE, FALSE, 0);
240 gtk_signal_connect (GTK_OBJECT (cond_btn), "clicked",
241 GTK_SIGNAL_FUNC (prefs_filtering_condition_define),
244 action_label = gtk_label_new (_("Action"));
245 gtk_widget_show (action_label);
246 gtk_misc_set_alignment (GTK_MISC (action_label), 0, 0.5);
247 gtk_box_pack_start (GTK_BOX (vbox1), action_label, FALSE, FALSE, 0);
249 hbox1 = gtk_hbox_new (FALSE, VSPACING);
250 gtk_widget_show (hbox1);
251 gtk_box_pack_start (GTK_BOX (vbox1), hbox1, FALSE, FALSE, 0);
252 gtk_container_set_border_width (GTK_CONTAINER (vbox1), 2);
254 action_entry = gtk_entry_new ();
255 gtk_widget_show (action_entry);
256 gtk_box_pack_start (GTK_BOX (hbox1), action_entry, TRUE, TRUE, 0);
258 action_btn = gtk_button_new_with_label (_("Define ..."));
259 gtk_widget_show (action_btn);
260 gtk_box_pack_start (GTK_BOX (hbox1), action_btn, FALSE, FALSE, 0);
261 gtk_signal_connect (GTK_OBJECT (action_btn), "clicked",
262 GTK_SIGNAL_FUNC (prefs_filtering_action_define),
265 /* register / substitute / delete */
267 reg_hbox = gtk_hbox_new (FALSE, 4);
268 gtk_widget_show (reg_hbox);
269 gtk_box_pack_start (GTK_BOX (vbox1), reg_hbox, FALSE, FALSE, 0);
271 arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_OUT);
272 gtk_widget_show (arrow);
273 gtk_box_pack_start (GTK_BOX (reg_hbox), arrow, FALSE, FALSE, 0);
274 gtk_widget_set_usize (arrow, -1, 16);
276 btn_hbox = gtk_hbox_new (TRUE, 4);
277 gtk_widget_show (btn_hbox);
278 gtk_box_pack_start (GTK_BOX (reg_hbox), btn_hbox, FALSE, FALSE, 0);
280 reg_btn = gtk_button_new_with_label (_("Add"));
281 gtk_widget_show (reg_btn);
282 gtk_box_pack_start (GTK_BOX (btn_hbox), reg_btn, FALSE, TRUE, 0);
283 gtk_signal_connect (GTK_OBJECT (reg_btn), "clicked",
284 GTK_SIGNAL_FUNC (prefs_filtering_register_cb), NULL);
286 subst_btn = gtk_button_new_with_label (_(" Replace "));
287 gtk_widget_show (subst_btn);
288 gtk_box_pack_start (GTK_BOX (btn_hbox), subst_btn, FALSE, TRUE, 0);
289 gtk_signal_connect (GTK_OBJECT (subst_btn), "clicked",
290 GTK_SIGNAL_FUNC (prefs_filtering_substitute_cb),
293 del_btn = gtk_button_new_with_label (_("Delete"));
294 gtk_widget_show (del_btn);
295 gtk_box_pack_start (GTK_BOX (btn_hbox), del_btn, FALSE, TRUE, 0);
296 gtk_signal_connect (GTK_OBJECT (del_btn), "clicked",
297 GTK_SIGNAL_FUNC (prefs_filtering_delete_cb), NULL);
299 cond_hbox = gtk_hbox_new (FALSE, 8);
300 gtk_widget_show (cond_hbox);
301 gtk_box_pack_start (GTK_BOX (vbox1), cond_hbox, TRUE, TRUE, 0);
303 cond_scrolledwin = gtk_scrolled_window_new (NULL, NULL);
304 gtk_widget_show (cond_scrolledwin);
305 gtk_widget_set_usize (cond_scrolledwin, -1, 150);
306 gtk_box_pack_start (GTK_BOX (cond_hbox), cond_scrolledwin,
308 gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (cond_scrolledwin),
309 GTK_POLICY_AUTOMATIC,
310 GTK_POLICY_AUTOMATIC);
312 title[0] = _("Current filtering/processing rules");
313 cond_clist = gtk_clist_new_with_titles(1, title);
314 gtk_widget_show (cond_clist);
315 gtk_container_add (GTK_CONTAINER (cond_scrolledwin), cond_clist);
316 gtk_clist_set_column_width (GTK_CLIST (cond_clist), 0, 80);
317 gtk_clist_set_selection_mode (GTK_CLIST (cond_clist),
318 GTK_SELECTION_BROWSE);
319 GTK_WIDGET_UNSET_FLAGS (GTK_CLIST (cond_clist)->column[0].button,
321 gtk_signal_connect (GTK_OBJECT (cond_clist), "select_row",
322 GTK_SIGNAL_FUNC (prefs_filtering_select), NULL);
324 btn_vbox = gtk_vbox_new (FALSE, 8);
325 gtk_widget_show (btn_vbox);
326 gtk_box_pack_start (GTK_BOX (cond_hbox), btn_vbox, FALSE, FALSE, 0);
328 top_btn = gtk_button_new_with_label (_("Top"));
329 gtk_widget_show (top_btn);
330 gtk_box_pack_start (GTK_BOX (btn_vbox), top_btn, FALSE, FALSE, 0);
331 gtk_signal_connect (GTK_OBJECT (top_btn), "clicked",
332 GTK_SIGNAL_FUNC (prefs_filtering_top), NULL);
334 PACK_VSPACER (btn_vbox, spc_vbox, VSPACING_NARROW_2);
336 up_btn = gtk_button_new_with_label (_("Up"));
337 gtk_widget_show (up_btn);
338 gtk_box_pack_start (GTK_BOX (btn_vbox), up_btn, FALSE, FALSE, 0);
339 gtk_signal_connect (GTK_OBJECT (up_btn), "clicked",
340 GTK_SIGNAL_FUNC (prefs_filtering_up), NULL);
342 down_btn = gtk_button_new_with_label (_("Down"));
343 gtk_widget_show (down_btn);
344 gtk_box_pack_start (GTK_BOX (btn_vbox), down_btn, FALSE, FALSE, 0);
345 gtk_signal_connect (GTK_OBJECT (down_btn), "clicked",
346 GTK_SIGNAL_FUNC (prefs_filtering_down), NULL);
348 PACK_VSPACER (btn_vbox, spc_vbox, VSPACING_NARROW_2);
350 bottom_btn = gtk_button_new_with_label (_("Bottom"));
351 gtk_widget_show (bottom_btn);
352 gtk_box_pack_start (GTK_BOX (btn_vbox), bottom_btn, FALSE, FALSE, 0);
353 gtk_signal_connect (GTK_OBJECT (bottom_btn), "clicked",
354 GTK_SIGNAL_FUNC (prefs_filtering_bottom), NULL);
356 gtk_widget_set_usize(window, 500, -1);
358 gtk_widget_show_all(window);
360 filtering.window = window;
361 filtering.ok_btn = ok_btn;
363 filtering.cond_entry = cond_entry;
364 filtering.action_entry = action_entry;
365 filtering.cond_clist = cond_clist;
368 static void prefs_filtering_update_hscrollbar(void)
370 gint optwidth = gtk_clist_optimal_column_width(GTK_CLIST(filtering.cond_clist), 0);
371 gtk_clist_set_column_width(GTK_CLIST(filtering.cond_clist), 0, optwidth);
374 static void rename_path(GSList * filters,
375 const gchar * old_path, const gchar * new_path);
377 void prefs_filtering_rename_path(const gchar *old_path, const gchar *new_path)
380 const gchar *paths[2] = {NULL, NULL};
383 for (cur = folder_get_list() ; cur != NULL ; cur = g_list_next(cur)) {
385 folder = (Folder *) cur->data;
386 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
387 prefs_filtering_rename_path_func, paths);
390 rename_path(pre_global_processing, old_path, new_path);
391 rename_path(post_global_processing, old_path, new_path);
392 rename_path(filtering_rules, old_path, new_path);
394 prefs_matcher_write_config();
397 static void rename_path(GSList * filters,
398 const gchar * old_path, const gchar * new_path)
404 gchar *old_path_with_sep;
411 oldpathlen = strlen(old_path);
412 old_path_with_sep = g_strconcat(old_path,G_DIR_SEPARATOR_S,NULL);
414 for (cur = filters; cur != NULL; cur = cur->next) {
415 FilteringProp *filtering = (FilteringProp *)cur->data;
417 for(action_cur = filtering->action_list ; action_cur != NULL ;
418 action_cur = action_cur->next) {
420 FilteringAction *action = action_cur->data;
422 if (!action->destination) continue;
424 destlen = strlen(action->destination);
426 if (destlen > oldpathlen) {
427 prefixlen = destlen - oldpathlen;
428 suffix = action->destination + prefixlen;
430 if (!strncmp(old_path, suffix, oldpathlen)) {
431 prefix = g_malloc0(prefixlen + 1);
432 strncpy2(prefix, action->destination, prefixlen);
434 base = suffix + oldpathlen;
435 while (*base == G_DIR_SEPARATOR) base++;
437 dest_path = g_strconcat(prefix,
441 dest_path = g_strconcat(prefix,
448 g_free(action->destination);
449 action->destination = dest_path;
450 } else { /* for non-leaf folders */
451 /* compare with trailing slash */
452 if (!strncmp(old_path_with_sep, action->destination, oldpathlen+1)) {
454 suffix = action->destination + oldpathlen + 1;
455 dest_path = g_strconcat(new_path,
458 g_free(action->destination);
459 action->destination = dest_path;
463 /* folder-moving a leaf */
464 if (!strcmp(old_path, action->destination)) {
465 dest_path = g_strdup(new_path);
466 g_free(action->destination);
467 action->destination = dest_path;
474 static gboolean prefs_filtering_rename_path_func(GNode *node, gpointer data)
477 const gchar * old_path;
478 const gchar * new_path;
479 const gchar ** paths;
486 g_return_val_if_fail(old_path != NULL, FALSE);
487 g_return_val_if_fail(new_path != NULL, FALSE);
488 g_return_val_if_fail(node != NULL, FALSE);
491 if (!item || !item->prefs)
493 filters = item->prefs->processing;
495 rename_path(filters, old_path, new_path);
500 void prefs_filtering_delete_path(const gchar *path)
503 for (cur = folder_get_list() ; cur != NULL ; cur = g_list_next(cur)) {
505 folder = (Folder *) cur->data;
506 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
507 prefs_filtering_delete_path_func, (gchar *)path);
509 delete_path(&pre_global_processing, path);
510 delete_path(&post_global_processing, path);
511 delete_path(&filtering_rules, path);
513 prefs_matcher_write_config();
516 static void delete_path(GSList ** p_filters, const gchar * path)
527 filters = *p_filters;
528 pathlen = strlen(path);
529 duplist = g_slist_copy(filters);
530 for (cur = duplist ; cur != NULL; cur = g_slist_next(cur)) {
531 FilteringProp *filtering = (FilteringProp *) cur->data;
533 for(action_cur = filtering->action_list ; action_cur != NULL ;
534 action_cur = action_cur->next) {
536 FilteringAction *action;
538 action = action_cur->data;
540 if (!action->destination) continue;
542 destlen = strlen(action->destination);
544 if (destlen > pathlen) {
545 prefixlen = destlen - pathlen;
546 suffix = action->destination + prefixlen;
548 if (suffix && !strncmp(path, suffix, pathlen)) {
549 filteringprop_free(filtering);
550 filters = g_slist_remove(filters, filtering);
552 } else if (strcmp(action->destination, path) == 0) {
553 filteringprop_free(filtering);
554 filters = g_slist_remove(filters, filtering);
558 g_slist_free(duplist);
560 * p_filters = filters;
563 static gboolean prefs_filtering_delete_path_func(GNode *node, gpointer data)
565 const gchar *path = data;
569 g_return_val_if_fail(path != NULL, FALSE);
570 g_return_val_if_fail(node != NULL, FALSE);
573 if (!item || !item->prefs)
575 p_filters = &item->prefs->processing;
577 delete_path(p_filters, path);
582 static void prefs_filtering_set_dialog(const gchar *header, const gchar *key)
584 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
586 GSList * prefs_filtering;
590 gtk_clist_freeze(clist);
591 gtk_clist_clear(clist);
593 cond_str[0] = _("(New)");
594 row = gtk_clist_append(clist, cond_str);
595 gtk_clist_set_row_data(clist, row, NULL);
597 prefs_filtering = * p_processing_list;
599 for(cur = prefs_filtering ; cur != NULL ; cur = g_slist_next(cur)) {
600 FilteringProp * prop = (FilteringProp *) cur->data;
602 cond_str[0] = filteringprop_to_string(prop);
603 subst_char(cond_str[0], '\t', ':');
604 row = gtk_clist_append(clist, cond_str);
605 gtk_clist_set_row_data(clist, row, prop);
610 prefs_filtering_update_hscrollbar();
611 gtk_clist_thaw(clist);
613 prefs_filtering_reset_dialog();
619 quoted_key = matcher_quote_str(key);
621 match_str = g_strconcat(header, " ", get_matchparser_tab_str(MATCHTYPE_MATCHCASE),
622 " \"", quoted_key, "\"", NULL);
625 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), match_str);
630 static void prefs_filtering_reset_dialog(void)
632 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), "");
633 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), "");
636 static void prefs_filtering_set_list(void)
641 gchar * filtering_str;
642 GSList * prefs_filtering;
644 prefs_filtering = * p_processing_list;
646 for(cur = prefs_filtering ; cur != NULL ; cur = g_slist_next(cur))
647 filteringprop_free((FilteringProp *) cur->data);
648 g_slist_free(prefs_filtering);
649 prefs_filtering = NULL;
651 while (gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
652 row, 0, &filtering_str)) {
653 if (strcmp(filtering_str, _("(New)")) != 0) {
654 prop = matcher_parser_get_filtering(filtering_str);
657 g_slist_append(prefs_filtering, prop);
662 * p_processing_list = prefs_filtering;
665 static gint prefs_filtering_clist_set_row(gint row, FilteringProp * prop)
667 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
672 cond_str[0] = _("(New)");
673 return gtk_clist_append(clist, cond_str);
676 str = filteringprop_to_string(prop);
683 row = gtk_clist_append(clist, cond_str);
685 gtk_clist_set_text(clist, row, 0, cond_str[0]);
691 static void prefs_filtering_condition_define_done(MatcherList * matchers)
695 if (matchers == NULL)
698 str = matcherlist_to_string(matchers);
701 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), str);
706 static void prefs_filtering_condition_define(void)
709 MatcherList * matchers = NULL;
711 cond_str = gtk_entry_get_text(GTK_ENTRY(filtering.cond_entry));
713 if (*cond_str != '\0') {
714 matchers = matcher_parser_get_cond(cond_str);
715 if (matchers == NULL)
716 alertpanel_error(_("Condition string is not valid."));
719 prefs_matcher_open(matchers, prefs_filtering_condition_define_done);
721 if (matchers != NULL)
722 matcherlist_free(matchers);
725 static void prefs_filtering_action_define_done(GSList * action_list)
729 if (action_list == NULL)
732 str = filteringaction_list_to_string(action_list);
735 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), str);
740 static void prefs_filtering_action_define(void)
743 GSList * action_list = NULL;
745 action_str = gtk_entry_get_text(GTK_ENTRY(filtering.action_entry));
747 if (*action_str != '\0') {
748 action_list = matcher_parser_get_action_list(action_str);
749 if (action_list == NULL)
750 alertpanel_error(_("Action string is not valid."));
753 prefs_filtering_action_open(action_list,
754 prefs_filtering_action_define_done);
756 if (action_list != NULL) {
758 for(cur = action_list ; cur != NULL ; cur = cur->next) {
759 filteringaction_free(cur->data);
765 /* register / substitute delete buttons */
768 static FilteringProp * prefs_filtering_dialog_to_filtering(gboolean alert)
773 FilteringProp * prop;
774 GSList * action_list;
776 cond_str = gtk_entry_get_text(GTK_ENTRY(filtering.cond_entry));
777 if (*cond_str == '\0') {
778 if(alert == TRUE) alertpanel_error(_("Condition string is empty."));
782 action_str = gtk_entry_get_text(GTK_ENTRY(filtering.action_entry));
783 if (*action_str == '\0') {
784 if(alert == TRUE) alertpanel_error(_("Action string is empty."));
788 cond = matcher_parser_get_cond(cond_str);
791 if(alert == TRUE) alertpanel_error(_("Condition string is not valid."));
795 action_list = matcher_parser_get_action_list(action_str);
797 if (action_list == NULL) {
798 if(alert == TRUE) alertpanel_error(_("Action string is not valid."));
802 prop = filteringprop_new(cond, action_list);
807 static void prefs_filtering_register_cb(void)
809 FilteringProp * prop;
811 prop = prefs_filtering_dialog_to_filtering(TRUE);
814 prefs_filtering_clist_set_row(-1, prop);
816 filteringprop_free(prop);
818 prefs_filtering_update_hscrollbar();
821 static void prefs_filtering_substitute_cb(void)
823 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
825 FilteringProp * prop;
827 if (!clist->selection) return;
829 row = GPOINTER_TO_INT(clist->selection->data);
830 if (row == 0) return;
832 prop = prefs_filtering_dialog_to_filtering(TRUE);
835 prefs_filtering_clist_set_row(row, prop);
837 filteringprop_free(prop);
839 prefs_filtering_update_hscrollbar();
842 static void prefs_filtering_delete_cb(void)
844 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
847 if (!clist->selection) return;
848 row = GPOINTER_TO_INT(clist->selection->data);
849 if (row == 0) return;
851 if (alertpanel(_("Delete rule"),
852 _("Do you really want to delete this rule?"),
853 _("Yes"), _("No"), NULL) == G_ALERTALTERNATE)
856 gtk_clist_remove(clist, row);
858 prefs_filtering_reset_dialog();
860 prefs_filtering_update_hscrollbar();
863 static void prefs_filtering_top(void)
865 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
868 if (!clist->selection) return;
870 row = GPOINTER_TO_INT(clist->selection->data);
872 gtk_clist_row_move(clist, row, 1);
875 static void prefs_filtering_up(void)
877 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
880 if (!clist->selection) return;
882 row = GPOINTER_TO_INT(clist->selection->data);
884 gtk_clist_row_move(clist, row, row - 1);
885 if (gtk_clist_row_is_visible(clist, row - 1) != GTK_VISIBILITY_FULL)
886 gtk_clist_moveto(clist, row - 1, 0, 0, 0);
890 static void prefs_filtering_down(void)
892 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
895 if (!clist->selection) return;
897 row = GPOINTER_TO_INT(clist->selection->data);
898 if (row > 0 && row < clist->rows - 1) {
899 gtk_clist_row_move(clist, row, row + 1);
900 if (gtk_clist_row_is_visible(clist, row + 1) != GTK_VISIBILITY_FULL)
901 gtk_clist_moveto(clist, row + 1, 0, 1, 0);
905 static void prefs_filtering_bottom(void)
907 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
910 if (!clist->selection) return;
912 row = GPOINTER_TO_INT(clist->selection->data);
913 if (row > 0 && row < clist->rows - 1)
914 gtk_clist_row_move(clist, row, clist->rows - 1);
917 static void prefs_filtering_select_set(FilteringProp *prop)
922 prefs_filtering_reset_dialog();
924 matcher_str = matcherlist_to_string(prop->matchers);
925 if (matcher_str == NULL) {
929 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), matcher_str);
931 action_str = filteringaction_list_to_string(prop->action_list);
932 if (matcher_str == NULL) {
935 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), action_str);
941 static void prefs_filtering_select(GtkCList *clist, gint row, gint column,
944 FilteringProp * prop;
945 gchar * filtering_str;
948 prefs_filtering_reset_dialog();
952 if (!gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
953 row, 0, &filtering_str))
956 prop = matcher_parser_get_filtering(filtering_str);
960 prefs_filtering_select_set(prop);
962 filteringprop_free(prop);
966 static gint prefs_filtering_deleted(GtkWidget *widget, GdkEventAny *event,
969 prefs_filtering_cancel();
973 static gboolean prefs_filtering_key_pressed(GtkWidget *widget, GdkEventKey *event,
976 if (event && event->keyval == GDK_Escape)
977 prefs_filtering_cancel();
981 static void prefs_filtering_ok(void)
983 FilteringProp * prop;
985 gchar * filtering_str;
989 prop = prefs_filtering_dialog_to_filtering(FALSE);
991 str = filteringprop_to_string(prop);
993 while (gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
994 row, 0, &filtering_str)) {
995 if (strcmp(filtering_str, str) == 0) break;
998 if (!filtering_str || strcmp(filtering_str, str) != 0) {
999 val = alertpanel(_("Entry not saved"),
1000 _("The entry was not saved. Close anyway?"),
1001 _("Yes"), _("No"), NULL);
1002 if (G_ALERTDEFAULT != val) {
1004 filteringprop_free(prop);
1009 filteringprop_free(prop);
1011 prefs_filtering_set_list();
1012 prefs_matcher_write_config();
1013 prefs_filtering_close();
1016 static void prefs_filtering_cancel(void)
1018 prefs_matcher_read_config();
1019 prefs_filtering_close();