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