Fix wrong malloc of clamd_socket struct, CID 1220477.
[claws.git] / src / prefs_matcher.c
1 /*
2  * Claws Mail -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2018 Hiroyuki Yamamoto and the Claws Mail team
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 3 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, see <http://www.gnu.org/licenses/>.
17  */
18
19 #ifdef HAVE_CONFIG_H
20 #  include "config.h"
21 #include "claws-features.h"
22 #endif
23
24 #include "defs.h"
25
26 #include <glib.h>
27 #include <glib/gi18n.h>
28 #include <gtk/gtk.h>
29 #include <gdk/gdkkeysyms.h>
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <string.h>
33 #include <errno.h>
34
35 #if !GTK_CHECK_VERSION(3, 0, 0)
36 #include "gtkcmoptionmenu.h"
37 #endif
38 #include "main.h"
39 #include "prefs_gtk.h"
40 #include "prefs_matcher.h"
41 #include "prefs_common.h"
42 #include "mainwindow.h"
43 #include "foldersel.h"
44 #include "manage_window.h"
45 #include "inc.h"
46 #include "matcher.h"
47 #include "utils.h"
48 #include "gtkutils.h"
49 #include "alertpanel.h"
50 #include "folder.h"
51 #include "description_window.h"
52 #include "combobox.h"
53
54 #include "matcher_parser.h"
55 #include "colorlabel.h"
56 #include "tags.h"
57 #ifndef USE_ALT_ADDRBOOK
58         #include "addressbook.h"
59 #endif
60
61 #ifndef USE_ALT_ADDRBOOK
62 static void prefs_matcher_addressbook_select(void);
63 #endif
64 static void prefs_matcher_test_info(GtkWidget *widget, GtkWidget *parent);
65
66 enum {
67         PREFS_MATCHER_COND,
68         PREFS_MATCHER_COND_VALID,
69         N_PREFS_MATCHER_COLUMNS
70 };
71
72 /*!
73  *\brief        UI data for matcher dialog
74  */
75 static struct Matcher {
76         GtkWidget *window;
77
78         GtkWidget *ok_btn;
79
80         GtkWidget *match_combo;
81         GtkWidget *header_addr_combo;
82         GtkWidget *bool_op_combo;
83         GtkWidget *criteria_label2;
84         GtkWidget *criteria_combo;
85         GtkWidget *criteria_combo2;
86         GtkWidget *match_combo2;
87         GtkWidget *match_label;
88         GtkWidget *match_label2;
89         GtkWidget *headers_combo;
90         GtkWidget *upper_filler;
91         GtkWidget *lower_filler;
92
93         GtkWidget *header_entry;
94         GtkWidget *header_addr_entry;
95         GtkWidget *string_entry;
96         GtkWidget *numeric_entry;
97         GtkWidget *numeric_label;
98         GtkWidget *addressbook_folder_combo;
99         GtkWidget *case_checkbtn;
100         GtkWidget *regexp_checkbtn;
101 #if !GTK_CHECK_VERSION(3, 0, 0)
102         GtkWidget *color_optmenu;
103 #endif
104
105         GtkWidget *test_btn;
106         GtkWidget *addressbook_select_btn;
107
108         GtkTreeModel *model_age;
109         GtkTreeModel *model_age_units;
110         GtkTreeModel *model_contain;
111         GtkTreeModel *model_found;
112         GtkTreeModel *model_flags;
113         GtkTreeModel *model_headers;
114         GtkTreeModel *model_partial;
115         GtkTreeModel *model_phrase;
116         GtkTreeModel *model_score;
117         GtkTreeModel *model_set;
118         GtkTreeModel *model_size;
119         GtkTreeModel *model_size_units;
120         GtkTreeModel *model_tags;
121         GtkTreeModel *model_test;
122         GtkTreeModel *model_thread;
123         
124         GtkWidget *cond_list_view;
125
126         gint selected_criteria; /*!< selected criteria in combobox */ 
127 } matcher;
128
129 /*!
130  *\brief        Conditions with a negate counterpart (like unread and ~unread)
131  *              have the same CRITERIA_XXX id). I.e. both unread and ~unread
132  *              have criteria id CRITERIA_UNREAD. This id is passed as the
133  *              first parameter to #matcherprop_new and #matcherprop_unquote_new.
134  */             
135 enum {
136         CRITERIA_ALL = 0,
137
138         CRITERIA_SUBJECT = 1,
139         CRITERIA_FROM = 2,
140         CRITERIA_TO = 3,
141         CRITERIA_CC = 4,
142         CRITERIA_TO_OR_CC = 5,
143         CRITERIA_NEWSGROUPS = 6,
144         CRITERIA_INREPLYTO = 7,
145         CRITERIA_REFERENCES = 8,
146         CRITERIA_AGE_GREATER = 9,
147         CRITERIA_AGE_LOWER = 10,
148         CRITERIA_HEADER = 11,
149         CRITERIA_HEADERS_PART = 12,
150         CRITERIA_BODY_PART = 13,
151         CRITERIA_MESSAGE = 14,
152
153         CRITERIA_UNREAD = 15,
154         CRITERIA_NEW = 16,
155         CRITERIA_MARKED = 17,
156         CRITERIA_DELETED = 18,
157         CRITERIA_REPLIED = 19,
158         CRITERIA_FORWARDED = 20,
159         CRITERIA_LOCKED = 21,
160         CRITERIA_SPAM = 22,
161         CRITERIA_COLORLABEL = 23,
162         CRITERIA_IGNORE_THREAD = 24,
163         CRITERIA_WATCH_THREAD = 25,
164
165         CRITERIA_SCORE_GREATER = 26,
166         CRITERIA_SCORE_LOWER = 27,
167         CRITERIA_SCORE_EQUAL = 28,
168
169         CRITERIA_TEST = 29,
170
171         CRITERIA_SIZE_GREATER = 30,
172         CRITERIA_SIZE_SMALLER = 31,
173         CRITERIA_SIZE_EQUAL   = 32,
174         
175         CRITERIA_PARTIAL = 33,
176
177         CRITERIA_FOUND_IN_ADDRESSBOOK = 34,
178         
179         CRITERIA_TAG = 35,
180         CRITERIA_TAGGED = 36,
181
182         CRITERIA_HAS_ATTACHMENT = 37,
183         CRITERIA_SIGNED = 38,
184
185         CRITERIA_AGE_GREATER_HOURS = 39,
186         CRITERIA_AGE_LOWER_HOURS = 40,
187
188         CRITERIA_MESSAGEID = 41,
189         CRITERIA_HEADERS_CONT = 42
190 };
191
192 enum {
193         MATCH_ALL       = 0,
194         MATCH_HEADER    = 1,
195         MATCH_AGE       = 2,
196         MATCH_PHRASE    = 3,
197         MATCH_FLAG      = 4,
198         MATCH_LABEL     = 5,
199         MATCH_THREAD    = 6,
200         MATCH_SCORE     = 7,
201         MATCH_SIZE      = 8,
202         MATCH_PARTIAL   = 9,
203         MATCH_ABOOK     = 10,
204         MATCH_TAGS      = 11,
205         MATCH_TEST      = 12
206 };
207
208 enum {
209         AGE_HOURS = 0,
210         AGE_DAYS  = 1,
211         AGE_WEEKS = 2
212 };
213
214 enum {
215         SIZE_UNIT_BYTES  = 0,
216         SIZE_UNIT_KBYTES = 1,
217         SIZE_UNIT_MBYTES = 2
218 };
219
220 #define MB_SIZE 0x100000
221 #define KB_SIZE 0x000400
222
223 enum {
224         THREAD_IGNORED = 0,
225         THREAD_NOT_IGNORED = 1,
226         THREAD_WATCHED = 2,
227         THREAD_NOT_WATCHED = 3
228 };
229
230 /*!
231  *\brief        Contains predicate      
232  */
233 enum {
234         PREDICATE_CONTAINS = 0,
235         PREDICATE_DOES_NOT_CONTAIN = 1
236 };
237
238 /*!
239  *\brief        Enabled predicate
240  */
241 enum {
242         PREDICATE_FLAG_ENABLED = 0,
243         PREDICATE_FLAG_DISABLED = 1
244 };
245
246 /*!
247  *\brief        Hooks
248  */
249 static PrefsMatcherSignal *matchers_callback;
250
251 /* widget creating functions */
252 static void prefs_matcher_create        (void);
253
254 static void prefs_matcher_set_dialog    (MatcherList *matchers);
255 static void prefs_matcher_list_view_set_row     (GtkTreeIter *row, 
256                                                  MatcherProp *prop);
257
258 /* callback functions */
259
260 static void prefs_matcher_register_cb   (void);
261 static void prefs_matcher_substitute_cb (void);
262 static void prefs_matcher_delete_cb     (void);
263 static void prefs_matcher_up            (void);
264 static void prefs_matcher_down          (void);
265 static gboolean prefs_matcher_key_pressed(GtkWidget     *widget,
266                                          GdkEventKey    *event,
267                                          gpointer        data);
268 static void prefs_matcher_ok            (void);
269 static void prefs_matcher_cancel        (void);
270 static gint prefs_matcher_deleted       (GtkWidget *widget, GdkEventAny *event,
271                                          gpointer data);
272 static void prefs_matcher_criteria_select       (GtkWidget *widget,
273                                                  gpointer   user_data);
274 static void prefs_matcher_second_criteria_sel   (GtkWidget *widget,
275                                                  gpointer   user_data);
276 static void prefs_matcher_set_model             (GtkWidget *wiget,
277                                                  GtkTreeModel *model);
278 static MatcherList *prefs_matcher_get_list      (void);
279
280 static GtkListStore* prefs_matcher_create_data_store    (void);
281
282 static void prefs_matcher_list_view_insert_matcher      (GtkWidget *list_view,
283                                                          GtkTreeIter *row_iter,
284                                                          const gchar *matcher,
285                                                          gboolean is_valid);
286
287 static GtkWidget *prefs_matcher_list_view_create        (void);
288
289 static void prefs_matcher_create_list_view_columns      (GtkWidget *list_view);
290
291 static gboolean prefs_matcher_selected                  (GtkTreeSelection *selector,
292                                                          GtkTreeModel *model, 
293                                                          GtkTreePath *path,
294                                                          gboolean currently_selected,
295                                                          gpointer data);
296
297 static int header_name_to_crit(const gchar *header)
298 {
299         if (header == NULL)
300                 return CRITERIA_HEADER;
301
302         if (!strcasecmp(header, "Subject"))
303                 return CRITERIA_SUBJECT;
304         if (!strcasecmp(header, "From"))
305                 return CRITERIA_FROM;
306         if (!strcasecmp(header, "To"))
307                 return CRITERIA_TO;
308         if (!strcasecmp(header, "Cc"))
309                 return CRITERIA_CC;
310         if (!strcasecmp(header, "To or Cc"))
311                 return CRITERIA_TO_OR_CC;
312         if (!strcasecmp(header, "Message-ID"))
313                 return CRITERIA_MESSAGEID;
314         if (!strcasecmp(header, "In-Reply-To"))
315                 return CRITERIA_INREPLYTO;
316         if (!strcasecmp(header, "Newsgroups"))
317                 return CRITERIA_NEWSGROUPS;
318         if (!strcasecmp(header, "References"))
319                 return CRITERIA_REFERENCES;
320
321         return CRITERIA_HEADER;
322 }
323
324 static void prefs_matcher_models_create(void)
325 {
326         GtkListStore *store;
327         GtkTreeIter iter;
328         
329         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
330         COMBOBOX_ADD(store, _("more than"), CRITERIA_AGE_GREATER);
331         COMBOBOX_ADD(store, _("less than"), CRITERIA_AGE_LOWER);
332         matcher.model_age = GTK_TREE_MODEL(store);
333
334         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
335         COMBOBOX_ADD(store, _("hours"), AGE_HOURS);
336         COMBOBOX_ADD(store, _("days"), AGE_DAYS);
337         COMBOBOX_ADD(store, _("weeks"), AGE_WEEKS);
338         matcher.model_age_units = GTK_TREE_MODEL(store);
339
340         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
341         COMBOBOX_ADD(store, _("higher than"), CRITERIA_SCORE_GREATER);
342         COMBOBOX_ADD(store, _("lower than"), CRITERIA_SCORE_LOWER);
343         COMBOBOX_ADD(store, _("exactly"), CRITERIA_SCORE_EQUAL);
344         matcher.model_score = GTK_TREE_MODEL(store);
345         
346         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
347         COMBOBOX_ADD(store, _("greater than"), CRITERIA_SIZE_GREATER);
348         COMBOBOX_ADD(store, _("smaller than"), CRITERIA_SIZE_SMALLER);
349         COMBOBOX_ADD(store, _("exactly"), CRITERIA_SIZE_EQUAL);
350         matcher.model_size = GTK_TREE_MODEL(store);
351
352         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
353         COMBOBOX_ADD(store, _("bytes"), SIZE_UNIT_BYTES);
354         COMBOBOX_ADD(store, _("kilobytes"), SIZE_UNIT_KBYTES);
355         COMBOBOX_ADD(store, _("megabytes"), SIZE_UNIT_MBYTES);
356         matcher.model_size_units = GTK_TREE_MODEL(store);
357         
358         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
359         COMBOBOX_ADD(store, _("contains"), 0);
360         COMBOBOX_ADD(store, _("doesn't contain"), 0);
361         matcher.model_contain = GTK_TREE_MODEL(store);
362         
363         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
364         COMBOBOX_ADD(store, "Subject", CRITERIA_SUBJECT);
365         COMBOBOX_ADD(store, "From", CRITERIA_FROM);
366         COMBOBOX_ADD(store, "To", CRITERIA_TO);
367         COMBOBOX_ADD(store, "Cc", CRITERIA_CC);
368         COMBOBOX_ADD(store, "To or Cc", CRITERIA_TO_OR_CC);
369         COMBOBOX_ADD(store, "Message-ID", CRITERIA_MESSAGEID);
370         COMBOBOX_ADD(store, "In-Reply-To", CRITERIA_INREPLYTO);
371         COMBOBOX_ADD(store, "Newsgroups", CRITERIA_NEWSGROUPS);
372         COMBOBOX_ADD(store, "References", CRITERIA_REFERENCES);
373         COMBOBOX_ADD(store, "Sender", CRITERIA_HEADER);
374         COMBOBOX_ADD(store, "X-ML-Name", CRITERIA_HEADER);
375         COMBOBOX_ADD(store, "X-List", CRITERIA_HEADER);
376         COMBOBOX_ADD(store, "X-Sequence", CRITERIA_HEADER);
377         COMBOBOX_ADD(store, "X-Mailer", CRITERIA_HEADER);
378         COMBOBOX_ADD(store, "X-BeenThere", CRITERIA_HEADER);
379         COMBOBOX_ADD(store, "List-Post", CRITERIA_HEADER);
380         COMBOBOX_ADD(store, "List-Id", CRITERIA_HEADER);
381         matcher.model_headers = GTK_TREE_MODEL(store);
382         
383         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
384         COMBOBOX_ADD(store, _("headers part"), CRITERIA_HEADERS_PART);
385         COMBOBOX_ADD(store, _("headers values"), CRITERIA_HEADERS_CONT);
386         COMBOBOX_ADD(store, _("body part"), CRITERIA_BODY_PART);
387         COMBOBOX_ADD(store, _("whole message"), CRITERIA_MESSAGE);
388         matcher.model_phrase = GTK_TREE_MODEL(store);
389         
390         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
391         COMBOBOX_ADD(store, _("Unread"), CRITERIA_UNREAD);
392         COMBOBOX_ADD(store, _("New"), CRITERIA_NEW);
393         COMBOBOX_ADD(store, _("Marked"), CRITERIA_MARKED);
394         COMBOBOX_ADD(store, _("Deleted"), CRITERIA_DELETED);
395         COMBOBOX_ADD(store, _("Replied"), CRITERIA_REPLIED);
396         COMBOBOX_ADD(store, _("Forwarded"), CRITERIA_FORWARDED);
397         COMBOBOX_ADD(store, _("Locked"), CRITERIA_LOCKED);
398         COMBOBOX_ADD(store, _("Spam"), CRITERIA_SPAM);
399         COMBOBOX_ADD(store, _("Has attachment"), CRITERIA_HAS_ATTACHMENT);
400         COMBOBOX_ADD(store, _("Signed"), CRITERIA_SIGNED);
401         matcher.model_flags = GTK_TREE_MODEL(store);
402         
403         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
404         COMBOBOX_ADD(store, _("set"), 0);
405         COMBOBOX_ADD(store, _("not set"), 1);
406         matcher.model_set = GTK_TREE_MODEL(store);
407
408         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
409         COMBOBOX_ADD(store, _("yes"), CRITERIA_PARTIAL);
410         COMBOBOX_ADD(store, _("no"), CRITERIA_PARTIAL);
411         matcher.model_partial = GTK_TREE_MODEL(store);
412
413         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
414         COMBOBOX_ADD(store, _("Any tags"), CRITERIA_TAGGED);
415         COMBOBOX_ADD(store, _("Specific tag"), CRITERIA_TAG);
416         matcher.model_tags = GTK_TREE_MODEL(store);
417
418         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
419         COMBOBOX_ADD(store, _("ignored"), CRITERIA_IGNORE_THREAD);
420         COMBOBOX_ADD(store, _("not ignored"), CRITERIA_IGNORE_THREAD);
421         COMBOBOX_ADD(store, _("watched"), CRITERIA_WATCH_THREAD);
422         COMBOBOX_ADD(store, _("not watched"), CRITERIA_WATCH_THREAD);
423         matcher.model_thread = GTK_TREE_MODEL(store);
424         
425         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
426         COMBOBOX_ADD(store, _("found"), 0);
427         COMBOBOX_ADD(store, _("not found"), 1);
428         matcher.model_found = GTK_TREE_MODEL(store);
429
430         store = gtk_list_store_new(3, G_TYPE_STRING, G_TYPE_INT, G_TYPE_BOOLEAN);
431         COMBOBOX_ADD(store, _("0 (Passed)"), 0);
432         COMBOBOX_ADD(store, _("non-0 (Failed)"), 1);
433         matcher.model_test = GTK_TREE_MODEL(store);
434 }
435
436 /*!
437  *\brief        Opens the matcher dialog with a list of conditions
438  *
439  *\param        matchers List of conditions
440  *\param        cb Callback
441  *
442  */
443 void prefs_matcher_open(MatcherList *matchers, PrefsMatcherSignal *cb)
444 {
445         inc_lock();
446
447         if (!matcher.window) {
448                 prefs_matcher_models_create();
449                 prefs_matcher_create();
450         } else {
451 #if !GTK_CHECK_VERSION(3, 0, 0)
452                 /* update color label menu */
453                 gtk_cmoption_menu_set_menu(GTK_CMOPTION_MENU(matcher.color_optmenu),
454                                 colorlabel_create_color_menu());
455 #endif
456         }
457
458         manage_window_set_transient(GTK_WINDOW(matcher.window));
459         gtk_widget_grab_focus(matcher.ok_btn);
460
461         matchers_callback = cb;
462
463         prefs_matcher_set_dialog(matchers);
464
465         gtk_widget_show(matcher.window);
466         gtk_window_set_modal(GTK_WINDOW(matcher.window), TRUE);
467 }
468
469 /*!
470  *\brief        Save Gtk object size to prefs dataset
471  */
472 static void prefs_matcher_size_allocate_cb(GtkWidget *widget,
473                                          GtkAllocation *allocation)
474 {
475         cm_return_if_fail(allocation != NULL);
476
477         prefs_common.matcherwin_width = allocation->width;
478         prefs_common.matcherwin_height = allocation->height;
479 }
480
481 /*!
482  *\brief        Create the matcher dialog
483  */
484 static void prefs_matcher_create(void)
485 {
486         GtkWidget *window;
487         GtkWidget *vbox;
488         GtkWidget *ok_btn;
489         GtkWidget *cancel_btn;
490         GtkWidget *confirm_area;
491
492         GtkWidget *vbox1;
493         GtkWidget *frame;
494         GtkWidget *table;
495         GtkWidget *upper_hbox;
496         GtkWidget *lower_hbox;
497         GtkWidget *match_hbox;
498         GtkWidget *criteria_combo;
499         GtkWidget *criteria_label;
500         GtkWidget *match_label;
501         GtkWidget *criteria_label2;
502         GtkWidget *headers_combo;
503         GtkWidget *match_combo2;
504         GtkWidget *match_label2;
505
506         GtkWidget *hbox;
507         GtkWidget *upper_filler;
508         GtkWidget *lower_filler;
509         
510         GtkWidget *criteria_combo2;
511         GtkWidget *header_entry;
512         GtkWidget *header_addr_combo;
513         GtkWidget *header_addr_entry;
514         GtkWidget *string_entry;
515         GtkWidget *addressbook_folder_combo;
516         GtkWidget *match_combo;
517         GtkWidget *bool_op_combo;
518         GtkWidget *bool_op_label;
519
520         GtkWidget *numeric_hbox;
521         GtkWidget *numeric_entry;
522         GtkWidget *numeric_label;
523         
524         GtkWidget *regexp_checkbtn;
525         GtkWidget *case_checkbtn;
526
527         GtkWidget *reg_hbox;
528         GtkWidget *btn_hbox;
529         GtkWidget *arrow;
530         GtkWidget *reg_btn;
531         GtkWidget *subst_btn;
532         GtkWidget *del_btn;
533
534         GtkWidget *cond_hbox;
535         GtkWidget *cond_scrolledwin;
536         GtkWidget *cond_list_view;
537
538         GtkWidget *btn_vbox;
539         GtkWidget *up_btn;
540         GtkWidget *down_btn;
541
542         GtkWidget *test_btn;
543         GtkWidget *addressbook_select_btn;
544 #if !GTK_CHECK_VERSION(3, 0, 0)
545         GtkWidget *color_optmenu;
546 #endif
547
548         static GdkGeometry geometry;
549         GtkSizeGroup *size_group;
550         GtkListStore *store;
551         GtkTreeIter iter;
552
553         debug_print("Creating matcher configuration window...\n");
554
555         window = gtkut_window_new(GTK_WINDOW_TOPLEVEL, "prefs_matcher");
556         gtk_container_set_border_width(GTK_CONTAINER(window), 4);
557         gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
558
559         vbox = gtk_vbox_new(FALSE, 6);
560         gtk_container_add(GTK_CONTAINER(window), vbox);
561
562         gtkut_stock_button_set_create(&confirm_area, &cancel_btn, GTK_STOCK_CANCEL,
563                                       &ok_btn, GTK_STOCK_OK, NULL, NULL);
564         gtk_box_pack_end(GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
565         gtk_widget_grab_default(ok_btn);
566
567         gtk_window_set_title(GTK_WINDOW(window),
568                              _("Condition configuration"));
569         g_signal_connect(G_OBJECT(window), "delete_event",
570                          G_CALLBACK(prefs_matcher_deleted), NULL);
571         g_signal_connect(G_OBJECT(window), "size_allocate",
572                          G_CALLBACK(prefs_matcher_size_allocate_cb), NULL);
573         g_signal_connect(G_OBJECT(window), "key_press_event",
574                          G_CALLBACK(prefs_matcher_key_pressed), NULL);
575         MANAGE_WINDOW_SIGNALS_CONNECT(window);
576         g_signal_connect(G_OBJECT(ok_btn), "clicked",
577                          G_CALLBACK(prefs_matcher_ok), NULL);
578         g_signal_connect(G_OBJECT(cancel_btn), "clicked",
579                          G_CALLBACK(prefs_matcher_cancel), NULL);
580
581         vbox1 = gtk_vbox_new(FALSE, VSPACING);
582         gtk_box_pack_start(GTK_BOX(vbox), vbox1, TRUE, TRUE, 0);
583         gtk_container_set_border_width(GTK_CONTAINER (vbox1), 2);
584
585         frame = gtk_frame_new(_("Rule"));
586         gtk_frame_set_label_align(GTK_FRAME(frame), 0.01, 0.5);
587         gtk_box_pack_start(GTK_BOX(vbox1), frame, FALSE, FALSE, 0);
588         
589         table = gtk_table_new(3, 3, FALSE);
590         gtk_container_add(GTK_CONTAINER(frame), table);
591         gtk_widget_set_size_request(frame, -1, -1);
592         
593         upper_hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
594         hbox = gtk_hbox_new(FALSE, 0);
595         gtk_box_pack_start(GTK_BOX(hbox), upper_hbox, FALSE, FALSE, 0);
596         gtk_box_pack_start(GTK_BOX(hbox), gtk_label_new(""), TRUE, TRUE, 0);
597         gtk_table_attach(GTK_TABLE(table), hbox, 2, 3, 0, 1, 
598                         GTK_FILL, GTK_SHRINK, 2, 2);
599         
600         lower_hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
601         hbox = gtk_hbox_new(FALSE, 0);
602         gtk_box_pack_start(GTK_BOX(hbox), lower_hbox, FALSE, FALSE, 0);
603         gtk_box_pack_start(GTK_BOX(hbox), gtk_label_new(""), TRUE, TRUE, 0);
604         gtk_table_attach(GTK_TABLE(table), hbox, 2, 3, 1, 2, 
605                          GTK_FILL, GTK_SHRINK, 2, 2);
606         
607         size_group = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
608         gtk_size_group_add_widget(size_group, upper_hbox);
609         gtk_size_group_add_widget(size_group, lower_hbox);
610         
611         /* criteria combo box */
612         criteria_label = gtk_label_new(_("Match criteria"));
613         gtk_misc_set_alignment(GTK_MISC(criteria_label), 1, 0.5);
614         gtk_widget_set_size_request(criteria_label, -1, -1);
615         gtk_table_attach(GTK_TABLE(table), criteria_label, 0, 1, 0, 1, 
616                          GTK_FILL, GTK_SHRINK, 2, 2);
617
618         criteria_combo = gtkut_sc_combobox_create(NULL, FALSE);
619         store = GTK_LIST_STORE(gtk_combo_box_get_model(
620                                 GTK_COMBO_BOX(criteria_combo)));
621         COMBOBOX_ADD(store, _("All messages"), 0);
622         COMBOBOX_ADD(store, _("Header"), 1);
623         COMBOBOX_ADD(store, _("Age"), 2);
624         COMBOBOX_ADD(store, _("Phrase"), 3);
625         COMBOBOX_ADD(store, _("Flags"), 4);
626         COMBOBOX_ADD(store, _("Color labels"), 5);
627         COMBOBOX_ADD(store, _("Thread"), 6);
628         COMBOBOX_ADD(store, _("Score"), 7);
629         COMBOBOX_ADD(store, _("Size"), 8);
630         COMBOBOX_ADD(store, _("Partially downloaded"), 9);
631         COMBOBOX_ADD(store, _("Address book"), 10);
632         COMBOBOX_ADD(store, _("Tags"), 11);
633         COMBOBOX_ADD(store, _("External program test"), 12);
634
635         gtk_widget_set_size_request(criteria_combo, 150, -1);
636         gtk_combo_box_set_active(GTK_COMBO_BOX(criteria_combo), MATCH_ALL);
637         gtk_table_attach(GTK_TABLE(table), criteria_combo, 1, 2, 0, 1,
638                          GTK_FILL, GTK_SHRINK, 2, 2);
639         g_signal_connect(G_OBJECT(criteria_combo), "changed",
640                          G_CALLBACK(prefs_matcher_criteria_select),
641                          NULL);
642         
643         upper_filler = gtk_label_new("");
644         gtk_box_pack_start(GTK_BOX(upper_hbox), upper_filler, TRUE, TRUE, 0); 
645         
646         lower_filler = gtk_label_new("");
647         gtk_box_pack_start(GTK_BOX(lower_hbox), lower_filler, TRUE, TRUE, 0);
648                          
649         criteria_label2 = gtk_label_new("");
650         gtk_box_pack_start(GTK_BOX(upper_hbox), criteria_label2, FALSE, FALSE, 0);
651
652         /* headers combo box entry */
653         headers_combo = gtk_combo_box_new_with_model_and_entry(matcher.model_headers);
654         GtkCellRenderer *cell = gtk_cell_renderer_text_new();
655         gtk_cell_renderer_set_alignment(cell, 0.0, 0.5);
656         gtk_cell_layout_pack_start(GTK_CELL_LAYOUT(headers_combo), cell, TRUE);
657         gtk_combo_box_set_entry_text_column(GTK_COMBO_BOX(headers_combo), 0);
658         gtk_widget_set_size_request(headers_combo, 100, -1);
659         gtk_box_pack_start(GTK_BOX(upper_hbox), headers_combo, TRUE, TRUE, 0);
660         header_entry = gtk_bin_get_child(GTK_BIN((headers_combo)));
661         
662         criteria_combo2 = gtkut_sc_combobox_create(NULL, TRUE);
663         prefs_matcher_set_model(criteria_combo2, matcher.model_phrase);
664         gtk_box_pack_start(GTK_BOX(upper_hbox), criteria_combo2, TRUE, TRUE, 0);
665         g_signal_connect(G_OBJECT(criteria_combo2), "changed",
666                          G_CALLBACK(prefs_matcher_second_criteria_sel),
667                          NULL);
668
669         /* book/folder value */
670         addressbook_folder_combo = combobox_text_new(TRUE, _("Any"), NULL);
671         gtk_widget_set_size_request(addressbook_folder_combo, 250, -1);
672         gtk_box_pack_start(GTK_BOX(upper_hbox), addressbook_folder_combo, TRUE, TRUE, 0);                        
673
674         addressbook_select_btn = gtk_button_new_with_label(_("Select..."));
675         gtk_box_pack_start(GTK_BOX(upper_hbox), addressbook_select_btn, FALSE, FALSE, 0);
676 #ifndef USE_ALT_ADDRBOOK
677         g_signal_connect(G_OBJECT (addressbook_select_btn), "clicked",
678                          G_CALLBACK(prefs_matcher_addressbook_select),
679                          NULL);
680 #else
681         gtk_widget_set_sensitive(GTK_WIDGET(addressbook_select_btn), FALSE);
682 #endif
683         match_label = gtk_label_new("");
684         gtk_misc_set_alignment(GTK_MISC(match_label), 1, 0.5);
685         gtk_table_attach(GTK_TABLE(table), match_label, 0, 1, 1, 2,
686                          GTK_FILL, GTK_SHRINK, 2, 2);
687
688         match_hbox = gtk_hbox_new(FALSE, 0);
689         gtk_table_attach(GTK_TABLE(table), match_hbox, 1, 2, 1, 2,
690                          GTK_FILL, GTK_SHRINK, 2, 2); 
691
692         match_combo = gtkut_sc_combobox_create(NULL, TRUE);
693         gtk_box_pack_start(GTK_BOX(match_hbox), match_combo, TRUE, TRUE, 0);
694         
695 #if !GTK_CHECK_VERSION(3, 0, 0)
696         /* color labels combo */
697         color_optmenu = gtk_cmoption_menu_new();
698         gtk_cmoption_menu_set_menu(GTK_CMOPTION_MENU(color_optmenu),
699                                  colorlabel_create_color_menu());
700         gtk_box_pack_start(GTK_BOX(match_hbox), color_optmenu, FALSE, FALSE, 0);
701 #endif
702         
703         /* address header name */
704         header_addr_combo = combobox_text_new(TRUE,
705                               C_("Filtering Matcher Menu", "All"), _("Any"),
706                               "From", "To", "Cc", "Reply-To", "Sender", NULL);
707         gtk_box_pack_start(GTK_BOX(match_hbox), header_addr_combo, FALSE, FALSE, 0);
708         header_addr_entry = gtk_bin_get_child(GTK_BIN((header_addr_combo)));
709         gtk_widget_set_size_request(header_addr_combo, 150, -1);
710         
711         match_label2 = gtk_label_new("");
712         gtk_box_pack_start(GTK_BOX(lower_hbox), match_label2, FALSE, FALSE, 0);
713
714         /* numeric value */
715         numeric_hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
716         gtk_box_pack_start(GTK_BOX(lower_hbox), numeric_hbox, FALSE, FALSE, 0);
717
718         numeric_entry = gtk_spin_button_new_with_range(0, 1000, 1);
719         gtk_spin_button_set_digits(GTK_SPIN_BUTTON(numeric_entry), 0);
720         gtk_box_pack_start(GTK_BOX(numeric_hbox), numeric_entry, FALSE, FALSE, 0);
721         
722         numeric_label = gtk_label_new("");
723         gtk_box_pack_start(GTK_BOX(numeric_hbox), numeric_label, FALSE, FALSE, 0);
724         gtk_box_pack_end(GTK_BOX(numeric_hbox), gtk_label_new(""), TRUE, TRUE, 0);
725
726         match_combo2 = gtkut_sc_combobox_create(NULL, TRUE);
727         gtk_box_pack_start(GTK_BOX(lower_hbox), match_combo2, TRUE, TRUE, 0);
728         
729         /* string value */
730         string_entry = gtk_entry_new();
731         gtk_box_pack_start(GTK_BOX(lower_hbox), string_entry, TRUE, TRUE, 0);
732         gtk_widget_set_size_request(string_entry, 300, -1);
733
734         hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
735         gtk_size_group_add_widget(size_group, hbox);
736
737         vbox = gtk_vbox_new(FALSE, VSPACING_NARROW);
738         PACK_CHECK_BUTTON(vbox, case_checkbtn, _("Case sensitive"));
739         PACK_CHECK_BUTTON(vbox, regexp_checkbtn, _("Use regexp"));
740         gtk_box_pack_start(GTK_BOX(hbox), vbox, TRUE, TRUE, 0);
741
742         gtk_box_pack_end(GTK_BOX(hbox), gtk_label_new(""), TRUE, TRUE, 0);
743         gtk_table_attach(GTK_TABLE(table), hbox, 2, 3, 2, 3,
744                          GTK_FILL, GTK_SHRINK, 4, 0);
745
746         /* test info button */
747         test_btn = gtk_button_new_from_stock(GTK_STOCK_INFO);
748         gtk_box_pack_start(GTK_BOX(lower_hbox), test_btn, FALSE, FALSE, 0);
749         g_signal_connect(G_OBJECT (test_btn), "clicked",
750                          G_CALLBACK(prefs_matcher_test_info),
751                          window);
752
753         /* register / substitute / delete */
754         reg_hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
755         gtk_box_pack_start(GTK_BOX(vbox1), reg_hbox, FALSE, FALSE, 0);
756
757         arrow = gtk_arrow_new(GTK_ARROW_DOWN, GTK_SHADOW_OUT);
758         gtk_box_pack_start(GTK_BOX(reg_hbox), arrow, FALSE, FALSE, 0);
759         gtk_widget_set_size_request(arrow, -1, 16);
760
761         btn_hbox = gtk_hbox_new(FALSE, HSPACING_NARROW);
762         gtk_box_pack_start(GTK_BOX(reg_hbox), btn_hbox, FALSE, FALSE, 0);
763
764         reg_btn = gtk_button_new_from_stock(GTK_STOCK_ADD);
765         gtk_box_pack_start(GTK_BOX(btn_hbox), reg_btn, FALSE, TRUE, 0);
766         g_signal_connect(G_OBJECT(reg_btn), "clicked",
767                          G_CALLBACK(prefs_matcher_register_cb), NULL);
768
769         subst_btn = gtkut_get_replace_btn(_("_Replace"));
770         gtk_box_pack_start(GTK_BOX(btn_hbox), subst_btn, FALSE, TRUE, 0);
771         g_signal_connect(G_OBJECT(subst_btn), "clicked",
772                          G_CALLBACK(prefs_matcher_substitute_cb),
773                          NULL);
774
775         del_btn = gtk_button_new_with_mnemonic (_("D_elete"));
776         gtk_button_set_image(GTK_BUTTON(del_btn),
777                         gtk_image_new_from_stock(GTK_STOCK_REMOVE,GTK_ICON_SIZE_BUTTON));
778         gtk_box_pack_start(GTK_BOX(btn_hbox), del_btn, FALSE, TRUE, 0);
779         g_signal_connect(G_OBJECT(del_btn), "clicked",
780                          G_CALLBACK(prefs_matcher_delete_cb), NULL);
781
782         cond_hbox = gtk_hbox_new(FALSE, VBOX_BORDER);
783         gtk_box_pack_start(GTK_BOX(vbox1), cond_hbox, TRUE, TRUE, 0);
784
785         cond_scrolledwin = gtk_scrolled_window_new(NULL, NULL);
786         gtk_widget_set_size_request(cond_scrolledwin, -1, 150);
787         gtk_box_pack_start(GTK_BOX(cond_hbox), cond_scrolledwin,
788                            TRUE, TRUE, 0);
789         gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(cond_scrolledwin),
790                                        GTK_POLICY_AUTOMATIC,
791                                        GTK_POLICY_AUTOMATIC);
792
793         cond_list_view = prefs_matcher_list_view_create();                                     
794         gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(cond_scrolledwin),
795                                             GTK_SHADOW_ETCHED_IN);
796         gtk_container_add(GTK_CONTAINER(cond_scrolledwin), cond_list_view);
797
798         btn_vbox = gtk_vbox_new(FALSE, VBOX_BORDER);
799         gtk_box_pack_start(GTK_BOX(cond_hbox), btn_vbox, FALSE, FALSE, 0);
800
801         up_btn = gtk_button_new_from_stock(GTK_STOCK_GO_UP);
802         gtk_box_pack_start(GTK_BOX(btn_vbox), up_btn, FALSE, FALSE, 0);
803         g_signal_connect(G_OBJECT(up_btn), "clicked",
804                          G_CALLBACK(prefs_matcher_up), NULL);
805
806         down_btn = gtk_button_new_from_stock(GTK_STOCK_GO_DOWN);
807         gtk_box_pack_start(GTK_BOX(btn_vbox), down_btn, FALSE, FALSE, 0);
808         g_signal_connect(G_OBJECT(down_btn), "clicked",
809                          G_CALLBACK(prefs_matcher_down), NULL);
810
811         /* boolean operation */
812         GtkWidget *hbox_bool = gtk_hbox_new(FALSE, HSPACING_NARROW);
813         gtk_box_pack_start(GTK_BOX(vbox1), hbox_bool, FALSE, FALSE, 0);
814
815         bool_op_label = gtk_label_new(_("Message must match"));
816         gtk_box_pack_start(GTK_BOX(hbox_bool), bool_op_label,
817                            FALSE, FALSE, 0);
818
819         bool_op_combo = combobox_text_new(FALSE, _("at least one"), 
820                                           _("all"), NULL);
821         gtk_box_pack_start(GTK_BOX(hbox_bool), bool_op_combo,
822                            FALSE, FALSE, 0);
823         gtk_box_pack_start(GTK_BOX(hbox_bool), gtk_label_new(_("of above rules")),
824                            FALSE, FALSE, 0);
825         gtk_box_pack_start(GTK_BOX(hbox_bool), gtk_label_new(""),
826                            TRUE, TRUE, 0);
827         
828         if (!geometry.min_height) {
829                 geometry.min_width = 630;
830                 geometry.min_height = 368;
831         }
832
833         gtk_window_set_geometry_hints(GTK_WINDOW(window), NULL, &geometry,
834                                       GDK_HINT_MIN_SIZE);
835         gtk_widget_set_size_request(window, prefs_common.matcherwin_width,
836                                     prefs_common.matcherwin_height);
837
838         gtk_widget_show_all(window);
839
840         matcher.window    = window;
841
842         matcher.ok_btn = ok_btn;
843
844         matcher.criteria_combo = criteria_combo;
845         matcher.criteria_combo2 = criteria_combo2;
846         matcher.header_entry = header_entry;
847         matcher.header_addr_combo = header_addr_combo;
848         matcher.header_addr_entry = header_addr_entry;
849         matcher.string_entry = string_entry;
850         matcher.numeric_entry = numeric_entry;
851         matcher.numeric_label = numeric_label;
852         matcher.addressbook_folder_combo = addressbook_folder_combo;
853         matcher.match_combo = match_combo;
854         matcher.case_checkbtn = case_checkbtn;
855         matcher.regexp_checkbtn = regexp_checkbtn;
856         matcher.bool_op_combo = bool_op_combo;
857         matcher.test_btn = test_btn;
858 #ifndef USE_ALT_ADDRBOOK
859         matcher.addressbook_select_btn = addressbook_select_btn;
860 #endif
861 #if !GTK_CHECK_VERSION(3, 0, 0)
862         matcher.color_optmenu = color_optmenu;
863 #endif
864         matcher.match_label = match_label;
865         matcher.criteria_label2 = criteria_label2;
866         matcher.headers_combo = headers_combo;
867         matcher.match_combo2 = match_combo2;
868         matcher.match_label2 = match_label2;
869         matcher.upper_filler = upper_filler;
870         matcher.lower_filler = lower_filler;
871         
872         matcher.cond_list_view = cond_list_view;
873
874         matcher.selected_criteria = -1;
875         prefs_matcher_criteria_select(criteria_combo, NULL);
876 }
877
878 /*!
879  *\brief        Set the contents of a row
880  *
881  *\param        row Index of row to set
882  *\param        prop Condition to set
883  *
884  *\return       gint Row index \a prop has been added
885  */
886 static void prefs_matcher_list_view_set_row(GtkTreeIter *row, MatcherProp *prop)
887 {
888         gchar *matcher_str;
889
890         if (prop == NULL) {
891                 prefs_matcher_list_view_insert_matcher(matcher.cond_list_view,
892                                                        NULL, _("(New)"), FALSE);
893                 return;                                                
894         }
895
896         matcher_str = matcherprop_to_string(prop);
897         if (!row)
898                 prefs_matcher_list_view_insert_matcher(matcher.cond_list_view,
899                                                        NULL, matcher_str,
900                                                        TRUE);
901         else
902                 prefs_matcher_list_view_insert_matcher(matcher.cond_list_view,
903                                                        row, matcher_str, 
904                                                        TRUE);
905         g_free(matcher_str);
906 }
907
908 static gboolean match_combo2_model_set(void)
909 {
910         GtkTreeModel *model = gtk_combo_box_get_model(GTK_COMBO_BOX(matcher.match_combo2));
911         if (model == matcher.model_age_units ||
912             model == matcher.model_found ||
913             model == matcher.model_partial ||
914             model == matcher.model_phrase ||
915             model == matcher.model_set ||
916             model == matcher.model_size_units ||
917             model == matcher.model_tags ||
918             model == matcher.model_thread)
919                 return TRUE;
920         else
921                 debug_print("match_combo2 model unset.\n");
922
923         return FALSE;
924 }
925
926 static gboolean match_combo_model_set(void)
927 {
928         GtkTreeModel *model = gtk_combo_box_get_model(GTK_COMBO_BOX(matcher.match_combo));
929         if (model == matcher.model_age ||
930             model == matcher.model_contain ||
931             model == matcher.model_flags ||
932             model == matcher.model_score ||
933             model == matcher.model_size ||
934             model == matcher.model_test)
935                 return TRUE;
936         else
937                 debug_print("match_combo model unset.\n");
938
939         return FALSE;
940 }
941
942 /*!
943  *\brief        Clears a condition in the list widget
944  */
945 static void prefs_matcher_reset_condition(void)
946 {
947         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo), MATCH_ALL);
948         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo2), 0);
949         if (match_combo_model_set())
950                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo), 0);
951         if (match_combo2_model_set())
952                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2), 0);
953 #if !GTK_CHECK_VERSION(3, 0, 0)
954         gtk_cmoption_menu_set_history(GTK_CMOPTION_MENU(matcher.color_optmenu), 0);
955 #endif
956         gtk_spin_button_set_value(GTK_SPIN_BUTTON(matcher.numeric_entry), 0);
957         gtk_entry_set_text(GTK_ENTRY(matcher.header_entry), "");
958         gtk_entry_set_text(GTK_ENTRY(matcher.header_addr_entry), "");
959         gtk_entry_set_text(GTK_ENTRY(matcher.string_entry), "");
960         gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN((matcher.addressbook_folder_combo)))), "");
961         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), FALSE);
962         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), FALSE);
963 }
964
965 /*!
966  *\brief        Initializes dialog with a set of conditions
967  *
968  *\param        matchers List of conditions
969  */
970 static void prefs_matcher_set_dialog(MatcherList *matchers)
971 {
972         GSList *cur;
973         gboolean bool_op = 1;
974         GtkListStore *store = GTK_LIST_STORE(gtk_tree_view_get_model
975                                 (GTK_TREE_VIEW(matcher.cond_list_view)));
976
977         gtk_list_store_clear(store);                            
978
979         prefs_matcher_list_view_set_row(NULL, NULL);
980         if (matchers != NULL) {
981                 for (cur = matchers->matchers; cur != NULL;
982                      cur = g_slist_next(cur)) {
983                         MatcherProp *prop;
984                         prop = (MatcherProp *) cur->data;
985                         prefs_matcher_list_view_set_row(NULL, prop);
986                 }
987
988                 bool_op = matchers->bool_and;
989         }
990         
991         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.bool_op_combo), bool_op);
992
993         prefs_matcher_reset_condition();
994         
995         combobox_set_sensitive(GTK_COMBO_BOX(matcher.criteria_combo), MATCH_TAGS,
996                         (tags_get_size() > 0) ? TRUE : FALSE);
997 }
998
999 /*!
1000  *\brief        Converts current conditions in list box in
1001  *              a matcher list used by the matcher.
1002  *
1003  *\return       MatcherList * List of conditions.
1004  */
1005 static MatcherList *prefs_matcher_get_list(void)
1006 {
1007         gchar *matcher_str;
1008         MatcherProp *prop;
1009         gboolean bool_and;
1010         GSList *matcher_list;
1011         MatcherList *matchers;
1012         GtkTreeModel *model;
1013         GtkTreeIter iter;
1014
1015         model = gtk_tree_view_get_model(GTK_TREE_VIEW(matcher.cond_list_view));
1016         if (!gtk_tree_model_get_iter_first(model, &iter))
1017                 return NULL;
1018
1019         matcher_list = NULL;
1020
1021         do {
1022                 gboolean is_valid;
1023         
1024                 gtk_tree_model_get(model, &iter,
1025                                    PREFS_MATCHER_COND, &matcher_str,
1026                                    PREFS_MATCHER_COND_VALID, &is_valid,
1027                                    -1);
1028                 
1029                 if (is_valid) {
1030                         /* tmp = matcher_str; */
1031                         prop = matcher_parser_get_prop(matcher_str);
1032                         if (prop == NULL) {
1033                                 g_free(matcher_str);
1034                                 break;
1035                         }
1036                         
1037                         matcher_list = g_slist_append(matcher_list, prop);
1038                 }
1039                 g_free(matcher_str);
1040         } while (gtk_tree_model_iter_next(model, &iter));
1041
1042         bool_and = gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.bool_op_combo));
1043
1044         matchers = matcherlist_new(matcher_list, bool_and);
1045
1046         return matchers;
1047 }
1048
1049 /*!
1050  *\brief        Maps a keyword id (see #get_matchparser_tab_id) to a 
1051  *              criteria type (see first parameter of #matcherprop_new
1052  *              or #matcherprop_unquote_new)
1053  *
1054  *\param        matching_id Id returned by the matcher parser.
1055  *
1056  *\return       gint One of the CRITERIA_xxx constants.
1057  */
1058 static gint prefs_matcher_get_criteria_from_matching(gint matching_id)
1059 {
1060         switch(matching_id) {
1061         case MATCHCRITERIA_ALL:
1062                 return CRITERIA_ALL;
1063         case MATCHCRITERIA_NOT_UNREAD:
1064         case MATCHCRITERIA_UNREAD:
1065                 return CRITERIA_UNREAD;
1066         case MATCHCRITERIA_NOT_NEW:
1067         case MATCHCRITERIA_NEW:
1068                 return CRITERIA_NEW;
1069         case MATCHCRITERIA_NOT_MARKED:
1070         case MATCHCRITERIA_MARKED:
1071                 return CRITERIA_MARKED;
1072         case MATCHCRITERIA_NOT_DELETED:
1073         case MATCHCRITERIA_DELETED:
1074                 return CRITERIA_DELETED;
1075         case MATCHCRITERIA_NOT_REPLIED:
1076         case MATCHCRITERIA_REPLIED:
1077                 return CRITERIA_REPLIED;
1078         case MATCHCRITERIA_NOT_FORWARDED:
1079         case MATCHCRITERIA_FORWARDED:
1080                 return CRITERIA_FORWARDED;
1081         case MATCHCRITERIA_LOCKED:
1082         case MATCHCRITERIA_NOT_LOCKED:
1083                 return CRITERIA_LOCKED;
1084         case MATCHCRITERIA_NOT_SPAM:
1085         case MATCHCRITERIA_SPAM:
1086                 return CRITERIA_SPAM;
1087         case MATCHCRITERIA_HAS_ATTACHMENT:
1088         case MATCHCRITERIA_HAS_NO_ATTACHMENT:
1089                 return CRITERIA_HAS_ATTACHMENT;
1090         case MATCHCRITERIA_SIGNED:
1091         case MATCHCRITERIA_NOT_SIGNED:
1092                 return CRITERIA_SIGNED;
1093         case MATCHCRITERIA_PARTIAL:
1094         case MATCHCRITERIA_NOT_PARTIAL:
1095                 return CRITERIA_PARTIAL;
1096 #if !GTK_CHECK_VERSION(3, 0, 0)
1097         case MATCHCRITERIA_COLORLABEL:
1098         case MATCHCRITERIA_NOT_COLORLABEL:
1099                 return CRITERIA_COLORLABEL;
1100 #endif
1101         case MATCHCRITERIA_IGNORE_THREAD:
1102         case MATCHCRITERIA_NOT_IGNORE_THREAD:
1103                 return CRITERIA_IGNORE_THREAD;
1104         case MATCHCRITERIA_WATCH_THREAD:
1105         case MATCHCRITERIA_NOT_WATCH_THREAD:
1106                 return CRITERIA_WATCH_THREAD;
1107         case MATCHCRITERIA_NOT_SUBJECT:
1108         case MATCHCRITERIA_SUBJECT:
1109                 return CRITERIA_SUBJECT;
1110         case MATCHCRITERIA_NOT_FROM:
1111         case MATCHCRITERIA_FROM:
1112                 return CRITERIA_FROM;
1113         case MATCHCRITERIA_NOT_TO:
1114         case MATCHCRITERIA_TO:
1115                 return CRITERIA_TO;
1116         case MATCHCRITERIA_NOT_CC:
1117         case MATCHCRITERIA_CC:
1118                 return CRITERIA_CC;
1119         case MATCHCRITERIA_NOT_NEWSGROUPS:
1120         case MATCHCRITERIA_NEWSGROUPS:
1121                 return CRITERIA_NEWSGROUPS;
1122         case MATCHCRITERIA_NOT_MESSAGEID:
1123         case MATCHCRITERIA_MESSAGEID:
1124                 return CRITERIA_MESSAGEID;
1125         case MATCHCRITERIA_NOT_INREPLYTO:
1126         case MATCHCRITERIA_INREPLYTO:
1127                 return CRITERIA_INREPLYTO;
1128         case MATCHCRITERIA_NOT_REFERENCES:
1129         case MATCHCRITERIA_REFERENCES:
1130                 return CRITERIA_REFERENCES;
1131         case MATCHCRITERIA_NOT_TO_AND_NOT_CC:
1132         case MATCHCRITERIA_TO_OR_CC:
1133                 return CRITERIA_TO_OR_CC;
1134         case MATCHCRITERIA_NOT_TAG:
1135         case MATCHCRITERIA_TAG:
1136                 return CRITERIA_TAG;
1137         case MATCHCRITERIA_NOT_TAGGED:
1138         case MATCHCRITERIA_TAGGED:
1139                 return CRITERIA_TAGGED;
1140         case MATCHCRITERIA_NOT_BODY_PART:
1141         case MATCHCRITERIA_BODY_PART:
1142                 return CRITERIA_BODY_PART;
1143         case MATCHCRITERIA_NOT_MESSAGE:
1144         case MATCHCRITERIA_MESSAGE:
1145                 return CRITERIA_MESSAGE;
1146         case MATCHCRITERIA_NOT_HEADERS_PART:
1147         case MATCHCRITERIA_HEADERS_PART:
1148                 return CRITERIA_HEADERS_PART;
1149         case MATCHCRITERIA_NOT_HEADERS_CONT:
1150         case MATCHCRITERIA_HEADERS_CONT:
1151                 return CRITERIA_HEADERS_CONT;
1152         case MATCHCRITERIA_NOT_HEADER:
1153         case MATCHCRITERIA_HEADER:
1154                 return CRITERIA_HEADER;
1155         case MATCHCRITERIA_AGE_GREATER_HOURS:
1156                 return CRITERIA_AGE_GREATER_HOURS;
1157         case MATCHCRITERIA_AGE_LOWER_HOURS:
1158                 return CRITERIA_AGE_LOWER_HOURS;
1159         case MATCHCRITERIA_AGE_GREATER:
1160                 return CRITERIA_AGE_GREATER;
1161         case MATCHCRITERIA_AGE_LOWER:
1162                 return CRITERIA_AGE_LOWER;
1163         case MATCHCRITERIA_SCORE_GREATER:
1164                 return CRITERIA_SCORE_GREATER;
1165         case MATCHCRITERIA_SCORE_LOWER:
1166                 return CRITERIA_SCORE_LOWER;
1167         case MATCHCRITERIA_SCORE_EQUAL:
1168                 return CRITERIA_SCORE_EQUAL;
1169         case MATCHCRITERIA_NOT_TEST:
1170         case MATCHCRITERIA_TEST:
1171                 return CRITERIA_TEST;
1172         case MATCHCRITERIA_SIZE_GREATER:
1173                 return CRITERIA_SIZE_GREATER;
1174         case MATCHCRITERIA_SIZE_SMALLER:
1175                 return CRITERIA_SIZE_SMALLER;
1176         case MATCHCRITERIA_SIZE_EQUAL:
1177                 return CRITERIA_SIZE_EQUAL;
1178         case MATCHCRITERIA_FOUND_IN_ADDRESSBOOK:
1179         case MATCHCRITERIA_NOT_FOUND_IN_ADDRESSBOOK:
1180                 return CRITERIA_FOUND_IN_ADDRESSBOOK;
1181         default:
1182                 return -1;
1183         }
1184 }
1185
1186 /*!
1187  *\brief        Returns the matcher keyword id from a criteria id
1188  *
1189  *\param        criteria_id Criteria id (should not be the negate
1190  *              one)
1191  *
1192  *\return       gint A matcher keyword id. See #get_matchparser_tab_id.
1193  */
1194 static gint prefs_matcher_get_matching_from_criteria(gint criteria_id)
1195 {
1196         switch (criteria_id) {
1197         case CRITERIA_ALL:
1198                 return MATCHCRITERIA_ALL;
1199         case CRITERIA_UNREAD:
1200                 return MATCHCRITERIA_UNREAD;
1201         case CRITERIA_NEW:
1202                 return MATCHCRITERIA_NEW;
1203         case CRITERIA_MARKED:
1204                 return MATCHCRITERIA_MARKED;
1205         case CRITERIA_DELETED:
1206                 return MATCHCRITERIA_DELETED;
1207         case CRITERIA_REPLIED:
1208                 return MATCHCRITERIA_REPLIED;
1209         case CRITERIA_FORWARDED:
1210                 return MATCHCRITERIA_FORWARDED;
1211         case CRITERIA_LOCKED:
1212                 return MATCHCRITERIA_LOCKED;
1213         case CRITERIA_SPAM:
1214                 return MATCHCRITERIA_SPAM;
1215         case CRITERIA_HAS_ATTACHMENT:
1216                 return MATCHCRITERIA_HAS_ATTACHMENT;
1217         case CRITERIA_SIGNED:
1218                 return MATCHCRITERIA_SIGNED;
1219         case CRITERIA_PARTIAL:
1220                 return MATCHCRITERIA_PARTIAL;
1221 #if !GTK_CHECK_VERSION(3, 0, 0)
1222         case CRITERIA_COLORLABEL:
1223                 return MATCHCRITERIA_COLORLABEL;
1224 #endif
1225         case CRITERIA_IGNORE_THREAD:
1226                 return MATCHCRITERIA_IGNORE_THREAD;
1227         case CRITERIA_WATCH_THREAD:
1228                 return MATCHCRITERIA_WATCH_THREAD;
1229         case CRITERIA_SUBJECT:
1230                 return MATCHCRITERIA_SUBJECT;
1231         case CRITERIA_FROM:
1232                 return MATCHCRITERIA_FROM;
1233         case CRITERIA_TO:
1234                 return MATCHCRITERIA_TO;
1235         case CRITERIA_CC:
1236                 return MATCHCRITERIA_CC;
1237         case CRITERIA_TO_OR_CC:
1238                 return MATCHCRITERIA_TO_OR_CC;
1239         case CRITERIA_TAG:
1240                 return MATCHCRITERIA_TAG;
1241         case CRITERIA_TAGGED:
1242                 return MATCHCRITERIA_TAGGED;
1243         case CRITERIA_NEWSGROUPS:
1244                 return MATCHCRITERIA_NEWSGROUPS;
1245         case CRITERIA_MESSAGEID:
1246                 return MATCHCRITERIA_MESSAGEID;
1247         case CRITERIA_INREPLYTO:
1248                 return MATCHCRITERIA_INREPLYTO;
1249         case CRITERIA_REFERENCES:
1250                 return MATCHCRITERIA_REFERENCES;
1251         case CRITERIA_AGE_GREATER:
1252                 return MATCHCRITERIA_AGE_GREATER;
1253         case CRITERIA_AGE_LOWER:
1254                 return MATCHCRITERIA_AGE_LOWER;
1255         case CRITERIA_AGE_GREATER_HOURS:
1256                 return MATCHCRITERIA_AGE_GREATER_HOURS;
1257         case CRITERIA_AGE_LOWER_HOURS:
1258                 return MATCHCRITERIA_AGE_LOWER_HOURS;
1259         case CRITERIA_SCORE_GREATER:
1260                 return MATCHCRITERIA_SCORE_GREATER;
1261         case CRITERIA_SCORE_LOWER:
1262                 return MATCHCRITERIA_SCORE_LOWER;
1263         case CRITERIA_SCORE_EQUAL:
1264                 return MATCHCRITERIA_SCORE_EQUAL;
1265         case CRITERIA_HEADER:
1266                 return MATCHCRITERIA_HEADER;
1267         case CRITERIA_HEADERS_PART:
1268                 return MATCHCRITERIA_HEADERS_PART;
1269         case CRITERIA_HEADERS_CONT:
1270                 return MATCHCRITERIA_HEADERS_CONT;
1271         case CRITERIA_BODY_PART:
1272                 return MATCHCRITERIA_BODY_PART;
1273         case CRITERIA_MESSAGE:
1274                 return MATCHCRITERIA_MESSAGE;
1275         case CRITERIA_TEST:
1276                 return MATCHCRITERIA_TEST;
1277         case CRITERIA_SIZE_GREATER:
1278                 return MATCHCRITERIA_SIZE_GREATER;
1279         case CRITERIA_SIZE_SMALLER:
1280                 return MATCHCRITERIA_SIZE_SMALLER;
1281         case CRITERIA_SIZE_EQUAL:
1282                 return MATCHCRITERIA_SIZE_EQUAL;
1283         case CRITERIA_FOUND_IN_ADDRESSBOOK:
1284                 return MATCHCRITERIA_FOUND_IN_ADDRESSBOOK;
1285         default:
1286                 return -1;
1287         }
1288 }
1289
1290 /*!
1291  *\brief        Returns the negate matcher keyword id from a matcher keyword
1292  *              id.
1293  *
1294  *\param        matcher_criteria Matcher keyword id. 
1295  *
1296  *\return       gint A matcher keyword id. See #get_matchparser_tab_id.
1297  */
1298 static gint prefs_matcher_not_criteria(gint matcher_criteria)
1299 {
1300         switch(matcher_criteria) {
1301         case MATCHCRITERIA_UNREAD:
1302                 return MATCHCRITERIA_NOT_UNREAD;
1303         case MATCHCRITERIA_NEW:
1304                 return MATCHCRITERIA_NOT_NEW;
1305         case MATCHCRITERIA_MARKED:
1306                 return MATCHCRITERIA_NOT_MARKED;
1307         case MATCHCRITERIA_DELETED:
1308                 return MATCHCRITERIA_NOT_DELETED;
1309         case MATCHCRITERIA_REPLIED:
1310                 return MATCHCRITERIA_NOT_REPLIED;
1311         case MATCHCRITERIA_FORWARDED:
1312                 return MATCHCRITERIA_NOT_FORWARDED;
1313         case MATCHCRITERIA_LOCKED:
1314                 return MATCHCRITERIA_NOT_LOCKED;
1315         case MATCHCRITERIA_SPAM:
1316                 return MATCHCRITERIA_NOT_SPAM;
1317         case MATCHCRITERIA_HAS_ATTACHMENT:
1318                 return MATCHCRITERIA_HAS_NO_ATTACHMENT;
1319         case MATCHCRITERIA_SIGNED:
1320                 return MATCHCRITERIA_NOT_SIGNED;
1321         case MATCHCRITERIA_PARTIAL:
1322                 return MATCHCRITERIA_NOT_PARTIAL;
1323 #if !GTK_CHECK_VERSION(3, 0, 0)
1324         case MATCHCRITERIA_COLORLABEL:
1325                 return MATCHCRITERIA_NOT_COLORLABEL;
1326 #endif
1327         case MATCHCRITERIA_IGNORE_THREAD:
1328                 return MATCHCRITERIA_NOT_IGNORE_THREAD;
1329         case MATCHCRITERIA_WATCH_THREAD:
1330                 return MATCHCRITERIA_NOT_WATCH_THREAD;
1331         case MATCHCRITERIA_SUBJECT:
1332                 return MATCHCRITERIA_NOT_SUBJECT;
1333         case MATCHCRITERIA_FROM:
1334                 return MATCHCRITERIA_NOT_FROM;
1335         case MATCHCRITERIA_TO:
1336                 return MATCHCRITERIA_NOT_TO;
1337         case MATCHCRITERIA_CC:
1338                 return MATCHCRITERIA_NOT_CC;
1339         case MATCHCRITERIA_TO_OR_CC:
1340                 return MATCHCRITERIA_NOT_TO_AND_NOT_CC;
1341         case MATCHCRITERIA_TAG:
1342                 return MATCHCRITERIA_NOT_TAG;
1343         case MATCHCRITERIA_TAGGED:
1344                 return MATCHCRITERIA_NOT_TAGGED;
1345         case MATCHCRITERIA_NEWSGROUPS:
1346                 return MATCHCRITERIA_NOT_NEWSGROUPS;
1347         case MATCHCRITERIA_MESSAGEID:
1348                 return MATCHCRITERIA_NOT_MESSAGEID;
1349         case MATCHCRITERIA_INREPLYTO:
1350                 return MATCHCRITERIA_NOT_INREPLYTO;
1351         case MATCHCRITERIA_REFERENCES:
1352                 return MATCHCRITERIA_NOT_REFERENCES;
1353         case MATCHCRITERIA_HEADER:
1354                 return MATCHCRITERIA_NOT_HEADER;
1355         case MATCHCRITERIA_HEADERS_PART:
1356                 return MATCHCRITERIA_NOT_HEADERS_PART;
1357         case MATCHCRITERIA_HEADERS_CONT:
1358                 return MATCHCRITERIA_NOT_HEADERS_CONT;
1359         case MATCHCRITERIA_MESSAGE:
1360                 return MATCHCRITERIA_NOT_MESSAGE;
1361         case MATCHCRITERIA_TEST:
1362                 return MATCHCRITERIA_NOT_TEST;
1363         case MATCHCRITERIA_BODY_PART:
1364                 return MATCHCRITERIA_NOT_BODY_PART;
1365         case MATCHCRITERIA_FOUND_IN_ADDRESSBOOK:
1366                 return MATCHCRITERIA_NOT_FOUND_IN_ADDRESSBOOK;
1367         default:
1368                 return matcher_criteria;
1369         }
1370 }
1371
1372 static gint prefs_matcher_get_criteria(void)
1373 {
1374         gint match_criteria = gtk_combo_box_get_active(GTK_COMBO_BOX(
1375                                         matcher.criteria_combo));
1376         const gchar *header = NULL;
1377           
1378         switch (match_criteria) {
1379         case MATCH_ABOOK:
1380                 return CRITERIA_FOUND_IN_ADDRESSBOOK;   
1381         case MATCH_ALL:
1382                 return CRITERIA_ALL;
1383         case MATCH_AGE:
1384         case MATCH_SCORE:
1385         case MATCH_SIZE:
1386         case MATCH_FLAG:
1387                 return combobox_get_active_data(GTK_COMBO_BOX(
1388                                         matcher.match_combo));
1389         case MATCH_HEADER:
1390                 header = gtk_entry_get_text(GTK_ENTRY(matcher.header_entry));
1391                 return header_name_to_crit(header);
1392 #if !GTK_CHECK_VERSION(3, 0, 0)
1393         case MATCH_LABEL:
1394                 return CRITERIA_COLORLABEL;
1395 #endif
1396         case MATCH_PARTIAL:
1397                 return CRITERIA_PARTIAL;
1398         case MATCH_TEST:
1399                 return CRITERIA_TEST;
1400         case MATCH_PHRASE:
1401         case MATCH_TAGS:
1402         case MATCH_THREAD:
1403                 return combobox_get_active_data(GTK_COMBO_BOX(
1404                                         matcher.criteria_combo2));
1405         }
1406         
1407         return -1;
1408 }
1409
1410 static gint prefs_matcher_get_pred(const gint criteria)
1411 {
1412         switch(criteria) {
1413         case CRITERIA_SUBJECT:
1414         case CRITERIA_FROM:
1415         case CRITERIA_TO:
1416         case CRITERIA_CC:
1417         case CRITERIA_TO_OR_CC:
1418         case CRITERIA_NEWSGROUPS:
1419         case CRITERIA_MESSAGEID:
1420         case CRITERIA_INREPLYTO:
1421         case CRITERIA_REFERENCES:
1422         case CRITERIA_HEADER:
1423         case CRITERIA_HEADERS_PART:
1424         case CRITERIA_HEADERS_CONT:
1425         case CRITERIA_BODY_PART:
1426         case CRITERIA_MESSAGE:
1427         case CRITERIA_TAG:
1428         case CRITERIA_TAGGED:
1429         case CRITERIA_TEST:
1430                 return gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.match_combo));
1431         case CRITERIA_FOUND_IN_ADDRESSBOOK:
1432         case CRITERIA_UNREAD:
1433         case CRITERIA_NEW:
1434         case CRITERIA_MARKED:
1435         case CRITERIA_DELETED:
1436         case CRITERIA_REPLIED:
1437         case CRITERIA_FORWARDED:
1438         case CRITERIA_LOCKED:
1439         case CRITERIA_SPAM:
1440         case CRITERIA_HAS_ATTACHMENT:
1441         case CRITERIA_SIGNED:
1442         case CRITERIA_COLORLABEL:
1443                 return gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.match_combo2));
1444         case CRITERIA_WATCH_THREAD:
1445                 return gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.criteria_combo2)) - 2;
1446         case CRITERIA_IGNORE_THREAD:
1447         case CRITERIA_PARTIAL:
1448                 return gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.criteria_combo2));
1449         }
1450         
1451         return 0;
1452 }
1453
1454 /*!
1455  *\brief        Converts the text in the selected row to a 
1456  *              matcher structure
1457  *
1458  *\return       MatcherProp * Newly allocated matcher structure.
1459  */
1460 static MatcherProp *prefs_matcher_dialog_to_matcher(void)
1461 {
1462         MatcherProp *matcherprop;
1463         gint criteria;
1464         gint matchtype;
1465         gint value_pred;
1466         gint value_criteria = prefs_matcher_get_criteria();
1467         gboolean use_regexp;
1468         gboolean case_sensitive;
1469         const gchar *header;
1470         const gchar *expr;
1471         gint value, sel;
1472
1473         if (value_criteria == -1)
1474                 return NULL;
1475
1476         use_regexp = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn));
1477         case_sensitive = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn));
1478
1479         if (use_regexp) {
1480                 if (case_sensitive)
1481                         matchtype = MATCHTYPE_REGEXP;
1482                 else
1483                         matchtype = MATCHTYPE_REGEXPCASE;
1484         }
1485         else {
1486                 if (case_sensitive)
1487                         matchtype = MATCHTYPE_MATCH;
1488                 else
1489                         matchtype = MATCHTYPE_MATCHCASE;
1490         }
1491
1492         header = NULL;
1493         expr = NULL;
1494         value = 0;
1495
1496         switch (value_criteria) {
1497         case CRITERIA_ALL:
1498         case CRITERIA_UNREAD:
1499         case CRITERIA_NEW:
1500         case CRITERIA_MARKED:
1501         case CRITERIA_DELETED:
1502         case CRITERIA_REPLIED:
1503         case CRITERIA_FORWARDED:
1504         case CRITERIA_LOCKED:
1505         case CRITERIA_SPAM:
1506         case CRITERIA_HAS_ATTACHMENT:
1507         case CRITERIA_SIGNED:
1508         case CRITERIA_PARTIAL:
1509         case CRITERIA_IGNORE_THREAD:
1510         case CRITERIA_WATCH_THREAD:
1511         case CRITERIA_TAGGED:
1512                 break;
1513
1514         case CRITERIA_SUBJECT:
1515         case CRITERIA_FROM:
1516         case CRITERIA_TO:
1517         case CRITERIA_CC:
1518         case CRITERIA_TO_OR_CC:
1519         case CRITERIA_TAG:
1520         case CRITERIA_NEWSGROUPS:
1521         case CRITERIA_MESSAGEID:
1522         case CRITERIA_INREPLYTO:
1523         case CRITERIA_REFERENCES:
1524         case CRITERIA_HEADERS_PART:
1525         case CRITERIA_HEADERS_CONT:
1526         case CRITERIA_BODY_PART:
1527         case CRITERIA_MESSAGE:
1528                 expr = gtk_entry_get_text(GTK_ENTRY(matcher.string_entry));
1529                 
1530                 if(*expr == '\0') {
1531                         alertpanel_error(_("Search pattern is not set."));
1532                         return NULL;
1533                 }
1534                 break;
1535
1536         case CRITERIA_TEST:
1537                 expr = gtk_entry_get_text(GTK_ENTRY(matcher.string_entry));
1538                 
1539                 if(*expr == '\0') {
1540                         alertpanel_error(_("Test command is not set."));
1541                         return NULL;
1542                 }
1543                 break;
1544
1545         case CRITERIA_AGE_GREATER:
1546         case CRITERIA_AGE_LOWER:
1547                 value = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(
1548                                                          matcher.numeric_entry));
1549                 sel = gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.match_combo2));
1550                 if(sel == AGE_WEEKS)
1551                         value *= 7;
1552                 else if (sel == AGE_HOURS) {
1553                         if (value_criteria == CRITERIA_AGE_GREATER)
1554                                 value_criteria = CRITERIA_AGE_GREATER_HOURS;
1555                         else 
1556                                 value_criteria = CRITERIA_AGE_LOWER_HOURS;
1557                 }
1558                 break;
1559                         
1560         case CRITERIA_SCORE_GREATER:
1561         case CRITERIA_SCORE_LOWER:
1562         case CRITERIA_SCORE_EQUAL:
1563                 value = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(
1564                                                          matcher.numeric_entry));
1565                 break;
1566                                                          
1567         case CRITERIA_SIZE_GREATER:
1568         case CRITERIA_SIZE_SMALLER:
1569         case CRITERIA_SIZE_EQUAL:
1570                 value = gtk_spin_button_get_value_as_int(GTK_SPIN_BUTTON(
1571                                                          matcher.numeric_entry));
1572                 sel = gtk_combo_box_get_active(GTK_COMBO_BOX(matcher.match_combo2));
1573                 if(sel == SIZE_UNIT_MBYTES)
1574                         value *= MB_SIZE;
1575                 if(sel == SIZE_UNIT_KBYTES)
1576                         value *= KB_SIZE;
1577                 break;
1578                 
1579 #if !GTK_CHECK_VERSION(3, 0, 0)
1580         case CRITERIA_COLORLABEL:
1581                 value = colorlabel_get_color_menu_active_item
1582                         (gtk_cmoption_menu_get_menu(GTK_CMOPTION_MENU
1583                                 (matcher.color_optmenu))); 
1584                 break;
1585 #endif
1586
1587         case CRITERIA_HEADER:
1588                 header = gtk_entry_get_text(GTK_ENTRY(matcher.header_entry));
1589                 expr = gtk_entry_get_text(GTK_ENTRY(matcher.string_entry));
1590
1591                 if (*header == '\0') {
1592                     alertpanel_error(_("Header name is not set."));
1593                     return NULL;
1594                 }
1595                 
1596                 if(*expr == '\0') {
1597                         alertpanel_error(_("Search pattern is not set."));
1598                         return NULL;
1599                 } 
1600                 break;
1601
1602         case CRITERIA_FOUND_IN_ADDRESSBOOK:
1603                 header = gtk_entry_get_text(GTK_ENTRY(matcher.header_addr_entry));
1604                 expr = gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN((matcher.addressbook_folder_combo)))));
1605
1606                 if (*header == '\0') {
1607                     alertpanel_error(_("Header name is not set."));
1608                     return NULL;
1609                 }
1610                 if (*expr == '\0') {
1611                         gchar *tmp;
1612
1613                         if (g_utf8_collate(header, C_("Filtering Matcher Menu", "All")) == 0)
1614                                 tmp = g_strdup(_("all addresses in all headers"));
1615                         else
1616                         if (g_utf8_collate(header, _("Any")) == 0)
1617                                 tmp = g_strdup(_("any address in any header"));
1618                         else
1619                                 tmp = g_strdup_printf(_("the address(es) in header '%s'"), header);
1620                         alertpanel_error(_("Book/folder path is not set.\n\n"
1621                                                 "If you want to match %s against the whole address book, "
1622                                                 "you have to select '%s' from the book/folder drop-down list."),
1623                                                 tmp, _("Any"));
1624                         g_free(tmp);
1625                     return NULL;
1626                 }
1627                 /* store UNtranslated "Any"/"All" in matcher expressions */
1628                 if (g_utf8_collate(header, C_("Filtering Matcher Menu", "All")) == 0)
1629                         header = "All";
1630                 else
1631                         if (g_utf8_collate(header, _("Any")) == 0)
1632                                 header = "Any";
1633                 if (g_utf8_collate(expr, _("Any")) == 0)
1634                         expr = "Any";
1635                 break;
1636         }
1637
1638         criteria = prefs_matcher_get_matching_from_criteria(value_criteria);
1639
1640         value_pred = prefs_matcher_get_pred(value_criteria);
1641         if(value_pred)
1642                 criteria = prefs_matcher_not_criteria(criteria);
1643
1644         matcherprop = matcherprop_new(criteria, header, matchtype,
1645                                       expr, value);
1646
1647         return matcherprop;
1648 }
1649
1650 /*!
1651  *\brief        Signal handler for register button
1652  */
1653 static void prefs_matcher_register_cb(void)
1654 {
1655         MatcherProp *matcherprop;
1656         
1657         matcherprop = prefs_matcher_dialog_to_matcher();
1658         if (matcherprop == NULL)
1659                 return;
1660
1661         prefs_matcher_list_view_set_row(NULL, matcherprop);
1662
1663         matcherprop_free(matcherprop);
1664         
1665         prefs_matcher_reset_condition();
1666 }
1667
1668 /*!
1669  *\brief        Signal handler for substitute button
1670  */
1671 static void prefs_matcher_substitute_cb(void)
1672 {
1673         MatcherProp *matcherprop;
1674         GtkTreeIter row;
1675         GtkTreeSelection *selection;
1676         GtkTreeModel *model;
1677         gboolean is_valid;
1678
1679         selection = gtk_tree_view_get_selection
1680                         (GTK_TREE_VIEW(matcher.cond_list_view));
1681         
1682         if (!gtk_tree_selection_get_selected(selection, &model, &row))
1683                 return;
1684         
1685         gtk_tree_model_get(model, &row, 
1686                            PREFS_MATCHER_COND_VALID, &is_valid,
1687                            -1);
1688         if (!is_valid)
1689                 return;
1690
1691         matcherprop = prefs_matcher_dialog_to_matcher();
1692         if (matcherprop == NULL)
1693                 return;
1694
1695         prefs_matcher_list_view_set_row(&row, matcherprop);
1696
1697         matcherprop_free(matcherprop);
1698 }
1699
1700 /*!
1701  *\brief        Signal handler for delete button
1702  */
1703 static void prefs_matcher_delete_cb(void)
1704 {
1705         GtkTreeIter row;
1706         GtkTreeSelection *selection;
1707         GtkTreeModel *model;
1708         gboolean is_valid;
1709
1710         selection = gtk_tree_view_get_selection
1711                         (GTK_TREE_VIEW(matcher.cond_list_view));
1712         
1713         if (!gtk_tree_selection_get_selected(selection, &model, &row))
1714                 return;
1715                 
1716         gtk_tree_model_get(model, &row, 
1717                            PREFS_MATCHER_COND_VALID, &is_valid,
1718                            -1);
1719
1720         if (!is_valid)
1721                 return;
1722
1723         gtk_list_store_remove(GTK_LIST_STORE(model), &row);             
1724
1725         prefs_matcher_reset_condition();
1726 }
1727
1728 /*!
1729  *\brief        Signal handler for 'move up' button
1730  */
1731 static void prefs_matcher_up(void)
1732 {
1733         GtkTreePath *prev, *sel, *try;
1734         GtkTreeIter isel;
1735         GtkListStore *store = NULL;
1736         GtkTreeModel *model = NULL;
1737         GtkTreeIter iprev;
1738         
1739         if (!gtk_tree_selection_get_selected
1740                 (gtk_tree_view_get_selection
1741                         (GTK_TREE_VIEW(matcher.cond_list_view)),
1742                  &model,        
1743                  &isel))
1744                 return;
1745         store = (GtkListStore *)model;
1746         sel = gtk_tree_model_get_path(GTK_TREE_MODEL(store), &isel);
1747         if (!sel)
1748                 return;
1749         
1750         /* no move if we're at row 0 or 1, looks phony, but other
1751          * solutions are more convoluted... */
1752         try = gtk_tree_path_copy(sel);
1753         if (!gtk_tree_path_prev(try) || !gtk_tree_path_prev(try)) {
1754                 gtk_tree_path_free(try);
1755                 gtk_tree_path_free(sel);
1756                 return;
1757         }
1758         gtk_tree_path_free(try);
1759
1760         prev = gtk_tree_path_copy(sel);         
1761         if (gtk_tree_path_prev(prev)) {
1762                 gtk_tree_model_get_iter(GTK_TREE_MODEL(store),
1763                                         &iprev, prev);
1764                 gtk_list_store_swap(store, &iprev, &isel);
1765                 /* XXX: GTK2 select row?? */
1766         }
1767
1768         gtk_tree_path_free(sel);
1769         gtk_tree_path_free(prev);
1770 }
1771
1772 /*!
1773  *\brief        Signal handler for 'move down' button
1774  */
1775 static void prefs_matcher_down(void)
1776 {
1777         GtkListStore *store = NULL;
1778         GtkTreeModel *model = NULL;
1779         GtkTreeIter next, sel;
1780         GtkTreePath *try;
1781         
1782         if (!gtk_tree_selection_get_selected
1783                 (gtk_tree_view_get_selection
1784                         (GTK_TREE_VIEW(matcher.cond_list_view)),
1785                  &model,
1786                  &sel))
1787                 return;
1788         store = (GtkListStore *)model;
1789         try = gtk_tree_model_get_path(GTK_TREE_MODEL(store), &sel);
1790         if (!try) 
1791                 return;
1792         
1793         /* move when not at row 0 ... */
1794         if (gtk_tree_path_prev(try)) {
1795                 next = sel;
1796                 if (gtk_tree_model_iter_next(GTK_TREE_MODEL(store), &next))
1797                         gtk_list_store_swap(store, &next, &sel);
1798         }
1799                 
1800         gtk_tree_path_free(try);
1801 }
1802
1803 static void prefs_matcher_enable_widget(GtkWidget* widget, const gboolean enable)
1804 {
1805         cm_return_if_fail(widget != NULL);
1806
1807         if(enable == TRUE) {
1808                 gtk_widget_set_sensitive(widget, TRUE);
1809                 gtk_widget_show(widget);        
1810         } else {
1811                 gtk_widget_set_sensitive(widget, FALSE);
1812                 gtk_widget_hide(widget);
1813         }
1814 }
1815
1816 static void prefs_matcher_set_model(GtkWidget *widget, GtkTreeModel *model)
1817 {
1818         cm_return_if_fail(widget != NULL);
1819         cm_return_if_fail(model != NULL);
1820         
1821         gtk_combo_box_set_model(GTK_COMBO_BOX(widget), model);
1822         gtk_combo_box_set_active(GTK_COMBO_BOX(widget), 0);
1823 }
1824
1825 static void prefs_matcher_second_criteria_sel(GtkWidget *widget,
1826                                               gpointer user_data)
1827 {
1828         gint criteria = gtk_combo_box_get_active(GTK_COMBO_BOX(
1829                                                 matcher.criteria_combo));
1830         gint criteria2 = combobox_get_active_data(GTK_COMBO_BOX(
1831                                                 matcher.criteria_combo2));
1832         
1833         if(criteria != MATCH_PHRASE && criteria != MATCH_TAGS) return;
1834         
1835         if(criteria == MATCH_PHRASE) {
1836                 switch(criteria2) {
1837                 case CRITERIA_HEADERS_PART:
1838                         gtk_label_set_text(GTK_LABEL(matcher.match_label),
1839                                         _("Headers part"));
1840                         break;
1841                 case CRITERIA_HEADERS_CONT:
1842                         gtk_label_set_text(GTK_LABEL(matcher.match_label),
1843                                         _("Headers values"));
1844                         break;
1845                 case CRITERIA_BODY_PART:
1846                         gtk_label_set_text(GTK_LABEL(matcher.match_label),
1847                                         _("Body part"));
1848                         break;  
1849                 case CRITERIA_MESSAGE:
1850                         gtk_label_set_text(GTK_LABEL(matcher.match_label),
1851                                         _("Whole message"));
1852                         break;
1853                 }
1854         }
1855         
1856         if(criteria == MATCH_TAGS) {
1857                 if(criteria2 == CRITERIA_TAGGED) {
1858                         prefs_matcher_enable_widget(matcher.upper_filler, FALSE);
1859                         prefs_matcher_enable_widget(matcher.match_label2, TRUE);
1860                         prefs_matcher_enable_widget(matcher.string_entry, FALSE);
1861                         prefs_matcher_enable_widget(matcher.case_checkbtn, FALSE);
1862                         prefs_matcher_enable_widget(matcher.regexp_checkbtn, FALSE);
1863                 } else {
1864                         prefs_matcher_enable_widget(matcher.upper_filler, TRUE);
1865                         prefs_matcher_enable_widget(matcher.match_label2, FALSE);
1866                         prefs_matcher_enable_widget(matcher.string_entry, TRUE);
1867                         prefs_matcher_enable_widget(matcher.case_checkbtn, TRUE);
1868                         prefs_matcher_enable_widget(matcher.regexp_checkbtn, TRUE);
1869                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(
1870                                                 matcher.regexp_checkbtn), FALSE);
1871                         gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(
1872                                                 matcher.case_checkbtn), FALSE);
1873                 }
1874         }
1875 }
1876
1877 #define MATCH_COMBO_IS_ENABLED(x) (x != MATCH_ALL && x != MATCH_ABOOK && \
1878                 x != MATCH_PARTIAL && x != MATCH_THREAD && x != MATCH_LABEL) ? TRUE : FALSE
1879 #define MATCH_CASE_REGEXP(x) (x == MATCH_HEADER || x == MATCH_PHRASE) ? TRUE : FALSE 
1880 #define MATCH_NUMERIC(x) (x == MATCH_AGE || x == MATCH_SCORE || \
1881                           x == MATCH_SIZE) ? TRUE : FALSE
1882
1883 /*!
1884  *\brief        Change widgets depending on the selected condition
1885  *
1886  *\param        criteria combo widget
1887  *\param        user_data Not used      
1888  */
1889 static void prefs_matcher_criteria_select(GtkWidget *widget,
1890                                           gpointer user_data)
1891 {
1892         gint value, old_value;
1893
1894         old_value = matcher.selected_criteria;
1895         matcher.selected_criteria = value = gtk_combo_box_get_active
1896                 (GTK_COMBO_BOX(matcher.criteria_combo));
1897
1898         if (old_value == matcher.selected_criteria)
1899                 return;
1900
1901         prefs_matcher_enable_widget(matcher.criteria_label2,
1902                                     (value == MATCH_ABOOK   ||
1903                                      value == MATCH_PHRASE  ||
1904                                      value == MATCH_HEADER  ||
1905                                      value == MATCH_PARTIAL ||
1906                                      value == MATCH_TAGS    ||
1907                                      value == MATCH_THREAD));
1908         prefs_matcher_enable_widget(matcher.headers_combo,
1909                                     (value == MATCH_HEADER));
1910         prefs_matcher_enable_widget(matcher.criteria_combo2,
1911                                     (value == MATCH_PHRASE  ||
1912                                      value == MATCH_PARTIAL ||
1913                                      value == MATCH_TAGS    ||
1914                                      value == MATCH_THREAD));
1915         prefs_matcher_enable_widget(matcher.match_combo2,
1916                                     (value == MATCH_ABOOK ||
1917                                      value == MATCH_AGE   ||
1918                                      value == MATCH_FLAG  ||
1919                                      value == MATCH_LABEL ||
1920                                      value == MATCH_SIZE));
1921         prefs_matcher_enable_widget(matcher.match_label2,
1922                                     (value == MATCH_ABOOK ||
1923                                      value == MATCH_FLAG  ||
1924                                      value == MATCH_LABEL ||
1925                                      value == MATCH_TAGS));
1926         prefs_matcher_enable_widget(matcher.header_addr_combo,
1927                                     (value == MATCH_ABOOK));
1928         prefs_matcher_enable_widget(matcher.string_entry,
1929                                     (MATCH_CASE_REGEXP(value) ||
1930                                      value == MATCH_TEST));
1931         prefs_matcher_enable_widget(matcher.numeric_entry,
1932                                     MATCH_NUMERIC(value));
1933         prefs_matcher_enable_widget(matcher.numeric_label,
1934                                     (value == MATCH_SCORE));
1935         prefs_matcher_enable_widget(matcher.addressbook_folder_combo,
1936                                     (value == MATCH_ABOOK));
1937         prefs_matcher_enable_widget(matcher.match_combo,
1938                                     MATCH_COMBO_IS_ENABLED(value));
1939         prefs_matcher_enable_widget(matcher.case_checkbtn,
1940                                     MATCH_CASE_REGEXP(value));
1941         prefs_matcher_enable_widget(matcher.regexp_checkbtn,
1942                                     MATCH_CASE_REGEXP(value));
1943         prefs_matcher_enable_widget(matcher.test_btn,
1944                                     (value == MATCH_TEST));
1945         prefs_matcher_enable_widget(matcher.addressbook_select_btn,
1946                                     (value == MATCH_ABOOK));
1947 #if !GTK_CHECK_VERSION(3, 0, 0)
1948         prefs_matcher_enable_widget(matcher.color_optmenu,
1949                                     (value == MATCH_LABEL));
1950 #endif
1951         prefs_matcher_enable_widget(matcher.upper_filler,
1952                                     MATCH_CASE_REGEXP(value));
1953         prefs_matcher_enable_widget(matcher.lower_filler,
1954                                     (value == MATCH_ABOOK));
1955                                 
1956         gtk_label_set_text(GTK_LABEL(matcher.match_label), "");
1957         gtk_entry_set_text(GTK_ENTRY(matcher.string_entry), "");
1958
1959         switch(value) {
1960         case MATCH_ABOOK:
1961                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.header_addr_combo), 0);  
1962                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.addressbook_folder_combo), 0);
1963                 prefs_matcher_set_model(matcher.match_combo2, matcher.model_found);
1964                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("in"));
1965                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Header"));
1966                 gtk_label_set_text(GTK_LABEL(matcher.match_label2), _("content is"));
1967                 break;
1968         case MATCH_AGE:
1969                 prefs_matcher_set_model(matcher.match_combo, matcher.model_age);
1970                 prefs_matcher_set_model(matcher.match_combo2, matcher.model_age_units);
1971                 gtk_spin_button_set_range(GTK_SPIN_BUTTON(
1972                                   matcher.numeric_entry), 0, 10000);
1973                 gtk_spin_button_set_value(GTK_SPIN_BUTTON(matcher.numeric_entry), 0);
1974                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2), AGE_DAYS);
1975                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Age is"));
1976                 break;
1977         case MATCH_FLAG:
1978                 prefs_matcher_set_model(matcher.match_combo, matcher.model_flags);
1979                 prefs_matcher_set_model(matcher.match_combo2, matcher.model_set);
1980                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Flag"));
1981                 gtk_label_set_text(GTK_LABEL(matcher.match_label2), _("is"));
1982                 break;
1983         case MATCH_HEADER:
1984                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.headers_combo), 0);
1985                 prefs_matcher_set_model(matcher.match_combo, matcher.model_contain);
1986                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("Name:"));
1987                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Header"));
1988                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), FALSE);
1989                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), FALSE);
1990                 break;
1991 #if !GTK_CHECK_VERSION(3, 0, 0)
1992         case MATCH_LABEL:
1993                 gtk_cmoption_menu_set_history(GTK_CMOPTION_MENU(matcher.color_optmenu), 0);
1994                 prefs_matcher_set_model(matcher.match_combo2, matcher.model_set);
1995                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Label"));
1996                 gtk_label_set_text(GTK_LABEL(matcher.match_label2), _("is"));
1997                 break;
1998 #endif
1999         case MATCH_PARTIAL:
2000                 prefs_matcher_set_model(matcher.criteria_combo2, matcher.model_partial);
2001                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("Value:"));
2002                 break;
2003         case MATCH_PHRASE:
2004                 prefs_matcher_set_model(matcher.criteria_combo2, matcher.model_phrase);
2005                 prefs_matcher_set_model(matcher.match_combo, matcher.model_contain);
2006                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("in"));
2007                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), FALSE);
2008                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), FALSE);
2009                 prefs_matcher_second_criteria_sel(NULL, NULL);
2010                 break;  
2011         case MATCH_SCORE:
2012                 prefs_matcher_set_model(matcher.match_combo, matcher.model_score);
2013                 gtk_spin_button_set_range(GTK_SPIN_BUTTON(
2014                                           matcher.numeric_entry), -1000, 1000);
2015                 gtk_spin_button_set_value(GTK_SPIN_BUTTON(matcher.numeric_entry), 0);
2016                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Score is"));
2017                 gtk_label_set_text(GTK_LABEL(matcher.numeric_label), _("points"));
2018                 break;  
2019         case MATCH_SIZE:
2020                 prefs_matcher_set_model(matcher.match_combo, matcher.model_size);
2021                 prefs_matcher_set_model(matcher.match_combo2, matcher.model_size_units);
2022                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2023                                          SIZE_UNIT_KBYTES);
2024                 gtk_spin_button_set_range(GTK_SPIN_BUTTON(
2025                                           matcher.numeric_entry), 0, 100000);
2026                 gtk_spin_button_set_value(GTK_SPIN_BUTTON(matcher.numeric_entry), 0);
2027                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Size is"));
2028                 break;
2029         case MATCH_TAGS:
2030                 prefs_matcher_set_model(matcher.criteria_combo2, matcher.model_tags);
2031                 prefs_matcher_set_model(matcher.match_combo, matcher.model_contain);
2032                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("Scope:"));
2033                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Message"));
2034                 gtk_label_set_text(GTK_LABEL(matcher.match_label2), _("tags"));
2035                 prefs_matcher_second_criteria_sel(NULL, NULL);
2036                 break;
2037         case MATCH_THREAD:
2038                 prefs_matcher_set_model(matcher.criteria_combo2, matcher.model_thread);
2039                 gtk_label_set_text(GTK_LABEL(matcher.criteria_label2), _("type is"));
2040                 break;
2041         case MATCH_TEST:
2042                 prefs_matcher_set_model(matcher.match_combo, matcher.model_test);
2043                 gtk_label_set_text(GTK_LABEL(matcher.match_label), _("Program returns"));
2044                 break;
2045         }       
2046 }
2047
2048 /*!
2049  *\brief        Handle key press
2050  *
2051  *\param        widget Widget receiving key press
2052  *\param        event Key event
2053  *\param        data User data
2054  */
2055 static gboolean prefs_matcher_key_pressed(GtkWidget *widget, GdkEventKey *event,
2056                                      gpointer data)
2057 {
2058         if (event && event->keyval == GDK_KEY_Escape) {
2059                 prefs_matcher_cancel();
2060                 return TRUE;            
2061         }
2062         return FALSE;
2063 }
2064
2065 /*!
2066  *\brief        Cancel matcher dialog
2067  */
2068 static void prefs_matcher_cancel(void)
2069 {
2070         gtk_widget_hide(matcher.window);
2071         gtk_window_set_modal(GTK_WINDOW(matcher.window), FALSE);
2072         inc_unlock();
2073 }
2074
2075 /*!
2076  *\brief        Accept current matchers
2077  */
2078 static void prefs_matcher_ok(void)
2079 {
2080         MatcherList *matchers;
2081         MatcherProp *matcherprop;
2082         AlertValue val;
2083         gchar *matcher_str = NULL;
2084         gchar *str = NULL;
2085         gint row = 1;
2086         GtkTreeModel *model;
2087         GtkTreeIter iter;
2088
2089         matchers = prefs_matcher_get_list();
2090
2091         if (matchers != NULL) {
2092                 matcherprop = prefs_matcher_dialog_to_matcher();
2093                 if (matcherprop != NULL) {
2094                         str = matcherprop_to_string(matcherprop);
2095                         matcherprop_free(matcherprop);
2096                         if (strcmp(str, "all") != 0) {
2097                                 model = gtk_tree_view_get_model(GTK_TREE_VIEW
2098                                                 (matcher.cond_list_view));
2099
2100                                 while (gtk_tree_model_iter_nth_child(model, &iter, NULL, row)) {
2101                                         gtk_tree_model_get(model, &iter,
2102                                                            PREFS_MATCHER_COND, &matcher_str,
2103                                                            -1);
2104                                         if (matcher_str && strcmp(matcher_str, str) == 0) 
2105                                                 break;
2106                                         row++;
2107                                         g_free(matcher_str);
2108                                         matcher_str = NULL;
2109                                 }
2110
2111                                 if (!matcher_str || strcmp(matcher_str, str) != 0) {
2112                                         val = alertpanel(_("Entry not saved"),
2113                                                  _("The entry was not saved.\nClose anyway?"),
2114                                                  GTK_STOCK_CLOSE,
2115                                                  _("_Continue editing"),
2116                                                  NULL,
2117                                                  ALERTFOCUS_SECOND);
2118                                         if (G_ALERTDEFAULT != val) {
2119                                                 g_free(matcher_str);                                             
2120                                                 g_free(str);
2121                                                 return;
2122                                         }
2123                                 }
2124                                 g_free(matcher_str);
2125                         }
2126                 }
2127                 g_free(str);
2128                 gtk_widget_hide(matcher.window);
2129                 gtk_window_set_modal(GTK_WINDOW(matcher.window), FALSE);
2130                 if (matchers_callback != NULL)
2131                         matchers_callback(matchers);
2132                 matcherlist_free(matchers);
2133         }
2134         inc_unlock();
2135 }
2136
2137 /*!
2138  *\brief        Called when closing dialog box
2139  *
2140  *\param        widget Dialog widget
2141  *\param        event Event info
2142  *\param        data User data
2143  *
2144  *\return       gint TRUE
2145  */
2146 static gint prefs_matcher_deleted(GtkWidget *widget, GdkEventAny *event,
2147                                   gpointer data)
2148 {
2149         prefs_matcher_cancel();
2150         return TRUE;
2151 }
2152
2153 /*
2154  * Strings describing test format strings
2155  * 
2156  * When adding new lines, remember to put 2 strings for each line
2157  */
2158 static gchar *test_desc_strings[] = {
2159         "%%",   N_("literal %"),
2160         "%s",   N_("Subject"),
2161         "%f",   N_("From"),
2162         "%t",   N_("To"),
2163         "%c",   N_("Cc"),
2164         "%d",   N_("Date"),
2165         "%i",   N_("Message-ID"),
2166         "%n",   N_("Newsgroups"),
2167         "%r",   N_("References"),
2168         "%F",   N_("filename (should not be modified)"),
2169         "\\n",  N_("new line"),
2170         "\\",   N_("escape character for quotes"),
2171         "\\\"", N_("quote character"),
2172         NULL,   NULL
2173 };
2174
2175 static DescriptionWindow test_desc_win = { 
2176         NULL,
2177         NULL, 
2178         TRUE,
2179         2,
2180         N_("Match Type: 'Test'"),
2181         N_("'Test' allows you to test a message or message element "
2182            "using an external program or script. The program will "
2183            "return either 0 or 1.\n\n"
2184            "The following symbols can be used:"),
2185         test_desc_strings
2186 };
2187
2188
2189
2190 /*!
2191  *\brief        Show Test action's info
2192  */
2193 static void prefs_matcher_test_info(GtkWidget *widget, GtkWidget *parent)
2194 {
2195         test_desc_win.parent = parent;
2196         description_window_create(&test_desc_win);
2197 }
2198
2199 #ifndef USE_ALT_ADDRBOOK
2200 static void prefs_matcher_addressbook_select(void)
2201 {
2202         const gchar *folderpath = NULL;
2203         gchar *new_path = NULL;
2204
2205         folderpath = gtk_entry_get_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN((matcher.addressbook_folder_combo)))));
2206         new_path = addressbook_folder_selection(folderpath);
2207         if (new_path) {
2208                 gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN((matcher.addressbook_folder_combo)))), new_path);
2209                 g_free(new_path);
2210         } 
2211 }
2212 #endif
2213
2214 /*
2215  * list view
2216  */
2217
2218 static GtkListStore* prefs_matcher_create_data_store(void)
2219 {
2220         return gtk_list_store_new(N_PREFS_MATCHER_COLUMNS,
2221                                   G_TYPE_STRING,        
2222                                   G_TYPE_BOOLEAN,
2223                                   -1);
2224 }
2225
2226 static void prefs_matcher_list_view_insert_matcher(GtkWidget *list_view,
2227                                                    GtkTreeIter *row_iter,
2228                                                    const gchar *matcher,
2229                                                    gboolean is_valid) 
2230 {
2231         GtkTreeIter iter;
2232         GtkListStore *list_store = GTK_LIST_STORE(gtk_tree_view_get_model
2233                                         (GTK_TREE_VIEW(list_view)));
2234
2235         if (row_iter == NULL) {
2236                 /* append new */
2237                 gtk_list_store_append(list_store, &iter);
2238         } else {
2239                 /* change existing */
2240                 iter = *row_iter;
2241         }
2242
2243         gtk_list_store_set(list_store, &iter,
2244                            PREFS_MATCHER_COND, matcher,
2245                            PREFS_MATCHER_COND_VALID, is_valid,
2246                            -1);
2247 }
2248
2249 static GtkWidget *prefs_matcher_list_view_create(void)
2250 {
2251         GtkTreeView *list_view;
2252         GtkTreeSelection *selector;
2253         GtkTreeModel *model;
2254
2255         model = GTK_TREE_MODEL(prefs_matcher_create_data_store());
2256         list_view = GTK_TREE_VIEW(gtk_tree_view_new_with_model(model));
2257         g_object_unref(model);  
2258         
2259         gtk_tree_view_set_rules_hint(list_view, prefs_common.use_stripes_everywhere);
2260         gtk_tree_view_set_reorderable(list_view, TRUE);
2261         
2262         selector = gtk_tree_view_get_selection(list_view);
2263         gtk_tree_selection_set_mode(selector, GTK_SELECTION_BROWSE);
2264         gtk_tree_selection_set_select_function(selector, prefs_matcher_selected,
2265                                                NULL, NULL);
2266
2267         /* create the columns */
2268         prefs_matcher_create_list_view_columns(GTK_WIDGET(list_view));
2269
2270         return GTK_WIDGET(list_view);
2271 }
2272
2273 static void prefs_matcher_create_list_view_columns(GtkWidget *list_view)
2274 {
2275         GtkTreeViewColumn *column;
2276         GtkCellRenderer *renderer;
2277
2278         renderer = gtk_cell_renderer_text_new();
2279         column = gtk_tree_view_column_new_with_attributes
2280                 (_("Current condition rules"),
2281                  renderer,
2282                  "text", PREFS_MATCHER_COND,
2283                  NULL);
2284         gtk_tree_view_append_column(GTK_TREE_VIEW(list_view), column);          
2285 }
2286
2287 static void prefs_matcher_set_criteria(const gint criteria)
2288 {
2289         gint match_criteria = 0;
2290         
2291         switch (criteria) {
2292         case CRITERIA_FOUND_IN_ADDRESSBOOK:
2293                 match_criteria = MATCH_ABOOK;
2294                 break;  
2295         case CRITERIA_ALL:
2296                 match_criteria = MATCH_ALL;
2297                 break;
2298         case CRITERIA_AGE_GREATER:
2299         case CRITERIA_AGE_LOWER:
2300         case CRITERIA_AGE_GREATER_HOURS:
2301         case CRITERIA_AGE_LOWER_HOURS:
2302                 match_criteria = MATCH_AGE;
2303                 break;
2304         case CRITERIA_SCORE_GREATER:
2305         case CRITERIA_SCORE_LOWER:
2306         case CRITERIA_SCORE_EQUAL:
2307                 match_criteria = MATCH_SCORE;
2308                 break;
2309         case CRITERIA_SIZE_GREATER:
2310         case CRITERIA_SIZE_SMALLER:
2311         case CRITERIA_SIZE_EQUAL:
2312                 match_criteria = MATCH_SIZE;
2313                 break;
2314         case CRITERIA_SUBJECT:
2315         case CRITERIA_FROM:
2316         case CRITERIA_TO:
2317         case CRITERIA_CC:
2318         case CRITERIA_TO_OR_CC:
2319         case CRITERIA_NEWSGROUPS:
2320         case CRITERIA_MESSAGEID:
2321         case CRITERIA_INREPLYTO:
2322         case CRITERIA_REFERENCES:
2323         case CRITERIA_HEADER:
2324                 match_criteria = MATCH_HEADER;
2325                 break;
2326         case CRITERIA_HEADERS_PART:
2327         case CRITERIA_HEADERS_CONT:
2328         case CRITERIA_BODY_PART:
2329         case CRITERIA_MESSAGE:
2330                 match_criteria = MATCH_PHRASE;
2331                 break;
2332         case CRITERIA_TEST:
2333                 match_criteria = MATCH_TEST;
2334                 break;
2335 #if !GTK_CHECK_VERSION(3, 0, 0)
2336         case CRITERIA_COLORLABEL:
2337                 match_criteria = MATCH_LABEL;
2338                 break;
2339 #endif
2340         case CRITERIA_TAG:
2341         case CRITERIA_TAGGED:
2342                 match_criteria = MATCH_TAGS;
2343                 break;
2344         case CRITERIA_UNREAD:
2345         case CRITERIA_NEW:
2346         case CRITERIA_MARKED:
2347         case CRITERIA_DELETED:
2348         case CRITERIA_REPLIED:
2349         case CRITERIA_FORWARDED:
2350         case CRITERIA_LOCKED:
2351         case CRITERIA_SPAM:
2352         case CRITERIA_HAS_ATTACHMENT:
2353         case CRITERIA_SIGNED:
2354                 match_criteria = MATCH_FLAG;
2355                 break;
2356         case CRITERIA_PARTIAL:
2357                 match_criteria = MATCH_PARTIAL;
2358                 break;
2359         case CRITERIA_IGNORE_THREAD:
2360         case CRITERIA_WATCH_THREAD:
2361                 match_criteria = MATCH_THREAD;
2362                 break;
2363         }
2364         
2365         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo),
2366                                  match_criteria);
2367         
2368         switch(match_criteria) {
2369         case MATCH_HEADER:
2370                 if(criteria != CRITERIA_HEADER)
2371                         combobox_select_by_data(GTK_COMBO_BOX(
2372                                                 matcher.headers_combo),
2373                                                 criteria);
2374                 break;
2375         case MATCH_AGE:
2376         case MATCH_SCORE:
2377         case MATCH_SIZE:
2378         case MATCH_FLAG:
2379                 combobox_select_by_data(GTK_COMBO_BOX(
2380                                         matcher.match_combo), criteria);
2381                 break;
2382         case MATCH_PHRASE:
2383         case MATCH_TAGS:
2384                 combobox_select_by_data(GTK_COMBO_BOX(
2385                                         matcher.criteria_combo2), criteria);
2386                 break;
2387         }
2388 }
2389
2390 static gboolean prefs_matcher_selected(GtkTreeSelection *selector,
2391                                        GtkTreeModel *model, 
2392                                        GtkTreePath *path,
2393                                        gboolean currently_selected,
2394                                        gpointer data)
2395 {
2396         gchar *matcher_str;
2397         MatcherProp *prop;
2398         gboolean negative_cond;
2399         gint criteria;
2400         GtkWidget *menu;
2401         GtkTreeIter iter;
2402         gboolean is_valid;
2403
2404         if (currently_selected)
2405                 return TRUE;
2406
2407         if (!gtk_tree_model_get_iter(model, &iter, path))
2408                 return TRUE;
2409
2410         gtk_tree_model_get(model, &iter, 
2411                            PREFS_MATCHER_COND_VALID,  &is_valid,
2412                            PREFS_MATCHER_COND, &matcher_str,
2413                            -1);
2414         
2415         if (!is_valid) {
2416                 g_free(matcher_str);
2417                 prefs_matcher_reset_condition();
2418                 return TRUE;
2419         }
2420
2421         negative_cond = FALSE;
2422
2423         prop = matcher_parser_get_prop(matcher_str);
2424         if (prop == NULL) {
2425                 g_free(matcher_str);
2426                 return TRUE;
2427         }               
2428
2429         criteria = prefs_matcher_get_criteria_from_matching(prop->criteria);
2430         prefs_matcher_set_criteria(criteria);
2431
2432         switch(prop->criteria) {
2433         case MATCHCRITERIA_NOT_UNREAD:
2434         case MATCHCRITERIA_NOT_NEW:
2435         case MATCHCRITERIA_NOT_MARKED:
2436         case MATCHCRITERIA_NOT_DELETED:
2437         case MATCHCRITERIA_NOT_REPLIED:
2438         case MATCHCRITERIA_NOT_FORWARDED:
2439         case MATCHCRITERIA_NOT_LOCKED:
2440         case MATCHCRITERIA_NOT_SPAM:
2441         case MATCHCRITERIA_HAS_NO_ATTACHMENT:
2442         case MATCHCRITERIA_NOT_SIGNED:
2443         case MATCHCRITERIA_NOT_PARTIAL:
2444         case MATCHCRITERIA_NOT_COLORLABEL:
2445         case MATCHCRITERIA_NOT_IGNORE_THREAD:
2446         case MATCHCRITERIA_NOT_WATCH_THREAD:
2447         case MATCHCRITERIA_NOT_SUBJECT:
2448         case MATCHCRITERIA_NOT_FROM:
2449         case MATCHCRITERIA_NOT_TO:
2450         case MATCHCRITERIA_NOT_CC:
2451         case MATCHCRITERIA_NOT_TO_AND_NOT_CC:
2452         case MATCHCRITERIA_NOT_TAG:
2453         case MATCHCRITERIA_NOT_TAGGED:
2454         case MATCHCRITERIA_NOT_NEWSGROUPS:
2455         case MATCHCRITERIA_NOT_MESSAGEID:
2456         case MATCHCRITERIA_NOT_INREPLYTO:
2457         case MATCHCRITERIA_NOT_REFERENCES:
2458         case MATCHCRITERIA_NOT_HEADER:
2459         case MATCHCRITERIA_NOT_HEADERS_PART:
2460         case MATCHCRITERIA_NOT_HEADERS_CONT:
2461         case MATCHCRITERIA_NOT_MESSAGE:
2462         case MATCHCRITERIA_NOT_BODY_PART:
2463         case MATCHCRITERIA_NOT_TEST:
2464         case MATCHCRITERIA_NOT_FOUND_IN_ADDRESSBOOK:
2465                 negative_cond = TRUE;
2466                 break;
2467         }
2468         
2469         switch(prop->criteria) {
2470         case MATCHCRITERIA_ALL:
2471                 break;
2472
2473         case MATCHCRITERIA_NOT_SUBJECT:
2474         case MATCHCRITERIA_NOT_FROM:
2475         case MATCHCRITERIA_NOT_TO:
2476         case MATCHCRITERIA_NOT_CC:
2477         case MATCHCRITERIA_NOT_TO_AND_NOT_CC:
2478         case MATCHCRITERIA_NOT_TAG:
2479         case MATCHCRITERIA_NOT_NEWSGROUPS:
2480         case MATCHCRITERIA_NOT_MESSAGEID:
2481         case MATCHCRITERIA_NOT_INREPLYTO:
2482         case MATCHCRITERIA_NOT_REFERENCES:
2483         case MATCHCRITERIA_NOT_HEADERS_PART:
2484         case MATCHCRITERIA_NOT_HEADERS_CONT:
2485         case MATCHCRITERIA_NOT_BODY_PART:
2486         case MATCHCRITERIA_NOT_MESSAGE:
2487         case MATCHCRITERIA_NOT_TEST:
2488         case MATCHCRITERIA_SUBJECT:
2489         case MATCHCRITERIA_FROM:
2490         case MATCHCRITERIA_TO:
2491         case MATCHCRITERIA_CC:
2492         case MATCHCRITERIA_TO_OR_CC:
2493         case MATCHCRITERIA_TAG:
2494         case MATCHCRITERIA_NEWSGROUPS:
2495         case MATCHCRITERIA_MESSAGEID:
2496         case MATCHCRITERIA_INREPLYTO:
2497         case MATCHCRITERIA_REFERENCES:
2498         case MATCHCRITERIA_HEADERS_PART:
2499         case MATCHCRITERIA_HEADERS_CONT:
2500         case MATCHCRITERIA_BODY_PART:
2501         case MATCHCRITERIA_MESSAGE:
2502         case MATCHCRITERIA_TEST:
2503                 gtk_entry_set_text(GTK_ENTRY(matcher.string_entry), prop->expr);
2504                 break;
2505
2506         case MATCHCRITERIA_FOUND_IN_ADDRESSBOOK:
2507         case MATCHCRITERIA_NOT_FOUND_IN_ADDRESSBOOK:
2508         {
2509                 gchar *header;
2510                 gchar *expr;
2511
2512                 /* matcher expressions contain UNtranslated "Any"/"All",
2513                   select the relevant translated combo item */
2514                 if (strcasecmp(prop->header, "All") == 0)
2515                         header = (gchar*)C_("Filtering Matcher Menu", "All");
2516                 else
2517                         if (strcasecmp(prop->header, "Any") == 0)
2518                                 header = _("Any");
2519                         else
2520                                 header = prop->header;
2521                 if (strcasecmp(prop->expr, "Any") == 0)
2522                         expr = _("Any");
2523                 else
2524                         expr = prop->expr;
2525
2526                 gtk_entry_set_text(GTK_ENTRY(matcher.header_addr_entry), header);
2527                 gtk_entry_set_text(GTK_ENTRY(gtk_bin_get_child(GTK_BIN((matcher.addressbook_folder_combo)))), expr);
2528                 break;
2529         }
2530
2531         case MATCHCRITERIA_AGE_GREATER:
2532         case MATCHCRITERIA_AGE_LOWER:
2533                 if(prop->value >= 7 && !(prop->value % 7)) {
2534                         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2535                                                  AGE_WEEKS);
2536                         gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2537                                         matcher.numeric_entry), prop->value/7);
2538                 } else {
2539                         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2540                                                  AGE_DAYS);
2541                         gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2542                                         matcher.numeric_entry), prop->value);
2543                 }
2544                 break;
2545                 
2546         case MATCHCRITERIA_AGE_GREATER_HOURS:
2547         case MATCHCRITERIA_AGE_LOWER_HOURS:
2548                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2549                                          AGE_HOURS);
2550                 gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2551                                 matcher.numeric_entry), prop->value);
2552                 break;
2553                 
2554         case MATCHCRITERIA_SCORE_GREATER:
2555         case MATCHCRITERIA_SCORE_LOWER:
2556         case MATCHCRITERIA_SCORE_EQUAL:
2557                 gtk_spin_button_set_value(GTK_SPIN_BUTTON(matcher.numeric_entry),
2558                                           prop->value);
2559                 break;
2560
2561         case MATCHCRITERIA_SIZE_GREATER:
2562         case MATCHCRITERIA_SIZE_SMALLER:
2563         case MATCHCRITERIA_SIZE_EQUAL:
2564                 if(prop->value >= MB_SIZE && !(prop->value % MB_SIZE)) {
2565                         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2566                                                  SIZE_UNIT_MBYTES);
2567                         gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2568                                         matcher.numeric_entry), prop->value/MB_SIZE);
2569                 } else if(prop->value >= KB_SIZE && !(prop->value % KB_SIZE)) {
2570                         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2571                                                  SIZE_UNIT_KBYTES);
2572                         gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2573                                         matcher.numeric_entry), prop->value/KB_SIZE);
2574                 } else {
2575                         gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2576                                                  SIZE_UNIT_BYTES);
2577                         gtk_spin_button_set_value(GTK_SPIN_BUTTON(
2578                                         matcher.numeric_entry), prop->value);           
2579                 }
2580                 break;
2581
2582 #if !GTK_CHECK_VERSION(3, 0, 0)
2583         case MATCHCRITERIA_NOT_COLORLABEL:
2584         case MATCHCRITERIA_COLORLABEL:
2585                 gtk_cmoption_menu_set_history(GTK_CMOPTION_MENU(matcher.color_optmenu),
2586                                             prop->value + 1);
2587                 menu = gtk_cmoption_menu_get_menu(GTK_CMOPTION_MENU(matcher.color_optmenu));
2588                 g_signal_emit_by_name(G_OBJECT(menu), "selection-done", menu);
2589                 break;
2590 #endif
2591
2592         case MATCHCRITERIA_NOT_HEADER:
2593         case MATCHCRITERIA_HEADER:
2594                 gtk_entry_set_text(GTK_ENTRY(matcher.header_entry), prop->header);
2595                 gtk_entry_set_text(GTK_ENTRY(matcher.string_entry), prop->expr);
2596                 break;
2597         }
2598
2599         switch(criteria) {
2600         case CRITERIA_SUBJECT:
2601         case CRITERIA_FROM:
2602         case CRITERIA_TO:
2603         case CRITERIA_CC:
2604         case CRITERIA_TO_OR_CC:
2605         case CRITERIA_NEWSGROUPS:
2606         case CRITERIA_MESSAGEID:
2607         case CRITERIA_INREPLYTO:
2608         case CRITERIA_REFERENCES:
2609         case CRITERIA_HEADER:
2610         case CRITERIA_HEADERS_PART:
2611         case CRITERIA_HEADERS_CONT:
2612         case CRITERIA_BODY_PART:
2613         case CRITERIA_MESSAGE:
2614         case CRITERIA_TAG:
2615         case CRITERIA_TAGGED:
2616         case CRITERIA_TEST:
2617                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo),
2618                                         negative_cond ? PREDICATE_DOES_NOT_CONTAIN :
2619                                                         PREDICATE_CONTAINS);
2620                 break;
2621         case CRITERIA_FOUND_IN_ADDRESSBOOK:
2622         case CRITERIA_UNREAD:
2623         case CRITERIA_NEW:
2624         case CRITERIA_MARKED:
2625         case CRITERIA_DELETED:
2626         case CRITERIA_REPLIED:
2627         case CRITERIA_FORWARDED:
2628         case CRITERIA_LOCKED:
2629         case CRITERIA_SPAM:
2630         case CRITERIA_HAS_ATTACHMENT:
2631         case CRITERIA_SIGNED:
2632         case CRITERIA_COLORLABEL:
2633                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.match_combo2),
2634                                          negative_cond ? PREDICATE_FLAG_DISABLED :
2635                                                          PREDICATE_FLAG_ENABLED);
2636                 break;
2637         case CRITERIA_WATCH_THREAD:
2638                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo2),
2639                                          negative_cond ? THREAD_NOT_WATCHED :
2640                                                          THREAD_WATCHED);
2641                 break;
2642         case CRITERIA_IGNORE_THREAD:
2643                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo2),
2644                                          negative_cond ? THREAD_NOT_IGNORED :
2645                                                          THREAD_IGNORED);
2646                 break;  
2647         case CRITERIA_PARTIAL:
2648                 gtk_combo_box_set_active(GTK_COMBO_BOX(matcher.criteria_combo2),
2649                                          negative_cond ? PREDICATE_FLAG_DISABLED :
2650                                                          PREDICATE_FLAG_ENABLED);
2651                 break;
2652         }
2653
2654         switch(prop->matchtype) {
2655         case MATCHTYPE_MATCH:
2656                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), FALSE);
2657                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), TRUE);
2658                 break;
2659
2660         case MATCHTYPE_MATCHCASE:
2661                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), FALSE);
2662                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), FALSE);
2663                 break;
2664
2665         case MATCHTYPE_REGEXP:
2666                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), TRUE);
2667                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), TRUE);
2668                 break;
2669
2670         case MATCHTYPE_REGEXPCASE:
2671                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.regexp_checkbtn), TRUE);
2672                 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(matcher.case_checkbtn), FALSE);
2673                 break;
2674         }
2675
2676         g_free(matcher_str);
2677         return TRUE;
2678 }
2679