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;
412 oldpathlen = strlen(old_path);
413 old_path_with_sep = g_strconcat(old_path,G_DIR_SEPARATOR_S,NULL);
415 for (cur = filters; cur != NULL; cur = cur->next) {
416 FilteringProp *filtering = (FilteringProp *)cur->data;
418 for(action_cur = filtering->action_list ; action_cur != NULL ;
419 action_cur = action_cur->next) {
421 FilteringAction *action = action_cur->data;
423 if (!action->destination) continue;
425 destlen = strlen(action->destination);
427 if (destlen > oldpathlen) {
428 prefixlen = destlen - oldpathlen;
429 suffix = action->destination + prefixlen;
431 if (!strncmp(old_path, suffix, oldpathlen)) {
432 prefix = g_malloc0(prefixlen + 1);
433 strncpy2(prefix, action->destination, prefixlen);
435 base = suffix + oldpathlen;
436 while (*base == G_DIR_SEPARATOR) base++;
438 dest_path = g_strconcat(prefix,
442 dest_path = g_strconcat(prefix,
449 g_free(action->destination);
450 action->destination = dest_path;
451 } else { /* for non-leaf folders */
452 /* compare with trailing slash */
453 if (!strncmp(old_path_with_sep, action->destination, oldpathlen+1)) {
455 suffix = action->destination + oldpathlen + 1;
456 dest_path = g_strconcat(new_path,
459 g_free(action->destination);
460 action->destination = dest_path;
464 /* folder-moving a leaf */
465 if (!strcmp(old_path, action->destination)) {
466 dest_path = g_strdup(new_path);
467 g_free(action->destination);
468 action->destination = dest_path;
475 static gboolean prefs_filtering_rename_path_func(GNode *node, gpointer data)
478 const gchar * old_path;
479 const gchar * new_path;
480 const gchar ** paths;
487 g_return_val_if_fail(old_path != NULL, FALSE);
488 g_return_val_if_fail(new_path != NULL, FALSE);
489 g_return_val_if_fail(node != NULL, FALSE);
492 if (!item || !item->prefs)
494 filters = item->prefs->processing;
496 rename_path(filters, old_path, new_path);
501 void prefs_filtering_delete_path(const gchar *path)
504 for (cur = folder_get_list() ; cur != NULL ; cur = g_list_next(cur)) {
506 folder = (Folder *) cur->data;
507 g_node_traverse(folder->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
508 prefs_filtering_delete_path_func, (gchar *)path);
510 delete_path(&pre_global_processing, path);
511 delete_path(&post_global_processing, path);
512 delete_path(&filtering_rules, path);
514 prefs_matcher_write_config();
517 static void delete_path(GSList ** p_filters, const gchar * path)
529 filters = *p_filters;
530 pathlen = strlen(path);
531 duplist = g_slist_copy(filters);
532 for (cur = duplist ; cur != NULL; cur = g_slist_next(cur)) {
533 FilteringProp *filtering = (FilteringProp *) cur->data;
535 for(action_cur = filtering->action_list ; action_cur != NULL ;
536 action_cur = action_cur->next) {
538 FilteringAction *action;
540 action = action_cur->data;
542 if (!action->destination) continue;
544 destlen = strlen(action->destination);
546 if (destlen > pathlen) {
547 prefixlen = destlen - pathlen;
548 suffix = action->destination + prefixlen;
550 if (suffix && !strncmp(path, suffix, pathlen)) {
551 filteringprop_free(filtering);
552 filters = g_slist_remove(filters, filtering);
554 } else if (strcmp(action->destination, path) == 0) {
555 filteringprop_free(filtering);
556 filters = g_slist_remove(filters, filtering);
560 g_slist_free(duplist);
562 * p_filters = filters;
565 static gboolean prefs_filtering_delete_path_func(GNode *node, gpointer data)
567 const gchar *path = data;
571 g_return_val_if_fail(path != NULL, FALSE);
572 g_return_val_if_fail(node != NULL, FALSE);
575 if (!item || !item->prefs)
577 p_filters = &item->prefs->processing;
579 delete_path(p_filters, path);
584 static void prefs_filtering_set_dialog(const gchar *header, const gchar *key)
586 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
588 GSList * prefs_filtering;
592 gtk_clist_freeze(clist);
593 gtk_clist_clear(clist);
595 cond_str[0] = _("(New)");
596 row = gtk_clist_append(clist, cond_str);
597 gtk_clist_set_row_data(clist, row, NULL);
599 prefs_filtering = * p_processing_list;
601 for(cur = prefs_filtering ; cur != NULL ; cur = g_slist_next(cur)) {
602 FilteringProp * prop = (FilteringProp *) cur->data;
604 cond_str[0] = filteringprop_to_string(prop);
605 subst_char(cond_str[0], '\t', ':');
606 row = gtk_clist_append(clist, cond_str);
607 gtk_clist_set_row_data(clist, row, prop);
612 prefs_filtering_update_hscrollbar();
613 gtk_clist_thaw(clist);
615 prefs_filtering_reset_dialog();
621 quoted_key = matcher_quote_str(key);
623 match_str = g_strconcat(header, " ", get_matchparser_tab_str(MATCHTYPE_MATCHCASE),
624 " \"", quoted_key, "\"", NULL);
627 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), match_str);
632 static void prefs_filtering_reset_dialog(void)
634 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), "");
635 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), "");
638 static void prefs_filtering_set_list(void)
643 gchar * filtering_str;
644 GSList * prefs_filtering;
646 prefs_filtering = * p_processing_list;
648 for(cur = prefs_filtering ; cur != NULL ; cur = g_slist_next(cur))
649 filteringprop_free((FilteringProp *) cur->data);
650 g_slist_free(prefs_filtering);
651 prefs_filtering = NULL;
653 while (gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
654 row, 0, &filtering_str)) {
655 if (strcmp(filtering_str, _("(New)")) != 0) {
656 prop = matcher_parser_get_filtering(filtering_str);
659 g_slist_append(prefs_filtering, prop);
664 * p_processing_list = prefs_filtering;
667 static gint prefs_filtering_clist_set_row(gint row, FilteringProp * prop)
669 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
674 cond_str[0] = _("(New)");
675 return gtk_clist_append(clist, cond_str);
678 str = filteringprop_to_string(prop);
685 row = gtk_clist_append(clist, cond_str);
687 gtk_clist_set_text(clist, row, 0, cond_str[0]);
693 static void prefs_filtering_condition_define_done(MatcherList * matchers)
697 if (matchers == NULL)
700 str = matcherlist_to_string(matchers);
703 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), str);
708 static void prefs_filtering_condition_define(void)
711 MatcherList * matchers = NULL;
713 cond_str = gtk_entry_get_text(GTK_ENTRY(filtering.cond_entry));
715 if (*cond_str != '\0') {
716 matchers = matcher_parser_get_cond(cond_str);
717 if (matchers == NULL)
718 alertpanel_error(_("Condition string is not valid."));
721 prefs_matcher_open(matchers, prefs_filtering_condition_define_done);
723 if (matchers != NULL)
724 matcherlist_free(matchers);
727 static void prefs_filtering_action_define_done(GSList * action_list)
731 if (action_list == NULL)
734 str = filteringaction_list_to_string(action_list);
737 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), str);
742 static void prefs_filtering_action_define(void)
745 GSList * action_list = NULL;
747 action_str = gtk_entry_get_text(GTK_ENTRY(filtering.action_entry));
749 if (*action_str != '\0') {
750 action_list = matcher_parser_get_action_list(action_str);
751 if (action_list == NULL)
752 alertpanel_error(_("Action string is not valid."));
755 prefs_filtering_action_open(action_list,
756 prefs_filtering_action_define_done);
758 if (action_list != NULL) {
760 for(cur = action_list ; cur != NULL ; cur = cur->next) {
761 filteringaction_free(cur->data);
767 /* register / substitute delete buttons */
770 static FilteringProp * prefs_filtering_dialog_to_filtering(gboolean alert)
775 FilteringProp * prop;
776 GSList * action_list;
778 cond_str = gtk_entry_get_text(GTK_ENTRY(filtering.cond_entry));
779 if (*cond_str == '\0') {
780 if(alert == TRUE) alertpanel_error(_("Condition string is empty."));
784 action_str = gtk_entry_get_text(GTK_ENTRY(filtering.action_entry));
785 if (*action_str == '\0') {
786 if(alert == TRUE) alertpanel_error(_("Action string is empty."));
790 cond = matcher_parser_get_cond(cond_str);
793 if(alert == TRUE) alertpanel_error(_("Condition string is not valid."));
797 action_list = matcher_parser_get_action_list(action_str);
799 if (action_list == NULL) {
800 if(alert == TRUE) alertpanel_error(_("Action string is not valid."));
804 prop = filteringprop_new(cond, action_list);
809 static void prefs_filtering_register_cb(void)
811 FilteringProp * prop;
813 prop = prefs_filtering_dialog_to_filtering(TRUE);
816 prefs_filtering_clist_set_row(-1, prop);
818 filteringprop_free(prop);
820 prefs_filtering_update_hscrollbar();
823 static void prefs_filtering_substitute_cb(void)
825 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
827 FilteringProp * prop;
829 if (!clist->selection) return;
831 row = GPOINTER_TO_INT(clist->selection->data);
832 if (row == 0) return;
834 prop = prefs_filtering_dialog_to_filtering(TRUE);
837 prefs_filtering_clist_set_row(row, prop);
839 filteringprop_free(prop);
841 prefs_filtering_update_hscrollbar();
844 static void prefs_filtering_delete_cb(void)
846 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
849 if (!clist->selection) return;
850 row = GPOINTER_TO_INT(clist->selection->data);
851 if (row == 0) return;
853 if (alertpanel(_("Delete rule"),
854 _("Do you really want to delete this rule?"),
855 _("Yes"), _("No"), NULL) == G_ALERTALTERNATE)
858 gtk_clist_remove(clist, row);
860 prefs_filtering_reset_dialog();
862 prefs_filtering_update_hscrollbar();
865 static void prefs_filtering_top(void)
867 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
870 if (!clist->selection) return;
872 row = GPOINTER_TO_INT(clist->selection->data);
874 gtk_clist_row_move(clist, row, 1);
877 static void prefs_filtering_up(void)
879 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
882 if (!clist->selection) return;
884 row = GPOINTER_TO_INT(clist->selection->data);
886 gtk_clist_row_move(clist, row, row - 1);
887 if (gtk_clist_row_is_visible(clist, row - 1) != GTK_VISIBILITY_FULL)
888 gtk_clist_moveto(clist, row - 1, 0, 0, 0);
892 static void prefs_filtering_down(void)
894 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
897 if (!clist->selection) return;
899 row = GPOINTER_TO_INT(clist->selection->data);
900 if (row > 0 && row < clist->rows - 1) {
901 gtk_clist_row_move(clist, row, row + 1);
902 if (gtk_clist_row_is_visible(clist, row + 1) != GTK_VISIBILITY_FULL)
903 gtk_clist_moveto(clist, row + 1, 0, 1, 0);
907 static void prefs_filtering_bottom(void)
909 GtkCList *clist = GTK_CLIST(filtering.cond_clist);
912 if (!clist->selection) return;
914 row = GPOINTER_TO_INT(clist->selection->data);
915 if (row > 0 && row < clist->rows - 1)
916 gtk_clist_row_move(clist, row, clist->rows - 1);
919 static void prefs_filtering_select_set(FilteringProp *prop)
924 prefs_filtering_reset_dialog();
926 matcher_str = matcherlist_to_string(prop->matchers);
927 if (matcher_str == NULL) {
931 gtk_entry_set_text(GTK_ENTRY(filtering.cond_entry), matcher_str);
933 action_str = filteringaction_list_to_string(prop->action_list);
934 if (matcher_str == NULL) {
937 gtk_entry_set_text(GTK_ENTRY(filtering.action_entry), action_str);
943 static void prefs_filtering_select(GtkCList *clist, gint row, gint column,
946 FilteringProp * prop;
947 gchar * filtering_str;
950 prefs_filtering_reset_dialog();
954 if (!gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
955 row, 0, &filtering_str))
958 prop = matcher_parser_get_filtering(filtering_str);
962 prefs_filtering_select_set(prop);
964 filteringprop_free(prop);
968 static gint prefs_filtering_deleted(GtkWidget *widget, GdkEventAny *event,
971 prefs_filtering_cancel();
975 static gboolean prefs_filtering_key_pressed(GtkWidget *widget, GdkEventKey *event,
978 if (event && event->keyval == GDK_Escape)
979 prefs_filtering_cancel();
983 static void prefs_filtering_ok(void)
985 FilteringProp * prop;
987 gchar * filtering_str;
991 prop = prefs_filtering_dialog_to_filtering(FALSE);
993 str = filteringprop_to_string(prop);
995 while (gtk_clist_get_text(GTK_CLIST(filtering.cond_clist),
996 row, 0, &filtering_str)) {
997 if (strcmp(filtering_str, str) == 0) break;
1000 if (strcmp(filtering_str, str) != 0) {
1001 val = alertpanel(_("Entry not saved"),
1002 _("The entry was not saved. Close anyway?"),
1003 _("Yes"), _("No"), NULL);
1004 if (G_ALERTDEFAULT != val) {
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();