add 'score equal to' option
[claws.git] / src / prefs_matcher.c
1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2001 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 <gdk/gdkkeysyms.h>
29 #include <stdio.h>
30 #include <stdlib.h>
31 #include <string.h>
32 #include <errno.h>
33
34 #include "intl.h"
35 #include "main.h"
36 #include "prefs.h"
37 #include "prefs_matcher.h"
38 #include "prefs_common.h"
39 #include "mainwindow.h"
40 #include "foldersel.h"
41 #include "manage_window.h"
42 #include "inc.h"
43 #include "matcher.h"
44 #include "utils.h"
45 #include "gtkutils.h"
46 #include "alertpanel.h"
47 #include "folder.h"
48
49 static struct Matcher {
50         GtkWidget *window;
51
52         GtkWidget *ok_btn;
53
54         GtkWidget *predicate_combo;
55         GtkWidget *predicate_flag_combo;
56         GtkWidget *header_combo;
57
58         GtkWidget *criteria_list;
59
60         GtkWidget *predicate_list;
61         GtkWidget *predicate_label;
62         GtkWidget *predicate_flag_list;
63
64         GtkWidget *bool_op_list;
65
66         GtkWidget *header_entry;
67         GtkWidget *header_label;
68         GtkWidget *value_entry;
69         GtkWidget *value_label;
70         GtkWidget *case_chkbtn;
71         GtkWidget *regexp_chkbtn;
72
73         GtkWidget *exec_btn;
74
75         GtkWidget *cond_clist;
76 } matcher;
77
78 /* choice in the list */
79
80 enum {
81         CRITERIA_ALL = 0,
82
83         CRITERIA_SUBJECT = 1,
84         CRITERIA_FROM = 2,
85         CRITERIA_TO = 3,
86         CRITERIA_CC = 4,
87         CRITERIA_TO_OR_CC = 5,
88         CRITERIA_NEWSGROUPS = 6,
89         CRITERIA_INREPLYTO = 7,
90         CRITERIA_REFERENCES = 8,
91         CRITERIA_AGE_GREATER = 9,
92         CRITERIA_AGE_LOWER = 10,
93         CRITERIA_HEADER = 11,
94         CRITERIA_HEADERS_PART = 12,
95         CRITERIA_BODY_PART = 13,
96         CRITERIA_MESSAGE = 14,
97
98         CRITERIA_UNREAD = 15,
99         CRITERIA_NEW = 16,
100         CRITERIA_MARKED = 17,
101         CRITERIA_DELETED = 18,
102         CRITERIA_REPLIED = 19,
103         CRITERIA_FORWARDED = 20,
104
105         CRITERIA_SCORE_GREATER = 21,
106         CRITERIA_SCORE_LOWER = 22,
107         CRITERIA_SCORE_EQUAL = 23,
108
109         CRITERIA_EXECUTE = 24
110 };
111
112 enum {
113         BOOL_OP_OR = 0,
114         BOOL_OP_AND = 1
115 };
116
117 gchar * bool_op_text [] = {
118         N_("or"), N_("and")
119 };
120
121 enum {
122         PREDICATE_CONTAINS = 0,
123         PREDICATE_DOES_NOT_CONTAIN = 1
124 };
125
126 gchar * predicate_text [] = {
127         N_("contains"), N_("does not contain")
128 };
129
130 enum {
131         PREDICATE_FLAG_ENABLED = 0,
132         PREDICATE_FLAG_DISABLED = 1
133 };
134
135 gchar * predicate_flag_text [] = {
136         N_("yes"), N_("no")
137 };
138
139 gchar * criteria_text [] = {
140         N_("All messages"), N_("Subject"),
141         N_("From"), N_("To"), N_("Cc"), N_("To or Cc"),
142         N_("Newsgroups"), N_("In reply to"), N_("References"),
143         N_("Age greater than"), N_("Age lower than"),
144         N_("Header"), N_("Headers part"),
145         N_("Body part"), N_("Whole message"),
146         N_("Unread flag"), N_("New flag"),
147         N_("Marked flag"), N_("Deleted flag"),
148         N_("Replied flag"), N_("Forwarded flag"),
149         N_("Score greater than"), N_("Score lower than"),
150         N_("Score equal to"),
151         N_("Execute")
152 };
153
154 static gint get_sel_from_list(GtkList * list)
155 {
156         gint row = 0;
157         void * sel;
158         GList * child;
159
160         sel = list->selection->data;
161         for(child = list->children ; child != NULL ;
162             child = g_list_next(child)) {
163                 if (child->data == sel)
164                         return row;
165                 row ++;
166         }
167         
168         return row;
169 }
170
171 static PrefsMatcherSignal * matchers_callback;
172
173 #define VSPACING                12
174 #define VSPACING_NARROW         4
175 #define DEFAULT_ENTRY_WIDTH     80
176 #define PREFSBUFSIZE            1024
177
178 /* widget creating functions */
179 static void prefs_matcher_create        (void);
180
181 static void prefs_matcher_set_dialog    (MatcherList * matchers);
182
183 /*
184 static void prefs_matcher_set_list      (void);
185 static gint prefs_matcher_clist_set_row (gint    row);
186 */
187
188 /* callback functions */
189
190 /*
191 static void prefs_matcher_select_dest_cb        (void);
192 */
193 static void prefs_matcher_register_cb   (void);
194 static void prefs_matcher_substitute_cb (void);
195 static void prefs_matcher_delete_cb     (void);
196 static void prefs_matcher_up            (void);
197 static void prefs_matcher_down          (void);
198 static void prefs_matcher_select        (GtkCList       *clist,
199                                          gint            row,
200                                          gint            column,
201                                          GdkEvent       *event);
202
203 static void prefs_matcher_key_pressed   (GtkWidget      *widget,
204                                          GdkEventKey    *event,
205                                          gpointer        data);
206 static void prefs_matcher_ok            (void);
207 static void prefs_matcher_cancel                (void);
208 static gint prefs_matcher_deleted(GtkWidget *widget, GdkEventAny *event,
209                                   gpointer data);
210 static void prefs_matcher_criteria_select(GtkList *list,
211                                           GtkWidget *widget,
212                                           gpointer user_data);
213 static MatcherList * prefs_matcher_get_list(void);
214 static void prefs_matcher_exec_info_create(void);
215
216 void prefs_matcher_open(MatcherList * matchers, PrefsMatcherSignal * cb)
217 {
218         inc_autocheck_timer_remove();
219
220         if (!matcher.window) {
221                 prefs_matcher_create();
222         }
223
224         manage_window_set_transient(GTK_WINDOW(matcher.window));
225         gtk_widget_grab_focus(matcher.ok_btn);
226
227         matchers_callback = cb;
228
229         prefs_matcher_set_dialog(matchers);
230
231         gtk_widget_show(matcher.window);
232 }
233
234 static void prefs_matcher_create(void)
235 {
236         GtkWidget *window;
237         GtkWidget *vbox;
238         GtkWidget *ok_btn;
239         GtkWidget *cancel_btn;
240         GtkWidget *confirm_area;
241
242         GtkWidget *vbox1;
243         GtkWidget *vbox2;
244         GtkWidget *vbox3;
245         GtkWidget *table1;
246
247         GtkWidget *hbox1;
248
249         GtkWidget *header_combo;
250         GtkWidget *header_entry;
251         GtkWidget *header_label;
252         GtkWidget *criteria_combo;
253         GtkWidget *criteria_list;
254         GtkWidget *criteria_label;
255         GtkWidget *value_label;
256         GtkWidget *value_entry;
257         GtkWidget *predicate_combo;
258         GtkWidget *predicate_list;
259         GtkWidget *predicate_flag_combo;
260         GtkWidget *predicate_flag_list;
261         GtkWidget *predicate_label;
262         GtkWidget *bool_op_combo;
263         GtkWidget *bool_op_list;
264         GtkWidget *bool_op_label;
265
266         GtkWidget *regexp_chkbtn;
267         GtkWidget *case_chkbtn;
268
269         GtkWidget *reg_hbox;
270         GtkWidget *btn_hbox;
271         GtkWidget *arrow;
272         GtkWidget *reg_btn;
273         GtkWidget *subst_btn;
274         GtkWidget *del_btn;
275
276         GtkWidget *cond_hbox;
277         GtkWidget *cond_scrolledwin;
278         GtkWidget *cond_clist;
279
280         GtkWidget *btn_vbox;
281         GtkWidget *up_btn;
282         GtkWidget *down_btn;
283
284         GtkWidget *exec_btn;
285
286         GList *combo_items;
287         gint i;
288
289         gchar *title[] = {_("Registered rules")};
290
291         debug_print(_("Creating matcher setting window...\n"));
292
293         window = gtk_window_new (GTK_WINDOW_DIALOG);
294         gtk_container_set_border_width (GTK_CONTAINER (window), 8);
295         gtk_window_position (GTK_WINDOW (window), GTK_WIN_POS_CENTER);
296         gtk_window_set_modal (GTK_WINDOW (window), TRUE);
297         gtk_window_set_policy (GTK_WINDOW (window), FALSE, TRUE, FALSE);
298
299         vbox = gtk_vbox_new (FALSE, 6);
300         gtk_widget_show (vbox);
301         gtk_container_add (GTK_CONTAINER (window), vbox);
302
303         gtkut_button_set_create(&confirm_area, &ok_btn, _("OK"),
304                                 &cancel_btn, _("Cancel"), NULL, NULL);
305         gtk_widget_show (confirm_area);
306         gtk_box_pack_end (GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
307         gtk_widget_grab_default (ok_btn);
308
309         gtk_window_set_title (GTK_WINDOW(window),
310                               _("Condition setting"));
311         gtk_signal_connect (GTK_OBJECT(window), "delete_event",
312                             GTK_SIGNAL_FUNC(prefs_matcher_deleted), NULL);
313         gtk_signal_connect (GTK_OBJECT(window), "key_press_event",
314                             GTK_SIGNAL_FUNC(prefs_matcher_key_pressed), NULL);
315         gtk_signal_connect (GTK_OBJECT(window), "focus_in_event",
316                             GTK_SIGNAL_FUNC(manage_window_focus_in), NULL);
317         gtk_signal_connect (GTK_OBJECT(window), "focus_out_event",
318                             GTK_SIGNAL_FUNC(manage_window_focus_out), NULL);
319         gtk_signal_connect (GTK_OBJECT(ok_btn), "clicked",
320                             GTK_SIGNAL_FUNC(prefs_matcher_ok), NULL);
321         gtk_signal_connect (GTK_OBJECT(cancel_btn), "clicked",
322                             GTK_SIGNAL_FUNC(prefs_matcher_cancel), NULL);
323
324         vbox1 = gtk_vbox_new (FALSE, VSPACING);
325         gtk_widget_show (vbox1);
326         gtk_box_pack_start (GTK_BOX (vbox), vbox1, TRUE, TRUE, 0);
327         gtk_container_set_border_width (GTK_CONTAINER (vbox1), 2);
328
329         table1 = gtk_table_new (2, 4, FALSE);
330         gtk_widget_show (table1);
331
332         gtk_box_pack_start (GTK_BOX (vbox1), table1, FALSE, FALSE, 0);
333         gtk_table_set_row_spacings (GTK_TABLE (table1), 8);
334         gtk_table_set_col_spacings (GTK_TABLE (table1), 8);
335
336         /* criteria combo box */
337
338         criteria_label = gtk_label_new (_("Match type"));
339         gtk_widget_show (criteria_label);
340         gtk_misc_set_alignment (GTK_MISC (criteria_label), 0, 0.5);
341         gtk_table_attach (GTK_TABLE (table1), criteria_label, 0, 1, 0, 1,
342                           GTK_FILL, 0, 0, 0);
343
344         criteria_combo = gtk_combo_new ();
345         gtk_widget_show (criteria_combo);
346
347         combo_items = NULL;
348
349         for(i = 0 ; i < (gint) (sizeof(criteria_text) / sizeof(gchar *)) ;
350             i++) {
351                 combo_items = g_list_append(combo_items,
352                                             (gpointer) _(criteria_text[i]));
353         }
354         gtk_combo_set_popdown_strings(GTK_COMBO(criteria_combo), combo_items);
355
356         g_list_free(combo_items);
357
358         gtk_widget_set_usize (criteria_combo, 120, -1);
359         gtk_table_attach (GTK_TABLE (table1), criteria_combo, 0, 1, 1, 2,
360                           0, 0, 0, 0);
361         criteria_list = GTK_COMBO(criteria_combo)->list;
362         gtk_signal_connect (GTK_OBJECT (criteria_list), "select-child",
363                             GTK_SIGNAL_FUNC (prefs_matcher_criteria_select),
364                             NULL);
365
366         gtk_entry_set_editable(GTK_ENTRY(GTK_COMBO(criteria_combo)->entry),
367                                FALSE);
368
369         /* header name */
370
371         header_label = gtk_label_new (_("Header name"));
372         gtk_widget_show (header_label);
373         gtk_misc_set_alignment (GTK_MISC (header_label), 0, 0.5);
374         gtk_table_attach (GTK_TABLE (table1), header_label, 1, 2, 0, 1,
375                           GTK_FILL, 0, 0, 0);
376
377         header_combo = gtk_combo_new ();
378         gtk_widget_show (header_combo);
379         gtk_widget_set_usize (header_combo, 96, -1);
380         gtkut_combo_set_items (GTK_COMBO (header_combo),
381                                "Subject", "From", "To", "Cc", "Reply-To",
382                                "Sender", "X-ML-Name", "X-List", "X-Sequence",
383                                "X-Mailer",
384                                NULL);
385         gtk_table_attach (GTK_TABLE (table1), header_combo, 1, 2, 1, 2,
386                           0, 0, 0, 0);
387         header_entry = GTK_COMBO (header_combo)->entry;
388         gtk_entry_set_editable (GTK_ENTRY (header_entry), TRUE);
389
390         /* value */
391
392         value_label = gtk_label_new (_("Value"));
393         gtk_widget_show (value_label);
394         gtk_misc_set_alignment (GTK_MISC (value_label), 0, 0.5);
395         gtk_table_attach (GTK_TABLE (table1), value_label, 2, 3, 0, 1,
396                           GTK_FILL | GTK_SHRINK | GTK_EXPAND, 0, 0, 0);
397
398         value_entry = gtk_entry_new ();
399         gtk_widget_show (value_entry);
400         gtk_widget_set_usize (value_entry, 200, -1);
401         gtk_table_attach (GTK_TABLE (table1), value_entry, 2, 3, 1, 2,
402                           GTK_FILL | GTK_SHRINK | GTK_EXPAND, 0, 0, 0);
403
404         exec_btn = gtk_button_new_with_label (_("Info ..."));
405         gtk_widget_show (exec_btn);
406         gtk_table_attach (GTK_TABLE (table1), exec_btn, 3, 4, 1, 2,
407                           GTK_FILL | GTK_SHRINK | GTK_EXPAND, 0, 0, 0);
408         gtk_signal_connect (GTK_OBJECT (exec_btn), "clicked",
409                             GTK_SIGNAL_FUNC (prefs_matcher_exec_info),
410                             NULL);
411
412         /* predicate */
413
414         vbox2 = gtk_vbox_new (FALSE, VSPACING);
415         gtk_widget_show (vbox2);
416         gtk_box_pack_start (GTK_BOX (vbox1), vbox2, FALSE, FALSE, 0);
417
418         hbox1 = gtk_hbox_new (FALSE, 8);
419         gtk_widget_show (hbox1);
420         gtk_box_pack_start (GTK_BOX (vbox2), hbox1, FALSE, FALSE, 0);
421
422         predicate_label = gtk_label_new (_("Predicate"));
423         gtk_widget_show (predicate_label);
424         gtk_box_pack_start (GTK_BOX (hbox1), predicate_label,
425                             FALSE, FALSE, 0);
426
427         predicate_combo = gtk_combo_new ();
428         gtk_widget_show (predicate_combo);
429         gtk_widget_set_usize (predicate_combo, 120, -1);
430         predicate_list = GTK_COMBO(predicate_combo)->list;
431         gtk_entry_set_editable(GTK_ENTRY(GTK_COMBO(predicate_combo)->entry),
432                                FALSE);
433
434         combo_items = NULL;
435
436         for(i = 0 ; i < (gint) (sizeof(predicate_text) / sizeof(gchar *)) ;
437             i++) {
438                 combo_items = g_list_append(combo_items,
439                                             (gpointer) _(predicate_text[i]));
440         }
441         gtk_combo_set_popdown_strings(GTK_COMBO(predicate_combo), combo_items);
442
443         g_list_free(combo_items);
444
445         gtk_box_pack_start (GTK_BOX (hbox1), predicate_combo,
446                             FALSE, FALSE, 0);
447
448         /* predicate flag */
449
450         predicate_flag_combo = gtk_combo_new ();
451         gtk_widget_hide (predicate_flag_combo);
452         gtk_widget_set_usize (predicate_flag_combo, 120, -1);
453         predicate_flag_list = GTK_COMBO(predicate_flag_combo)->list;
454         gtk_entry_set_editable(GTK_ENTRY(GTK_COMBO(predicate_flag_combo)->entry), FALSE);
455
456         combo_items = NULL;
457
458         for(i = 0 ; i < (gint) (sizeof(predicate_text) / sizeof(gchar *)) ;
459             i++) {
460                 combo_items = g_list_append(combo_items, (gpointer) _(predicate_flag_text[i]));
461         }
462         gtk_combo_set_popdown_strings(GTK_COMBO(predicate_flag_combo),
463                                       combo_items);
464
465         g_list_free(combo_items);
466
467         gtk_box_pack_start (GTK_BOX (hbox1), predicate_flag_combo,
468                             FALSE, FALSE, 0);
469
470         vbox3 = gtk_vbox_new (FALSE, 0);
471         gtk_widget_show (vbox3);
472         gtk_box_pack_start (GTK_BOX (hbox1), vbox3, FALSE, FALSE, 0);
473
474         PACK_CHECK_BUTTON (vbox3, case_chkbtn, _("Case sensitive"));
475         PACK_CHECK_BUTTON (vbox3, regexp_chkbtn, _("Use regexp"));
476
477         /* register / substitute / delete */
478
479         reg_hbox = gtk_hbox_new (FALSE, 4);
480         gtk_widget_show (reg_hbox);
481         gtk_box_pack_start (GTK_BOX (vbox1), reg_hbox, FALSE, FALSE, 0);
482
483         arrow = gtk_arrow_new (GTK_ARROW_DOWN, GTK_SHADOW_OUT);
484         gtk_widget_show (arrow);
485         gtk_box_pack_start (GTK_BOX (reg_hbox), arrow, FALSE, FALSE, 0);
486         gtk_widget_set_usize (arrow, -1, 16);
487
488         btn_hbox = gtk_hbox_new (TRUE, 4);
489         gtk_widget_show (btn_hbox);
490         gtk_box_pack_start (GTK_BOX (reg_hbox), btn_hbox, FALSE, FALSE, 0);
491
492         reg_btn = gtk_button_new_with_label (_("Register"));
493         gtk_widget_show (reg_btn);
494         gtk_box_pack_start (GTK_BOX (btn_hbox), reg_btn, FALSE, TRUE, 0);
495         gtk_signal_connect (GTK_OBJECT (reg_btn), "clicked",
496                             GTK_SIGNAL_FUNC (prefs_matcher_register_cb), NULL);
497
498         subst_btn = gtk_button_new_with_label (_(" Substitute "));
499         gtk_widget_show (subst_btn);
500         gtk_box_pack_start (GTK_BOX (btn_hbox), subst_btn, FALSE, TRUE, 0);
501         gtk_signal_connect (GTK_OBJECT (subst_btn), "clicked",
502                             GTK_SIGNAL_FUNC (prefs_matcher_substitute_cb),
503                             NULL);
504
505         del_btn = gtk_button_new_with_label (_("Delete"));
506         gtk_widget_show (del_btn);
507         gtk_box_pack_start (GTK_BOX (btn_hbox), del_btn, FALSE, TRUE, 0);
508         gtk_signal_connect (GTK_OBJECT (del_btn), "clicked",
509                             GTK_SIGNAL_FUNC (prefs_matcher_delete_cb), NULL);
510
511         /* boolean operation */
512
513         bool_op_label = gtk_label_new (_("Boolean Op"));
514         gtk_misc_set_alignment (GTK_MISC (value_label), 0, 0.5);
515         gtk_widget_show (bool_op_label);
516         gtk_box_pack_start (GTK_BOX (btn_hbox), bool_op_label,
517                             FALSE, FALSE, 0);
518
519         bool_op_combo = gtk_combo_new ();
520         gtk_widget_show (bool_op_combo);
521         gtk_widget_set_usize (bool_op_combo, 50, -1);
522         bool_op_list = GTK_COMBO(bool_op_combo)->list;
523         gtk_entry_set_editable(GTK_ENTRY(GTK_COMBO(bool_op_combo)->entry),
524                                FALSE);
525
526         combo_items = NULL;
527
528         for(i = 0 ; i < (gint) (sizeof(bool_op_text) / sizeof(gchar *)) ;
529             i++) {
530                 combo_items = g_list_append(combo_items,
531                                             (gpointer) _(bool_op_text[i]));
532         }
533         gtk_combo_set_popdown_strings(GTK_COMBO(bool_op_combo), combo_items);
534
535         g_list_free(combo_items);
536
537         gtk_box_pack_start (GTK_BOX (btn_hbox), bool_op_combo,
538                             FALSE, FALSE, 0);
539
540         cond_hbox = gtk_hbox_new (FALSE, 8);
541         gtk_widget_show (cond_hbox);
542         gtk_box_pack_start (GTK_BOX (vbox1), cond_hbox, TRUE, TRUE, 0);
543
544         cond_scrolledwin = gtk_scrolled_window_new (NULL, NULL);
545         gtk_widget_show (cond_scrolledwin);
546         gtk_widget_set_usize (cond_scrolledwin, -1, 150);
547         gtk_box_pack_start (GTK_BOX (cond_hbox), cond_scrolledwin,
548                             TRUE, TRUE, 0);
549         gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (cond_scrolledwin),
550                                         GTK_POLICY_AUTOMATIC,
551                                         GTK_POLICY_AUTOMATIC);
552
553         cond_clist = gtk_clist_new_with_titles(1, title);
554         gtk_widget_show (cond_clist);
555         gtk_container_add (GTK_CONTAINER (cond_scrolledwin), cond_clist);
556         gtk_clist_set_column_width (GTK_CLIST (cond_clist), 0, 80);
557         gtk_clist_set_selection_mode (GTK_CLIST (cond_clist),
558                                       GTK_SELECTION_BROWSE);
559         GTK_WIDGET_UNSET_FLAGS (GTK_CLIST (cond_clist)->column[0].button,
560                                 GTK_CAN_FOCUS);
561         gtk_signal_connect (GTK_OBJECT (cond_clist), "select_row",
562                             GTK_SIGNAL_FUNC (prefs_matcher_select), NULL);
563
564         btn_vbox = gtk_vbox_new (FALSE, 8);
565         gtk_widget_show (btn_vbox);
566         gtk_box_pack_start (GTK_BOX (cond_hbox), btn_vbox, FALSE, FALSE, 0);
567
568         up_btn = gtk_button_new_with_label (_("Up"));
569         gtk_widget_show (up_btn);
570         gtk_box_pack_start (GTK_BOX (btn_vbox), up_btn, FALSE, FALSE, 0);
571         gtk_signal_connect (GTK_OBJECT (up_btn), "clicked",
572                             GTK_SIGNAL_FUNC (prefs_matcher_up), NULL);
573
574         down_btn = gtk_button_new_with_label (_("Down"));
575         gtk_widget_show (down_btn);
576         gtk_box_pack_start (GTK_BOX (btn_vbox), down_btn, FALSE, FALSE, 0);
577         gtk_signal_connect (GTK_OBJECT (down_btn), "clicked",
578                             GTK_SIGNAL_FUNC (prefs_matcher_down), NULL);
579
580         gtk_widget_show_all(window);
581
582         matcher.window    = window;
583
584         matcher.ok_btn = ok_btn;
585
586         matcher.criteria_list = criteria_list;
587         matcher.header_combo = header_combo;
588         matcher.header_entry = header_entry;
589         matcher.header_label = header_label;
590         matcher.value_entry = value_entry;
591         matcher.value_label = value_label;
592         matcher.predicate_label = predicate_label;
593         matcher.predicate_list = predicate_list;
594         matcher.predicate_combo = predicate_combo;
595         matcher.predicate_flag_list = predicate_flag_list;
596         matcher.predicate_flag_combo = predicate_flag_combo;
597         matcher.case_chkbtn = case_chkbtn;
598         matcher.regexp_chkbtn = regexp_chkbtn;
599         matcher.bool_op_list = bool_op_list;
600         matcher.exec_btn = exec_btn;
601
602         matcher.cond_clist   = cond_clist;
603 }
604
605 static gint prefs_matcher_clist_set_row(gint row, MatcherProp * prop)
606 {
607         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
608         gchar * cond_str[1];
609         gchar * matcher_str;
610
611         if (prop == NULL) {
612                 cond_str[0] = _("(New)");
613                 return gtk_clist_append(clist, cond_str);
614         }
615
616         matcher_str = matcherprop_to_string(prop);
617         cond_str[0] = matcher_str;
618         if (row < 0)
619                 row = gtk_clist_append(clist, cond_str);
620         else
621                 gtk_clist_set_text(clist, row, 0, cond_str[0]);
622         g_free(matcher_str);
623
624         return row;
625 }
626
627 static void prefs_matcher_reset_condition(void)
628 {
629         gtk_list_select_item(GTK_LIST(matcher.criteria_list), 0);
630         gtk_list_select_item(GTK_LIST(matcher.predicate_list), 0);
631         gtk_entry_set_text(GTK_ENTRY(matcher.header_entry), "");
632         gtk_entry_set_text(GTK_ENTRY(matcher.value_entry), "");
633 }
634
635 static void prefs_matcher_update_hscrollbar(void)
636 {
637         gint optwidth = gtk_clist_optimal_column_width(GTK_CLIST(matcher.cond_clist), 0);
638         gtk_clist_set_column_width(GTK_CLIST(matcher.cond_clist), 0, optwidth);
639 }
640
641 static void prefs_matcher_set_dialog(MatcherList * matchers)
642 {
643         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
644         GSList * cur;
645         gboolean bool_op = 1;
646
647         gtk_clist_freeze(clist);
648         gtk_clist_clear(clist);
649
650         prefs_matcher_clist_set_row(-1, NULL);
651         if (matchers != NULL) {
652                 for (cur = matchers->matchers ; cur != NULL ;
653                      cur = g_slist_next(cur)) {
654                         MatcherProp * prop;
655                         prop = (MatcherProp *) cur->data;
656                         prefs_matcher_clist_set_row(-1, prop);
657                 }
658
659                 bool_op = matchers->bool_and;
660         }
661         
662         prefs_matcher_update_hscrollbar();
663
664         gtk_clist_thaw(clist);
665
666         gtk_list_select_item(GTK_LIST(matcher.bool_op_list), bool_op);
667
668         prefs_matcher_reset_condition();
669 }
670
671 static MatcherList * prefs_matcher_get_list(void)
672 {
673         gchar * matcher_str;
674         MatcherProp * prop;
675         gint row = 1;
676         gchar * tmp;
677         gboolean bool_and;
678         GSList * matcher_list;
679         MatcherList * matchers;
680
681         matcher_list = NULL;
682
683         while (gtk_clist_get_text(GTK_CLIST(matcher.cond_clist),
684                                   row, 0, &matcher_str)) {
685
686                 if (strcmp(matcher_str, _("(New)")) != 0) {
687                         tmp = matcher_str;
688                         prop = matcherprop_parse(&tmp);
689                         
690                         if (tmp == NULL)
691                                 break;
692                         
693                         matcher_list = g_slist_append(matcher_list, prop);
694                 }
695                 row ++;
696         }
697
698         bool_and = get_sel_from_list(GTK_LIST(matcher.bool_op_list));
699
700         matchers = matcherlist_new(matcher_list, bool_and);
701
702         return matchers;
703 }
704
705 static gint prefs_matcher_get_criteria_from_matching(gint matching_id)
706 {
707         switch(matching_id) {
708         case MATCHING_ALL:
709                 return CRITERIA_ALL;
710         case MATCHING_NOT_UNREAD:
711         case MATCHING_UNREAD:
712                 return CRITERIA_UNREAD;
713         case MATCHING_NOT_NEW:
714         case MATCHING_NEW:
715                 return CRITERIA_NEW;
716         case MATCHING_NOT_MARKED:
717         case MATCHING_MARKED:
718                 return CRITERIA_MARKED;
719         case MATCHING_NOT_DELETED:
720         case MATCHING_DELETED:
721                 return CRITERIA_DELETED;
722                 break;
723         case MATCHING_NOT_REPLIED:
724         case MATCHING_REPLIED:
725                 return CRITERIA_REPLIED;
726         case MATCHING_NOT_FORWARDED:
727         case MATCHING_FORWARDED:
728                 return CRITERIA_FORWARDED;
729         case MATCHING_NOT_SUBJECT:
730         case MATCHING_SUBJECT:
731                 return CRITERIA_SUBJECT;
732         case MATCHING_NOT_FROM:
733         case MATCHING_FROM:
734                 return CRITERIA_FROM;
735         case MATCHING_NOT_TO:
736         case MATCHING_TO:
737                 return CRITERIA_TO;
738         case MATCHING_NOT_CC:
739         case MATCHING_CC:
740                 return CRITERIA_CC;
741         case MATCHING_NOT_NEWSGROUPS:
742         case MATCHING_NEWSGROUPS:
743                 return CRITERIA_NEWSGROUPS;
744         case MATCHING_NOT_INREPLYTO:
745         case MATCHING_INREPLYTO:
746                 return CRITERIA_INREPLYTO;
747         case MATCHING_NOT_REFERENCES:
748         case MATCHING_REFERENCES:
749                 return CRITERIA_REFERENCES;
750         case MATCHING_NOT_TO_AND_NOT_CC:
751         case MATCHING_TO_OR_CC:
752                 return CRITERIA_TO_OR_CC;
753         case MATCHING_NOT_BODY_PART:
754         case MATCHING_BODY_PART:
755                 return CRITERIA_BODY_PART;
756         case MATCHING_NOT_MESSAGE:
757         case MATCHING_MESSAGE:
758                 return CRITERIA_MESSAGE;
759                 break;
760         case MATCHING_NOT_HEADERS_PART:
761         case MATCHING_HEADERS_PART:
762                 return CRITERIA_HEADERS_PART;
763         case MATCHING_NOT_HEADER:
764         case MATCHING_HEADER:
765                 return CRITERIA_HEADER;
766         case MATCHING_AGE_GREATER:
767                 return CRITERIA_AGE_GREATER;
768         case MATCHING_AGE_LOWER:
769                 return CRITERIA_AGE_LOWER;
770         case MATCHING_SCORE_GREATER:
771                 return CRITERIA_SCORE_GREATER;
772         case MATCHING_SCORE_LOWER:
773                 return CRITERIA_SCORE_LOWER;
774         case MATCHING_SCORE_EQUAL:
775                 return CRITERIA_SCORE_EQUAL;
776         case MATCHING_NOT_EXECUTE:
777         case MATCHING_EXECUTE:
778                 return CRITERIA_EXECUTE;
779                 break;
780         default:
781                 return -1;
782         }
783 }
784
785 static gint prefs_matcher_get_matching_from_criteria(gint criteria_id)
786 {
787         switch (criteria_id) {
788         case CRITERIA_ALL:
789                 return MATCHING_ALL;
790         case CRITERIA_UNREAD:
791                 return MATCHING_UNREAD;
792         case CRITERIA_NEW:
793                 return MATCHING_NEW;
794         case CRITERIA_MARKED:
795                 return MATCHING_MARKED;
796         case CRITERIA_DELETED:
797                 return MATCHING_DELETED;
798         case CRITERIA_REPLIED:
799                 return MATCHING_REPLIED;
800         case CRITERIA_FORWARDED:
801                 return MATCHING_FORWARDED;
802         case CRITERIA_SUBJECT:
803                 return MATCHING_SUBJECT;
804         case CRITERIA_FROM:
805                 return MATCHING_FROM;
806         case CRITERIA_TO:
807                 return MATCHING_TO;
808         case CRITERIA_CC:
809                 return MATCHING_CC;
810         case CRITERIA_TO_OR_CC:
811                 return MATCHING_TO_OR_CC;
812         case CRITERIA_NEWSGROUPS:
813                 return MATCHING_NEWSGROUPS;
814         case CRITERIA_INREPLYTO:
815                 return MATCHING_INREPLYTO;
816         case CRITERIA_REFERENCES:
817                 return MATCHING_REFERENCES;
818         case CRITERIA_AGE_GREATER:
819                 return MATCHING_AGE_GREATER;
820         case CRITERIA_AGE_LOWER:
821                 return MATCHING_AGE_LOWER;
822         case CRITERIA_SCORE_GREATER:
823                 return MATCHING_SCORE_GREATER;
824         case CRITERIA_SCORE_LOWER:
825                 return MATCHING_SCORE_LOWER;
826         case CRITERIA_SCORE_EQUAL:
827                 return MATCHING_SCORE_EQUAL;
828         case CRITERIA_HEADER:
829                 return MATCHING_HEADER;
830         case CRITERIA_HEADERS_PART:
831                 return MATCHING_HEADERS_PART;
832         case CRITERIA_BODY_PART:
833                 return MATCHING_BODY_PART;
834         case CRITERIA_MESSAGE:
835                 return MATCHING_MESSAGE;
836         case CRITERIA_EXECUTE:
837                 return MATCHING_EXECUTE;
838         default:
839                 return -1;
840         }
841 }
842
843 static gint prefs_matcher_not_criteria(gint matcher_criteria)
844 {
845         switch(matcher_criteria) {
846         case MATCHING_UNREAD:
847                 return MATCHING_NOT_UNREAD;
848         case MATCHING_NEW:
849                 return MATCHING_NOT_NEW;
850         case MATCHING_MARKED:
851                 return MATCHING_NOT_MARKED;
852         case MATCHING_DELETED:
853                 return MATCHING_NOT_DELETED;
854         case MATCHING_REPLIED:
855                 return MATCHING_NOT_REPLIED;
856         case MATCHING_FORWARDED:
857                 return MATCHING_NOT_FORWARDED;
858         case MATCHING_SUBJECT:
859                 return MATCHING_NOT_SUBJECT;
860         case MATCHING_FROM:
861                 return MATCHING_NOT_FROM;
862         case MATCHING_TO:
863                 return MATCHING_NOT_TO;
864         case MATCHING_CC:
865                 return MATCHING_NOT_CC;
866         case MATCHING_TO_OR_CC:
867                 return MATCHING_NOT_TO_AND_NOT_CC;
868         case MATCHING_NEWSGROUPS:
869                 return MATCHING_NOT_NEWSGROUPS;
870         case MATCHING_INREPLYTO:
871                 return MATCHING_NOT_INREPLYTO;
872         case MATCHING_REFERENCES:
873                 return MATCHING_NOT_REFERENCES;
874         case MATCHING_HEADER:
875                 return MATCHING_NOT_HEADER;
876         case MATCHING_HEADERS_PART:
877                 return MATCHING_NOT_HEADERS_PART;
878         case MATCHING_MESSAGE:
879                 return MATCHING_NOT_MESSAGE;
880         case MATCHING_EXECUTE:
881                 return MATCHING_NOT_EXECUTE;
882         case MATCHING_BODY_PART:
883                 return MATCHING_NOT_BODY_PART;
884         default:
885                 return matcher_criteria;
886         }
887 }
888
889 static MatcherProp * prefs_matcher_dialog_to_matcher()
890 {
891         MatcherProp * matcherprop;
892         gint criteria;
893         gint matchtype;
894         gint value_pred;
895         gint value_pred_flag;
896         gint value_criteria;
897         gboolean use_regexp;
898         gboolean case_sensitive;
899         gchar * header;
900         gchar * expr;
901         gint value;
902         gchar * value_str;
903
904         value_criteria = get_sel_from_list(GTK_LIST(matcher.criteria_list));
905
906         criteria = prefs_matcher_get_matching_from_criteria(value_criteria);
907
908         value_pred = get_sel_from_list(GTK_LIST(matcher.predicate_list));
909         value_pred_flag = get_sel_from_list(GTK_LIST(matcher.predicate_flag_list));
910
911         use_regexp = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(matcher.regexp_chkbtn));
912         case_sensitive = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(matcher.case_chkbtn));
913
914         switch (value_criteria) {
915         case CRITERIA_UNREAD:
916         case CRITERIA_NEW:
917         case CRITERIA_MARKED:
918         case CRITERIA_DELETED:
919         case CRITERIA_REPLIED:
920         case CRITERIA_FORWARDED:
921                 if (value_pred_flag == PREDICATE_FLAG_DISABLED)
922                         criteria = prefs_matcher_not_criteria(criteria);
923                 break;
924         case CRITERIA_SUBJECT:
925         case CRITERIA_FROM:
926         case CRITERIA_TO:
927         case CRITERIA_CC:
928         case CRITERIA_TO_OR_CC:
929         case CRITERIA_NEWSGROUPS:
930         case CRITERIA_INREPLYTO:
931         case CRITERIA_REFERENCES:
932         case CRITERIA_HEADERS_PART:
933         case CRITERIA_BODY_PART:
934         case CRITERIA_MESSAGE:
935         case CRITERIA_EXECUTE:
936         case CRITERIA_AGE_GREATER:
937         case CRITERIA_AGE_LOWER:
938         case CRITERIA_HEADER:
939                 if (value_pred == PREDICATE_DOES_NOT_CONTAIN)
940                         criteria = prefs_matcher_not_criteria(criteria);
941                 break;
942         }
943
944         if (use_regexp) {
945                 if (case_sensitive)
946                         matchtype = MATCHING_REGEXP;
947                 else
948                         matchtype = MATCHING_REGEXPCASE;
949         }
950         else {
951                 if (case_sensitive)
952                         matchtype = MATCHING_MATCH;
953                 else
954                         matchtype = MATCHING_MATCHCASE;
955         }
956
957         header = NULL;
958         expr = NULL;
959         value = 0;
960
961         switch (value_criteria) {
962         case CRITERIA_ALL:
963         case CRITERIA_UNREAD:
964         case CRITERIA_NEW:
965         case CRITERIA_MARKED:
966         case CRITERIA_DELETED:
967         case CRITERIA_REPLIED:
968         case CRITERIA_FORWARDED:
969                 break;
970
971         case CRITERIA_SUBJECT:
972         case CRITERIA_FROM:
973         case CRITERIA_TO:
974         case CRITERIA_CC:
975         case CRITERIA_TO_OR_CC:
976         case CRITERIA_NEWSGROUPS:
977         case CRITERIA_INREPLYTO:
978         case CRITERIA_REFERENCES:
979         case CRITERIA_HEADERS_PART:
980         case CRITERIA_BODY_PART:
981         case CRITERIA_MESSAGE:
982         case CRITERIA_EXECUTE:
983                 expr = gtk_entry_get_text(GTK_ENTRY(matcher.value_entry));
984
985                 /*
986                 if (*expr == '\0') {
987                     alertpanel_error(_("Match string is not set."));
988                     return NULL;
989                 }
990                 */
991                 break;
992
993         case CRITERIA_AGE_GREATER:
994         case CRITERIA_AGE_LOWER:
995         case CRITERIA_SCORE_GREATER:
996         case CRITERIA_SCORE_LOWER:
997         case CRITERIA_SCORE_EQUAL:
998                 value_str = gtk_entry_get_text(GTK_ENTRY(matcher.value_entry));
999
1000                 if (*value_str == '\0') {
1001                     alertpanel_error(_("Value is not set."));
1002                     return NULL;
1003                 }
1004
1005                 value = atoi(value_str);
1006
1007                 break;
1008
1009         case CRITERIA_HEADER:
1010
1011                 header = gtk_entry_get_text(GTK_ENTRY(matcher.header_entry));
1012                 expr = gtk_entry_get_text(GTK_ENTRY(matcher.value_entry));
1013
1014                 if (*header == '\0') {
1015                     alertpanel_error(_("Header name is not set."));
1016                     return NULL;
1017                 }
1018                 /*
1019                 if (*expr == '\0') {
1020                     alertpanel_error(_("Match string is not set."));
1021                     return NULL;
1022                 }
1023                 */
1024                 break;
1025         }
1026
1027         matcherprop =  matcherprop_new(criteria, header, matchtype,
1028                                        expr, value);
1029
1030         return matcherprop;
1031 }
1032
1033 static void prefs_matcher_register_cb(void)
1034 {
1035         MatcherProp * matcherprop;
1036         
1037         matcherprop = prefs_matcher_dialog_to_matcher();
1038         if (matcherprop == NULL)
1039                 return;
1040
1041         prefs_matcher_clist_set_row(-1, matcherprop);
1042
1043         matcherprop_free(matcherprop);
1044
1045         prefs_matcher_reset_condition();
1046         prefs_matcher_update_hscrollbar();
1047 }
1048
1049 static void prefs_matcher_substitute_cb(void)
1050 {
1051         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
1052         gint row;
1053         MatcherProp * matcherprop;
1054
1055         row = GPOINTER_TO_INT(clist->selection->data);
1056         if (row == 0)
1057                 return;
1058         
1059         matcherprop = prefs_matcher_dialog_to_matcher();
1060         if (matcherprop == NULL)
1061                 return;
1062
1063         prefs_matcher_clist_set_row(row, matcherprop);
1064
1065         matcherprop_free(matcherprop);
1066
1067         prefs_matcher_reset_condition();
1068         
1069         prefs_matcher_update_hscrollbar();
1070 }
1071
1072 static void prefs_matcher_delete_cb(void)
1073 {
1074         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
1075         gint row;
1076
1077         if (!clist->selection) return;
1078         row = GPOINTER_TO_INT(clist->selection->data);
1079         if (row == 0)
1080                 return;
1081
1082         gtk_clist_remove(clist, row);
1083         
1084         prefs_matcher_update_hscrollbar();
1085 }
1086
1087 static void prefs_matcher_up(void)
1088 {
1089         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
1090         gint row;
1091
1092         if (!clist->selection) return;
1093
1094         row = GPOINTER_TO_INT(clist->selection->data);
1095         if (row > 1) {
1096                 gtk_clist_row_move(clist, row, row - 1);
1097                 if(gtk_clist_row_is_visible(clist, row - 1) != GTK_VISIBILITY_FULL) {
1098                         gtk_clist_moveto(clist, row - 1, 0, 0, 0);
1099                 } 
1100         }
1101 }
1102
1103 static void prefs_matcher_down(void)
1104 {
1105         GtkCList *clist = GTK_CLIST(matcher.cond_clist);
1106         gint row;
1107
1108         if (!clist->selection) return;
1109
1110         row = GPOINTER_TO_INT(clist->selection->data);
1111         if (row >= 1 && row < clist->rows - 1) {
1112                 gtk_clist_row_move(clist, row, row + 1);
1113                 if(gtk_clist_row_is_visible(clist, row + 1) != GTK_VISIBILITY_FULL) {
1114                         gtk_clist_moveto(clist, row + 1, 0, 1, 0);
1115                 } 
1116         }
1117 }
1118
1119 static void prefs_matcher_select(GtkCList *clist, gint row, gint column,
1120                                  GdkEvent *event)
1121 {
1122         gchar * matcher_str;
1123         gchar * tmp;
1124         MatcherProp * prop;
1125         gboolean negative_cond;
1126         gint criteria;
1127
1128         if (!gtk_clist_get_text(GTK_CLIST(matcher.cond_clist),
1129                                 row, 0, &matcher_str))
1130                 return;
1131
1132         negative_cond = FALSE;
1133
1134         if (row == 0) {
1135                 prefs_matcher_reset_condition();
1136                 return;
1137         }
1138
1139         tmp = matcher_str;
1140         prop = matcherprop_parse(&tmp);
1141         if (tmp == NULL)
1142                 return;
1143
1144         criteria = prefs_matcher_get_criteria_from_matching(prop->criteria);
1145         if (criteria != -1)
1146                 gtk_list_select_item(GTK_LIST(matcher.criteria_list),
1147                                      criteria);
1148
1149         switch(prop->criteria) {
1150         case MATCHING_NOT_UNREAD:
1151         case MATCHING_NOT_NEW:
1152         case MATCHING_NOT_MARKED:
1153         case MATCHING_NOT_DELETED:
1154         case MATCHING_NOT_REPLIED:
1155         case MATCHING_NOT_FORWARDED:
1156         case MATCHING_NOT_SUBJECT:
1157         case MATCHING_NOT_FROM:
1158         case MATCHING_NOT_TO:
1159         case MATCHING_NOT_CC:
1160         case MATCHING_NOT_NEWSGROUPS:
1161         case MATCHING_NOT_INREPLYTO:
1162         case MATCHING_NOT_REFERENCES:
1163         case MATCHING_NOT_TO_AND_NOT_CC:
1164         case MATCHING_NOT_BODY_PART:
1165         case MATCHING_NOT_MESSAGE:
1166         case MATCHING_NOT_HEADERS_PART:
1167         case MATCHING_NOT_HEADER:
1168                 negative_cond = TRUE;
1169                 break;
1170         }
1171         
1172         switch(prop->criteria) {
1173         case MATCHING_ALL:
1174                 break;
1175
1176         case MATCHING_NOT_SUBJECT:
1177         case MATCHING_NOT_FROM:
1178         case MATCHING_NOT_TO:
1179         case MATCHING_NOT_CC:
1180         case MATCHING_NOT_TO_AND_NOT_CC:
1181         case MATCHING_NOT_NEWSGROUPS:
1182         case MATCHING_NOT_INREPLYTO:
1183         case MATCHING_NOT_REFERENCES:
1184         case MATCHING_NOT_HEADERS_PART:
1185         case MATCHING_NOT_BODY_PART:
1186         case MATCHING_NOT_MESSAGE:
1187         case MATCHING_SUBJECT:
1188         case MATCHING_FROM:
1189         case MATCHING_TO:
1190         case MATCHING_CC:
1191         case MATCHING_TO_OR_CC:
1192         case MATCHING_NEWSGROUPS:
1193         case MATCHING_INREPLYTO:
1194         case MATCHING_REFERENCES:
1195         case MATCHING_HEADERS_PART:
1196         case MATCHING_BODY_PART:
1197         case MATCHING_MESSAGE:
1198                 gtk_entry_set_text(GTK_ENTRY(matcher.value_entry), prop->expr);
1199                 break;
1200
1201         case MATCHING_AGE_GREATER:
1202         case MATCHING_AGE_LOWER:
1203         case MATCHING_SCORE_GREATER:
1204         case MATCHING_SCORE_LOWER:
1205         case MATCHING_SCORE_EQUAL:
1206                 gtk_entry_set_text(GTK_ENTRY(matcher.value_entry), itos(prop->value));
1207                 break;
1208
1209         case MATCHING_NOT_HEADER:
1210         case MATCHING_HEADER:
1211                 gtk_entry_set_text(GTK_ENTRY(matcher.header_entry), prop->header);
1212                 gtk_entry_set_text(GTK_ENTRY(matcher.value_entry), prop->expr);
1213                 break;
1214         }
1215
1216         if (negative_cond)
1217                 gtk_list_select_item(GTK_LIST(matcher.predicate_list), 1);
1218         else
1219                 gtk_list_select_item(GTK_LIST(matcher.predicate_list), 0);
1220
1221         switch(prop->matchtype) {
1222         case MATCHING_MATCH:
1223                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_chkbtn), FALSE);
1224                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_chkbtn), TRUE);
1225                 break;
1226
1227         case MATCHING_MATCHCASE:
1228                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_chkbtn), FALSE);
1229                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_chkbtn), FALSE);
1230                 break;
1231
1232         case MATCHING_REGEXP:
1233                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_chkbtn), TRUE);
1234                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_chkbtn), TRUE);
1235                 break;
1236
1237         case MATCHING_REGEXPCASE:
1238                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_chkbtn), TRUE);
1239                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_chkbtn), FALSE);
1240                 break;
1241         }
1242 }
1243
1244 static void prefs_matcher_criteria_select(GtkList *list,
1245                                           GtkWidget *widget,
1246                                           gpointer user_data)
1247 {
1248         gint value;
1249
1250         value = get_sel_from_list(GTK_LIST(matcher.criteria_list));
1251
1252         switch (value) {
1253         case CRITERIA_ALL:
1254                 gtk_widget_set_sensitive(matcher.header_combo, FALSE);
1255                 gtk_widget_set_sensitive(matcher.header_label, FALSE);
1256                 gtk_widget_set_sensitive(matcher.value_label, FALSE);
1257                 gtk_widget_set_sensitive(matcher.value_entry, FALSE);
1258                 gtk_widget_set_sensitive(matcher.predicate_label, FALSE);
1259                 gtk_widget_set_sensitive(matcher.predicate_combo, FALSE);
1260                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, FALSE);
1261                 gtk_widget_hide(matcher.predicate_combo);
1262                 gtk_widget_show(matcher.predicate_flag_combo);
1263                 gtk_widget_set_sensitive(matcher.case_chkbtn, FALSE);
1264                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, FALSE);
1265                 gtk_widget_set_sensitive(matcher.exec_btn, FALSE);
1266                 break;
1267
1268         case CRITERIA_UNREAD:
1269         case CRITERIA_NEW:
1270         case CRITERIA_MARKED:
1271         case CRITERIA_DELETED:
1272         case CRITERIA_REPLIED:
1273         case CRITERIA_FORWARDED:
1274                 gtk_widget_set_sensitive(matcher.header_combo, FALSE);
1275                 gtk_widget_set_sensitive(matcher.header_label, FALSE);
1276                 gtk_widget_set_sensitive(matcher.value_label, FALSE);
1277                 gtk_widget_set_sensitive(matcher.value_entry, FALSE);
1278                 gtk_widget_set_sensitive(matcher.predicate_label, TRUE);
1279                 gtk_widget_set_sensitive(matcher.predicate_combo, FALSE);
1280                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, TRUE);
1281                 gtk_widget_hide(matcher.predicate_combo);
1282                 gtk_widget_show(matcher.predicate_flag_combo);
1283                 gtk_widget_set_sensitive(matcher.case_chkbtn, FALSE);
1284                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, FALSE);
1285                 gtk_widget_set_sensitive(matcher.exec_btn, FALSE);
1286                 break;
1287
1288         case CRITERIA_SUBJECT:
1289         case CRITERIA_FROM:
1290         case CRITERIA_TO:
1291         case CRITERIA_CC:
1292         case CRITERIA_TO_OR_CC:
1293         case CRITERIA_NEWSGROUPS:
1294         case CRITERIA_INREPLYTO:
1295         case CRITERIA_REFERENCES:
1296         case CRITERIA_HEADERS_PART:
1297         case CRITERIA_BODY_PART:
1298         case CRITERIA_MESSAGE:
1299                 gtk_widget_set_sensitive(matcher.header_combo, FALSE);
1300                 gtk_widget_set_sensitive(matcher.header_label, FALSE);
1301                 gtk_widget_set_sensitive(matcher.value_label, TRUE);
1302                 gtk_widget_set_sensitive(matcher.value_entry, TRUE);
1303                 gtk_widget_set_sensitive(matcher.predicate_label, TRUE);
1304                 gtk_widget_set_sensitive(matcher.predicate_combo, TRUE);
1305                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, FALSE);
1306                 gtk_widget_show(matcher.predicate_combo);
1307                 gtk_widget_hide(matcher.predicate_flag_combo);
1308                 gtk_widget_set_sensitive(matcher.case_chkbtn, TRUE);
1309                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, TRUE);
1310                 gtk_widget_set_sensitive(matcher.exec_btn, FALSE);
1311                 break;
1312
1313         case CRITERIA_EXECUTE:
1314                 gtk_widget_set_sensitive(matcher.header_combo, FALSE);
1315                 gtk_widget_set_sensitive(matcher.header_label, FALSE);
1316                 gtk_widget_set_sensitive(matcher.value_label, TRUE);
1317                 gtk_widget_set_sensitive(matcher.value_entry, TRUE);
1318                 gtk_widget_set_sensitive(matcher.predicate_label, TRUE);
1319                 gtk_widget_set_sensitive(matcher.predicate_combo, FALSE);
1320                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, TRUE);
1321                 gtk_widget_hide(matcher.predicate_combo);
1322                 gtk_widget_show(matcher.predicate_flag_combo);
1323                 gtk_widget_set_sensitive(matcher.case_chkbtn, FALSE);
1324                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, FALSE);
1325                 gtk_widget_set_sensitive(matcher.exec_btn, TRUE);
1326                 break;
1327
1328         case CRITERIA_AGE_GREATER:
1329         case CRITERIA_AGE_LOWER:
1330         case CRITERIA_SCORE_GREATER:
1331         case CRITERIA_SCORE_LOWER:
1332         case CRITERIA_SCORE_EQUAL:
1333                 gtk_widget_set_sensitive(matcher.header_combo, FALSE);
1334                 gtk_widget_set_sensitive(matcher.header_label, FALSE);
1335                 gtk_widget_set_sensitive(matcher.value_label, TRUE);
1336                 gtk_widget_set_sensitive(matcher.value_entry, TRUE);
1337                 gtk_widget_set_sensitive(matcher.predicate_label, FALSE);
1338                 gtk_widget_set_sensitive(matcher.predicate_combo, FALSE);
1339                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, FALSE);
1340                 gtk_widget_show(matcher.predicate_combo);
1341                 gtk_widget_hide(matcher.predicate_flag_combo);
1342                 gtk_widget_set_sensitive(matcher.case_chkbtn, FALSE);
1343                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, FALSE);
1344                 gtk_widget_set_sensitive(matcher.exec_btn, FALSE);
1345                 break;
1346
1347         case CRITERIA_HEADER:
1348                 gtk_widget_set_sensitive(matcher.header_combo, TRUE);
1349                 gtk_widget_set_sensitive(matcher.header_label, TRUE);
1350                 gtk_widget_set_sensitive(matcher.value_label, TRUE);
1351                 gtk_widget_set_sensitive(matcher.value_entry, TRUE);
1352                 gtk_widget_set_sensitive(matcher.predicate_label, TRUE);
1353                 gtk_widget_set_sensitive(matcher.predicate_combo, TRUE);
1354                 gtk_widget_set_sensitive(matcher.predicate_flag_combo, FALSE);
1355                 gtk_widget_show(matcher.predicate_combo);
1356                 gtk_widget_hide(matcher.predicate_flag_combo);
1357                 gtk_widget_set_sensitive(matcher.case_chkbtn, TRUE);
1358                 gtk_widget_set_sensitive(matcher.regexp_chkbtn, TRUE);
1359                 gtk_widget_set_sensitive(matcher.exec_btn, FALSE);
1360                 break;
1361         }
1362 }
1363
1364 static void prefs_matcher_key_pressed(GtkWidget *widget, GdkEventKey *event,
1365                                      gpointer data)
1366 {
1367         if (event && event->keyval == GDK_Escape)
1368                 prefs_matcher_cancel();
1369 }
1370
1371 static void prefs_matcher_cancel(void)
1372 {
1373         gtk_widget_hide(matcher.window);
1374 }
1375
1376 static void prefs_matcher_ok(void)
1377 {
1378         MatcherList * matchers;
1379
1380         matchers = prefs_matcher_get_list();
1381         if (matchers != NULL) {
1382                 gtk_widget_hide(matcher.window);
1383                 if (matchers_callback != NULL)
1384                         matchers_callback(matchers);
1385                 matcherlist_free(matchers);
1386         }
1387         else {
1388                 gtk_widget_hide(matcher.window);
1389         }
1390 }
1391
1392 static gint prefs_matcher_deleted(GtkWidget *widget, GdkEventAny *event,
1393                                   gpointer data)
1394 {
1395         prefs_matcher_cancel();
1396         return TRUE;
1397 }
1398
1399 static GtkWidget * exec_info_win;
1400
1401 void prefs_matcher_exec_info(void)
1402 {
1403         if (!exec_info_win)
1404                 prefs_matcher_exec_info_create();
1405
1406         gtk_widget_show(exec_info_win);
1407         gtk_main();
1408         gtk_widget_hide(exec_info_win);
1409 }
1410
1411 static void prefs_matcher_exec_info_create(void)
1412 {
1413         GtkWidget *vbox;
1414         GtkWidget *hbox;
1415         GtkWidget *hbbox;
1416         GtkWidget *label;
1417         GtkWidget *ok_btn;
1418
1419         exec_info_win = gtk_window_new(GTK_WINDOW_DIALOG);
1420         gtk_window_set_title(GTK_WINDOW(exec_info_win),
1421                              _("Description of symbols"));
1422         gtk_container_set_border_width(GTK_CONTAINER(exec_info_win), 8);
1423         gtk_window_set_position(GTK_WINDOW(exec_info_win), GTK_WIN_POS_CENTER);
1424         gtk_window_set_modal(GTK_WINDOW(exec_info_win), TRUE);
1425         gtk_window_set_policy(GTK_WINDOW(exec_info_win), FALSE, TRUE, FALSE);
1426
1427         vbox = gtk_vbox_new(FALSE, 8);
1428         gtk_container_add(GTK_CONTAINER(exec_info_win), vbox);
1429
1430         hbox = gtk_hbox_new(FALSE, 4);
1431         gtk_box_pack_start(GTK_BOX(vbox), hbox, TRUE, TRUE, 0);
1432
1433         label = gtk_label_new
1434                 ("%%:\n"
1435                  "%s:\n"
1436                  "%f:\n"
1437                  "%t:\n"
1438                  "%c:\n"
1439                  "%d:\n"
1440                  "%i:\n"
1441                  "%n:\n"
1442                  "%r:\n"
1443                  "%F:\n"
1444                  "\\n:\n"
1445                  "\\:\n"
1446                  "\\\":\n"
1447                  "%%:");
1448
1449         gtk_box_pack_start(GTK_BOX(hbox), label, TRUE, TRUE, 0);
1450         gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_LEFT);
1451
1452         label = gtk_label_new
1453                 (_("%\n"
1454                    "Subject\n"
1455                    "From\n"
1456                    "To\n"
1457                    "Cc\n"
1458                    "Date\n"
1459                    "Message-ID\n"
1460                    "Newsgroups\n"
1461                    "References\n"
1462                    "Filename - should not be modified\n"
1463                    "new line\n"
1464                    "escape character for quotes\n"
1465                    "quote character\n"
1466                    "%"));
1467
1468         gtk_box_pack_start(GTK_BOX(hbox), label, TRUE, TRUE, 0);
1469         gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_LEFT);
1470
1471         gtkut_button_set_create(&hbbox, &ok_btn, _("OK"),
1472                                 NULL, NULL, NULL, NULL);
1473         gtk_box_pack_end(GTK_BOX(vbox), hbbox, FALSE, FALSE, 0);
1474
1475         gtk_widget_grab_default(ok_btn);
1476         gtk_signal_connect(GTK_OBJECT(ok_btn), "clicked",
1477                                   GTK_SIGNAL_FUNC(gtk_main_quit), NULL);
1478
1479         gtk_signal_connect(GTK_OBJECT(exec_info_win), "delete_event",
1480                                           GTK_SIGNAL_FUNC(gtk_main_quit), NULL);
1481
1482         gtk_widget_show_all(vbox);
1483 }