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