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