Statistics
| Revision:

root / src / prefs_filter_edit.c @ 198

History | View | Annotate | Download (57.8 kB)

1
/*
2
 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3
 * Copyright (C) 1999-2005 Hiroyuki Yamamoto
4
 *
5
 * This program is free software; you can redistribute it and/or modify
6
 * it under the terms of the GNU General Public License as published by
7
 * the Free Software Foundation; either version 2 of the License, or
8
 * (at your option) any later version.
9
 *
10
 * This program is distributed in the hope that it will be useful,
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
 * GNU General Public License for more details.
14
 *
15
 * You should have received a copy of the GNU General Public License
16
 * along with this program; if not, write to the Free Software
17
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
18
 */
19
20
#ifdef HAVE_CONFIG_H
21
#  include "config.h"
22
#endif
23
24
#include "defs.h"
25
26
#include <glib.h>
27
#include <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
#include "main.h"
36
#include "prefs.h"
37
#include "prefs_filter.h"
38
#include "prefs_filter_edit.h"
39
#include "prefs_common.h"
40
#include "mainwindow.h"
41
#include "foldersel.h"
42
#include "colorlabel.h"
43
#include "manage_window.h"
44
#include "procheader.h"
45
#include "menu.h"
46
#include "filter.h"
47
#include "utils.h"
48
#include "gtkutils.h"
49
#include "stock_pixmap.h"
50
#include "alertpanel.h"
51
#include "folder.h"
52
53
typedef enum
54
{
55
        PF_COND_HEADER,
56
        PF_COND_TO_OR_CC,
57
        PF_COND_ANY_HEADER,
58
        PF_COND_BODY,
59
        PF_COND_CMD_TEST,
60
        PF_COND_SIZE,
61
        PF_COND_AGE,
62
        PF_COND_ACCOUNT,
63
        PF_COND_EDIT_HEADER,
64
        PF_COND_SEPARATOR,
65
        PF_COND_NONE
66
} CondMenuType;
67
68
typedef enum
69
{
70
        PF_MATCH_CONTAIN,
71
        PF_MATCH_NOT_CONTAIN,
72
        PF_MATCH_EQUAL,
73
        PF_MATCH_NOT_EQUAL,
74
        PF_MATCH_REGEX,
75
        PF_MATCH_NOT_REGEX,
76
        PF_MATCH_NONE
77
} MatchMenuType;
78
79
typedef enum
80
{
81
        PF_SIZE_LARGER,
82
        PF_SIZE_SMALLER
83
} SizeMatchType;
84
85
typedef enum
86
{
87
        PF_AGE_LONGER,
88
        PF_AGE_SHORTER
89
} AgeMatchType;
90
91
typedef enum
92
{
93
        PF_ACTION_MOVE,
94
        PF_ACTION_COPY,
95
        PF_ACTION_NOT_RECEIVE,
96
        PF_ACTION_DELETE,
97
        PF_ACTION_EXEC,
98
        PF_ACTION_EXEC_ASYNC,
99
        PF_ACTION_MARK,
100
        PF_ACTION_COLOR_LABEL,
101
        PF_ACTION_MARK_READ,
102
        PF_ACTION_FORWARD,
103
        PF_ACTION_FORWARD_AS_ATTACHMENT,
104
        PF_ACTION_REDIRECT,
105
        PF_ACTION_STOP_EVAL,
106
        PF_ACTION_SEPARATOR,
107
        PF_ACTION_NONE
108
} ActionMenuType;
109
110
static struct FilterRuleEditWindow {
111
        GtkWidget *window;
112
113
        GtkWidget *name_entry;
114
        GtkWidget *bool_op_optmenu;
115
116
        GtkWidget *cond_scrolled_win;
117
        GtkWidget *cond_vbox;
118
        GSList *cond_hbox_list;
119
120
        GtkWidget *action_scrolled_win;
121
        GtkWidget *action_vbox;
122
        GSList *action_hbox_list;
123
124
        GtkWidget *ok_btn;
125
        GtkWidget *cancel_btn;
126
127
        GSList *hdr_list;
128
        GSList *rule_hdr_list;
129
130
        FilterRule *new_rule;
131
        gboolean edit_finished;
132
} rule_edit_window;
133
134
static struct FilterEditHeaderListDialog {
135
        GtkWidget *window;
136
        GtkWidget *clist;
137
        GtkWidget *entry;
138
139
        gboolean finished;
140
        gboolean ok;
141
} edit_header_list_dialog;
142
143
typedef struct _CondHBox {
144
        GtkWidget *hbox;
145
146
        GtkWidget *cond_type_optmenu;
147
        GtkWidget *match_type_optmenu;
148
        GtkWidget *size_match_optmenu;
149
        GtkWidget *age_match_optmenu;
150
        GtkWidget *key_entry;
151
        GtkWidget *spin_btn;
152
        GtkWidget *label;
153
154
        GtkWidget *del_btn;
155
        GtkWidget *add_btn;
156
157
        CondMenuType cur_type;
158
        gchar *cur_header_name;
159
} CondHBox;
160
161
typedef struct _ActionHBox {
162
        GtkWidget *hbox;
163
164
        GtkWidget *action_type_optmenu;
165
        GtkWidget *label;
166
        GtkWidget *folder_entry;
167
        GtkWidget *cmd_entry;
168
        GtkWidget *address_entry;
169
        GtkWidget *clabel_optmenu;
170
171
        GtkWidget *folder_sel_btn;
172
173
        GtkWidget *action_type_menu_items[PF_ACTION_NONE];
174
175
        GtkWidget *del_btn;
176
        GtkWidget *add_btn;
177
} ActionHBox;
178
179
static void prefs_filter_edit_create                (void);
180
static void prefs_filter_edit_clear                (void);
181
static void prefs_filter_edit_rule_to_dialog        (FilterRule        *rule);
182
static void prefs_filter_edit_set_header_list        (FilterRule        *rule);
183
static void prefs_filter_edit_update_header_list(void);
184
185
static CondHBox *prefs_filter_edit_cond_hbox_create        (void);
186
static ActionHBox *prefs_filter_edit_action_hbox_create        (void);
187
static void prefs_filter_edit_cond_hbox_set                (CondHBox        *hbox,
188
                                                         FilterCond        *cond);
189
static void prefs_filter_edit_action_hbox_set                (ActionHBox        *hbox,
190
                                                         FilterAction        *action);
191
192
static void prefs_filter_edit_cond_hbox_select        (CondHBox        *hbox,
193
                                                 CondMenuType         type,
194
                                                 const gchar        *header_name);
195
196
static void prefs_filter_edit_set_cond_hbox_widgets        (CondHBox        *hbox,
197
                                                         CondMenuType         type);
198
static void prefs_filter_edit_set_action_hbox_widgets        (ActionHBox        *hbox,
199
                                                         ActionMenuType         type);
200
201
static void prefs_filter_edit_set_action_hbox_menu_sensitive
202
                                                (ActionHBox        *hbox,
203
                                                 ActionMenuType         type,
204
                                                 gboolean         sensitive);
205
static void prefs_filter_edit_set_action_hbox_menus_sensitive
206
                                                (void);
207
208
static void prefs_filter_edit_get_action_hbox_menus_selection
209
                                                (gboolean        *selection);
210
static ActionMenuType prefs_filter_edit_get_action_hbox_type
211
                                                (ActionHBox        *hbox);
212
213
static void prefs_filter_edit_insert_cond_hbox        (CondHBox        *hbox,
214
                                                 gint                 pos);
215
static void prefs_filter_edit_insert_action_hbox(ActionHBox        *hbox,
216
                                                 gint                 pos);
217
static void prefs_filter_edit_remove_cond_hbox        (CondHBox        *hbox);
218
static void prefs_filter_edit_remove_action_hbox(ActionHBox        *hbox);
219
220
static void prefs_filter_edit_add_rule_cond        (FilterRule        *rule);
221
static void prefs_filter_edit_add_rule_action        (FilterRule        *rule);
222
223
static void prefs_filter_edit_set_cond_header_menu
224
                                                (CondHBox        *hbox);
225
226
static void prefs_filter_edit_activate_cond_header
227
                                                (const gchar        *header);
228
229
static void prefs_filter_edit_edit_header_list                (void);
230
231
static FilterRule *prefs_filter_edit_dialog_to_rule        (void);
232
233
/* callback functions */
234
static gint prefs_filter_edit_deleted                (GtkWidget        *widget,
235
                                                 GdkEventAny        *event,
236
                                                 gpointer         data);
237
static gboolean prefs_filter_edit_key_pressed        (GtkWidget        *widget,
238
                                                 GdkEventKey        *event,
239
                                                 gpointer         data);
240
static void prefs_filter_edit_ok                (void);
241
static void prefs_filter_edit_cancel                (void);
242
243
static void prefs_filter_cond_activated_cb        (GtkWidget        *widget,
244
                                                 gpointer         data);
245
static void prefs_filter_action_activated_cb        (GtkWidget        *widget,
246
                                                 gpointer         data);
247
248
static void prefs_filter_action_select_dest_cb        (GtkWidget        *widget,
249
                                                 gpointer         data);
250
251
static void prefs_filter_cond_del_cb                (GtkWidget        *widget,
252
                                                 gpointer         data);
253
static void prefs_filter_cond_add_cb                (GtkWidget        *widget,
254
                                                 gpointer         data);
255
static void prefs_filter_action_del_cb                (GtkWidget        *widget,
256
                                                 gpointer         data);
257
static void prefs_filter_action_add_cb                (GtkWidget        *widget,
258
                                                 gpointer         data);
259
260
261
FilterRule *prefs_filter_edit_open(FilterRule *rule, const gchar *header)
262
{
263
        static gboolean lock = FALSE;
264
        FilterRule *new_rule;
265
266
        if (lock)
267
                return NULL;
268
269
        lock = TRUE;
270
271
        if (!rule_edit_window.window)
272
                prefs_filter_edit_create();
273
274
        manage_window_set_transient(GTK_WINDOW(rule_edit_window.window));
275
276
        prefs_filter_edit_set_header_list(rule);
277
        prefs_filter_edit_rule_to_dialog(rule);
278
        if (header)
279
                prefs_filter_edit_activate_cond_header(header);
280
        GTK_EVENTS_FLUSH();
281
        gtk_widget_show(rule_edit_window.window);
282
283
        rule_edit_window.new_rule = NULL;
284
        rule_edit_window.edit_finished = FALSE;
285
        while (rule_edit_window.edit_finished == FALSE)
286
                gtk_main_iteration();
287
288
        gtk_widget_hide(rule_edit_window.window);
289
        prefs_filter_edit_clear();
290
        prefs_filter_set_msg_header_list(NULL);
291
292
        new_rule = rule_edit_window.new_rule;
293
        rule_edit_window.new_rule = NULL;
294
295
        if (new_rule)
296
                debug_print("new rule created: %s\n", new_rule->name);
297
298
        lock = FALSE;
299
300
        return new_rule;
301
}
302
303
static void prefs_filter_edit_create(void)
304
{
305
        GtkWidget *window;
306
307
        GtkWidget *vbox;
308
309
        GtkWidget *hbox;
310
        GtkWidget *label;
311
        GtkWidget *name_entry;
312
313
        GtkWidget *bool_op_optmenu;
314
        GtkWidget *menu;
315
        GtkWidget *menuitem;
316
317
        GtkWidget *cond_scrolled_win;
318
        GtkWidget *cond_vbox;
319
320
        GtkWidget *action_scrolled_win;
321
        GtkWidget *action_vbox;
322
323
        GtkWidget *ok_btn;
324
        GtkWidget *cancel_btn;
325
        GtkWidget *confirm_area;
326
327
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
328
        gtk_container_set_border_width(GTK_CONTAINER(window), 8);
329
        gtk_widget_set_size_request(window, 632, 405);
330
        gtk_window_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
331
        gtk_window_set_modal(GTK_WINDOW(window), TRUE);
332
        gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, FALSE);
333
        gtk_widget_realize(window);
334
335
        vbox = gtk_vbox_new(FALSE, 6);
336
        gtk_widget_show(vbox);
337
        gtk_container_add(GTK_CONTAINER(window), vbox);
338
339
        gtkut_stock_button_set_create(&confirm_area, &ok_btn, GTK_STOCK_OK,
340
                                      &cancel_btn, GTK_STOCK_CANCEL,
341
                                      NULL, NULL);
342
        gtk_widget_show(confirm_area);
343
        gtk_box_pack_end(GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
344
        gtk_widget_grab_default(ok_btn);
345
346
        gtk_window_set_title(GTK_WINDOW(window),
347
                             _("Filter rule"));
348
        g_signal_connect(G_OBJECT(window), "delete_event",
349
                         G_CALLBACK(prefs_filter_edit_deleted), NULL);
350
        g_signal_connect(G_OBJECT(window), "key_press_event",
351
                         G_CALLBACK(prefs_filter_edit_key_pressed), NULL);
352
        MANAGE_WINDOW_SIGNALS_CONNECT (window);
353
        g_signal_connect(G_OBJECT(ok_btn), "clicked",
354
                         G_CALLBACK(prefs_filter_edit_ok), NULL);
355
        g_signal_connect(G_OBJECT(cancel_btn), "clicked",
356
                         G_CALLBACK(prefs_filter_edit_cancel), NULL);
357
358
        hbox = gtk_hbox_new(FALSE, 4);
359
        gtk_widget_show(hbox);
360
        gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
361
362
        label = gtk_label_new(_("Name:"));
363
        gtk_widget_show(label);
364
        gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
365
366
        name_entry = gtk_entry_new();
367
        gtk_widget_show(name_entry);
368
        gtk_box_pack_start(GTK_BOX(hbox), name_entry, TRUE, TRUE, 0);
369
370
        hbox = gtk_hbox_new(FALSE, 4);
371
        gtk_widget_show(hbox);
372
        gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
373
374
        bool_op_optmenu = gtk_option_menu_new();
375
        gtk_widget_show(bool_op_optmenu);
376
        gtk_box_pack_start(GTK_BOX(hbox), bool_op_optmenu, FALSE, FALSE, 0);
377
378
        menu = gtk_menu_new();
379
        gtk_widget_show(menu);
380
        MENUITEM_ADD(menu, menuitem,
381
                     _("If any of the following condition matches"), FLT_OR);
382
        MENUITEM_ADD(menu, menuitem,
383
                     _("If all of the following conditions match"), FLT_AND);
384
        gtk_option_menu_set_menu(GTK_OPTION_MENU(bool_op_optmenu), menu);
385
386
        cond_scrolled_win = gtk_scrolled_window_new(NULL, NULL);
387
        gtk_widget_show(cond_scrolled_win);
388
        gtk_widget_set_size_request(cond_scrolled_win, -1, 125);
389
        gtk_box_pack_start(GTK_BOX(vbox), cond_scrolled_win, TRUE, TRUE, 0);
390
        gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(cond_scrolled_win),
391
                                       GTK_POLICY_AUTOMATIC,
392
                                       GTK_POLICY_AUTOMATIC);
393
394
        cond_vbox = gtk_vbox_new(FALSE, 2);
395
        gtk_widget_show(cond_vbox);
396
        gtk_container_set_border_width(GTK_CONTAINER(cond_vbox), 2);
397
        gtk_scrolled_window_add_with_viewport
398
                (GTK_SCROLLED_WINDOW(cond_scrolled_win), cond_vbox);
399
400
        hbox = gtk_hbox_new(FALSE, 4);
401
        gtk_widget_show(hbox);
402
        gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
403
404
        label = gtk_label_new(_("Perform the following actions:"));
405
        gtk_widget_show(label);
406
        gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
407
408
        action_scrolled_win = gtk_scrolled_window_new(NULL, NULL);
409
        gtk_widget_show(action_scrolled_win);
410
        gtk_box_pack_start(GTK_BOX(vbox), action_scrolled_win, TRUE, TRUE, 0);
411
        gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(action_scrolled_win),
412
                                       GTK_POLICY_AUTOMATIC,
413
                                       GTK_POLICY_AUTOMATIC);
414
415
        action_vbox = gtk_vbox_new(FALSE, 2);
416
        gtk_widget_show(action_vbox);
417
        gtk_container_set_border_width(GTK_CONTAINER(action_vbox), 2);
418
        gtk_scrolled_window_add_with_viewport
419
                (GTK_SCROLLED_WINDOW(action_scrolled_win), action_vbox);
420
421
        rule_edit_window.window = window;
422
        rule_edit_window.name_entry = name_entry;
423
424
        rule_edit_window.bool_op_optmenu = bool_op_optmenu;
425
        rule_edit_window.cond_scrolled_win = cond_scrolled_win;
426
        rule_edit_window.cond_vbox = cond_vbox;
427
        rule_edit_window.action_scrolled_win = action_scrolled_win;
428
        rule_edit_window.action_vbox = action_vbox;
429
430
        rule_edit_window.ok_btn = ok_btn;
431
        rule_edit_window.cancel_btn = cancel_btn;
432
}
433
434
static void prefs_filter_edit_clear(void)
435
{
436
        while (rule_edit_window.cond_hbox_list) {
437
                CondHBox *hbox =
438
                        (CondHBox *)rule_edit_window.cond_hbox_list->data;
439
                prefs_filter_edit_remove_cond_hbox(hbox);
440
        }
441
        while (rule_edit_window.action_hbox_list) {
442
                ActionHBox *hbox =
443
                        (ActionHBox *)rule_edit_window.action_hbox_list->data;
444
                prefs_filter_edit_remove_action_hbox(hbox);
445
        }
446
447
        g_slist_free(rule_edit_window.hdr_list);
448
        rule_edit_window.hdr_list = NULL;
449
        procheader_header_list_destroy(rule_edit_window.rule_hdr_list);
450
        rule_edit_window.rule_hdr_list = NULL;
451
}
452
453
static void prefs_filter_edit_rule_to_dialog(FilterRule *rule)
454
{
455
        gint index = 0;
456
        static gint count = 1;
457
458
        if (rule && rule->name)
459
                gtk_entry_set_text(GTK_ENTRY(rule_edit_window.name_entry),
460
                                   rule->name);
461
        else {
462
                gchar rule_name[32];
463
                g_snprintf(rule_name, sizeof(rule_name), "Rule %d", count++);
464
                gtk_entry_set_text(GTK_ENTRY(rule_edit_window.name_entry),
465
                                   rule_name);
466
        }
467
468
        if (rule) {
469
                index = menu_find_option_menu_index
470
                        (GTK_OPTION_MENU(rule_edit_window.bool_op_optmenu),
471
                         GINT_TO_POINTER(rule->bool_op), NULL);
472
                if (index < 0)
473
                        index = 0;
474
        }
475
        if (index >= 0) {
476
                gtk_option_menu_set_history
477
                        (GTK_OPTION_MENU(rule_edit_window.bool_op_optmenu),
478
                         index);
479
        }
480
481
        gtkut_scrolled_window_reset_position
482
                (GTK_SCROLLED_WINDOW(rule_edit_window.cond_scrolled_win));
483
        gtkut_scrolled_window_reset_position
484
                (GTK_SCROLLED_WINDOW(rule_edit_window.action_scrolled_win));
485
486
        prefs_filter_edit_add_rule_cond(rule);
487
        prefs_filter_edit_add_rule_action(rule);
488
}
489
490
static void prefs_filter_edit_set_header_list(FilterRule *rule)
491
{
492
        GSList *list;
493
        GSList *rule_hdr_list = NULL;
494
        GSList *cur;
495
        FilterCond *cond;
496
497
        g_slist_free(rule_edit_window.hdr_list);
498
        rule_edit_window.hdr_list = NULL;
499
        procheader_header_list_destroy(rule_edit_window.rule_hdr_list);
500
        rule_edit_window.rule_hdr_list = NULL;
501
502
        list = prefs_filter_get_header_list();
503
        rule_edit_window.hdr_list = list;
504
505
        if (!rule)
506
                return;
507
508
        for (cur = rule->cond_list; cur != NULL; cur = cur->next) {
509
                cond = (FilterCond *)cur->data;
510
511
                if (cond->type == FLT_COND_HEADER &&
512
                    procheader_find_header_list
513
                        (rule_hdr_list, cond->header_name) < 0)
514
                        rule_hdr_list = procheader_add_header_list
515
                                (rule_hdr_list, cond->header_name, NULL);
516
        }
517
518
        rule_edit_window.rule_hdr_list = rule_hdr_list;
519
520
        rule_edit_window.hdr_list =
521
                procheader_merge_header_list(list, rule_hdr_list);
522
}
523
524
static void prefs_filter_edit_update_header_list(void)
525
{
526
        GSList *list;
527
528
        g_slist_free(rule_edit_window.hdr_list);
529
        rule_edit_window.hdr_list = NULL;
530
531
        list = prefs_filter_get_header_list();
532
        rule_edit_window.hdr_list = list;
533
534
        rule_edit_window.hdr_list =
535
                procheader_merge_header_list(list,
536
                                             rule_edit_window.rule_hdr_list);
537
}
538
539
static CondHBox *prefs_filter_edit_cond_hbox_create(void)
540
{
541
        CondHBox *cond_hbox;
542
        GtkWidget *hbox;
543
        GtkWidget *cond_type_optmenu;
544
        GtkWidget *match_type_optmenu;
545
        GtkWidget *size_match_optmenu;
546
        GtkWidget *age_match_optmenu;
547
        GtkWidget *menu;
548
        GtkWidget *menuitem;
549
        GtkWidget *key_entry;
550
        GtkObject *spin_btn_adj;
551
        GtkWidget *spin_btn;
552
        GtkWidget *label;
553
        GtkWidget *del_btn;
554
        GtkWidget *add_btn;
555
        GtkWidget *del_img;
556
        GtkWidget *add_img;
557
558
        cond_hbox = g_new0(CondHBox, 1);
559
560
        hbox = gtk_hbox_new(FALSE, 4);
561
        gtk_widget_show(hbox);
562
563
        cond_type_optmenu = gtk_option_menu_new();
564
        gtk_widget_show(cond_type_optmenu);
565
        gtk_box_pack_start(GTK_BOX(hbox), cond_type_optmenu, FALSE, FALSE, 0);
566
567
#define COND_MENUITEM_ADD(str, action)                                            \
568
{                                                                            \
569
        MENUITEM_ADD(menu, menuitem, str, action);                            \
570
        g_signal_connect(G_OBJECT(menuitem), "activate",                    \
571
                         G_CALLBACK(prefs_filter_cond_activated_cb),            \
572
                         cond_hbox);                                            \
573
}
574
575
        menu = gtk_menu_new();
576
        gtk_widget_show(menu);
577
        MENUITEM_ADD(menu, menuitem, NULL,        PF_COND_SEPARATOR);
578
        COND_MENUITEM_ADD(_("To or Cc"),          PF_COND_TO_OR_CC);
579
        COND_MENUITEM_ADD(_("Any header"),        PF_COND_ANY_HEADER);
580
        COND_MENUITEM_ADD(_("Edit header..."),    PF_COND_EDIT_HEADER);
581
582
        MENUITEM_ADD(menu, menuitem, NULL,        PF_COND_SEPARATOR);
583
        COND_MENUITEM_ADD(_("Message body"),      PF_COND_BODY);
584
        COND_MENUITEM_ADD(_("Result of command"), PF_COND_CMD_TEST);
585
        COND_MENUITEM_ADD(_("Size"),              PF_COND_SIZE);
586
        COND_MENUITEM_ADD(_("Age"),               PF_COND_AGE);
587
        /* COND_MENUITEM_ADD(_("Account"),           PF_COND_ACCOUNT); */
588
589
        gtk_option_menu_set_menu(GTK_OPTION_MENU(cond_type_optmenu), menu);
590
591
#undef COND_MENUITEM_ADD
592
593
        match_type_optmenu = gtk_option_menu_new();
594
        gtk_widget_show(match_type_optmenu);
595
        gtk_box_pack_start(GTK_BOX(hbox), match_type_optmenu, FALSE, FALSE, 0);
596
597
        menu = gtk_menu_new();
598
        gtk_widget_show(menu);
599
        MENUITEM_ADD(menu, menuitem, _("contains"),
600
                     PF_MATCH_CONTAIN);
601
        MENUITEM_ADD(menu, menuitem, _("doesn't contain"),
602
                     PF_MATCH_NOT_CONTAIN);
603
        MENUITEM_ADD(menu, menuitem, _("is"),
604
                     PF_MATCH_EQUAL);
605
        MENUITEM_ADD(menu, menuitem, _("is not"),
606
                     PF_MATCH_NOT_EQUAL);
607
        MENUITEM_ADD(menu, menuitem, _("match to regex"),
608
                     PF_MATCH_REGEX);
609
        MENUITEM_ADD(menu, menuitem, _("doesn't match to regex"),
610
                     PF_MATCH_NOT_REGEX);
611
        gtk_option_menu_set_menu(GTK_OPTION_MENU(match_type_optmenu), menu);
612
613
        size_match_optmenu = gtk_option_menu_new();
614
        gtk_widget_show(size_match_optmenu);
615
        gtk_box_pack_start(GTK_BOX(hbox), size_match_optmenu, FALSE, FALSE, 0);
616
617
        menu = gtk_menu_new();
618
        gtk_widget_show(menu);
619
        MENUITEM_ADD(menu, menuitem, _("is larger than"), PF_SIZE_LARGER);
620
        MENUITEM_ADD(menu, menuitem, _("is smaller than"), PF_SIZE_SMALLER);
621
        gtk_option_menu_set_menu(GTK_OPTION_MENU(size_match_optmenu), menu);
622
623
        age_match_optmenu = gtk_option_menu_new();
624
        gtk_widget_show(age_match_optmenu);
625
        gtk_box_pack_start(GTK_BOX(hbox), age_match_optmenu, FALSE, FALSE, 0);
626
627
        menu = gtk_menu_new();
628
        gtk_widget_show(menu);
629
        MENUITEM_ADD(menu, menuitem, _("is longer than"), PF_AGE_LONGER);
630
        MENUITEM_ADD(menu, menuitem, _("is shorter than"), PF_AGE_SHORTER);
631
        gtk_option_menu_set_menu(GTK_OPTION_MENU(age_match_optmenu), menu);
632
633
        key_entry = gtk_entry_new();
634
        gtk_widget_show(key_entry);
635
        gtk_box_pack_start(GTK_BOX(hbox), key_entry, TRUE, TRUE, 0);
636
637
        spin_btn_adj = gtk_adjustment_new(0, 0, 99999, 1, 10, 100);
638
        spin_btn = gtk_spin_button_new(GTK_ADJUSTMENT(spin_btn_adj), 1, 0);
639
        gtk_box_pack_start(GTK_BOX(hbox), spin_btn, FALSE, FALSE, 0);
640
        gtk_widget_set_size_request(spin_btn, 64, -1);
641
        gtk_spin_button_set_numeric(GTK_SPIN_BUTTON(spin_btn), TRUE);
642
643
        label = gtk_label_new(_("KB"));
644
        gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
645
646
        del_img = gtk_image_new_from_stock
647
                (GTK_STOCK_REMOVE, GTK_ICON_SIZE_MENU);
648
        gtk_widget_show(del_img);
649
650
        del_btn = gtk_button_new();
651
        gtk_container_add(GTK_CONTAINER(del_btn), del_img);
652
        gtk_widget_show(del_btn);
653
        gtk_box_pack_end(GTK_BOX(hbox), del_btn, FALSE, FALSE, 0);
654
655
        add_img = gtk_image_new_from_stock(GTK_STOCK_ADD, GTK_ICON_SIZE_MENU);
656
        gtk_widget_show(add_img);
657
658
        add_btn = gtk_button_new();
659
        gtk_container_add(GTK_CONTAINER(add_btn), add_img);
660
        gtk_widget_show(add_btn);
661
        gtk_box_pack_end(GTK_BOX(hbox), add_btn, FALSE, FALSE, 0);
662
663
        g_signal_connect(G_OBJECT(del_btn), "clicked",
664
                         G_CALLBACK(prefs_filter_cond_del_cb), cond_hbox);
665
        g_signal_connect(G_OBJECT(add_btn), "clicked",
666
                         G_CALLBACK(prefs_filter_cond_add_cb), cond_hbox);
667
668
        cond_hbox->hbox = hbox;
669
        cond_hbox->cond_type_optmenu = cond_type_optmenu;
670
        cond_hbox->match_type_optmenu = match_type_optmenu;
671
        cond_hbox->size_match_optmenu = size_match_optmenu;
672
        cond_hbox->age_match_optmenu = age_match_optmenu;
673
        cond_hbox->key_entry = key_entry;
674
        cond_hbox->spin_btn = spin_btn;
675
        cond_hbox->label = label;
676
        cond_hbox->del_btn = del_btn;
677
        cond_hbox->add_btn = add_btn;
678
        cond_hbox->cur_type = PF_COND_HEADER;
679
        cond_hbox->cur_header_name = NULL;
680
681
        prefs_filter_edit_set_cond_header_menu(cond_hbox);
682
        gtk_option_menu_set_history(GTK_OPTION_MENU(cond_type_optmenu), 0);
683
684
        return cond_hbox;
685
}
686
687
static ActionHBox *prefs_filter_edit_action_hbox_create(void)
688
{
689
        ActionHBox *action_hbox;
690
        GtkWidget *hbox;
691
        GtkWidget *action_type_optmenu;
692
        GtkWidget *menu;
693
        GtkWidget *menuitem;
694
        GtkWidget *label;
695
        GtkWidget *folder_entry;
696
        GtkWidget *cmd_entry;
697
        GtkWidget *address_entry;
698
        GtkWidget *folder_img;
699
        GtkWidget *folder_sel_btn;
700
        GtkWidget *clabel_optmenu;
701
        GtkWidget *del_btn;
702
        GtkWidget *add_btn;
703
        GtkWidget *del_img;
704
        GtkWidget *add_img;
705
706
        action_hbox = g_new0(ActionHBox, 1);
707
708
        hbox = gtk_hbox_new(FALSE, 4);
709
        gtk_widget_show(hbox);
710
711
        action_type_optmenu = gtk_option_menu_new();
712
        gtk_widget_show(action_type_optmenu);
713
        gtk_box_pack_start(GTK_BOX(hbox), action_type_optmenu, FALSE, FALSE, 0);
714
715
        memset(action_hbox->action_type_menu_items, 0,
716
               sizeof(action_hbox->action_type_menu_items));
717
718
#define ACTION_MENUITEM_ADD(str, action)                                      \
719
{                                                                              \
720
        MENUITEM_ADD(menu, menuitem, str, action);                              \
721
        action_hbox->action_type_menu_items[action] = menuitem;                      \
722
        g_signal_connect(G_OBJECT(menuitem), "activate",                      \
723
                         G_CALLBACK(prefs_filter_action_activated_cb),              \
724
                         action_hbox);                                              \
725
}
726
727
        menu = gtk_menu_new();
728
        gtk_widget_show(menu);
729
        ACTION_MENUITEM_ADD(_("Move to"),               PF_ACTION_MOVE);
730
        ACTION_MENUITEM_ADD(_("Copy to"),               PF_ACTION_COPY);
731
        ACTION_MENUITEM_ADD(_("Don't receive"),         PF_ACTION_NOT_RECEIVE);
732
        ACTION_MENUITEM_ADD(_("Delete from server"),    PF_ACTION_DELETE);
733
734
        MENUITEM_ADD(menu, menuitem, NULL,              PF_ACTION_SEPARATOR);
735
        ACTION_MENUITEM_ADD(_("Set mark"),              PF_ACTION_MARK);
736
        ACTION_MENUITEM_ADD(_("Set color"),             PF_ACTION_COLOR_LABEL);
737
        ACTION_MENUITEM_ADD(_("Mark as read"),          PF_ACTION_MARK_READ);
738
739
#if 0
740
        MENUITEM_ADD(menu, menuitem, NULL,              PF_ACTION_SEPARATOR);
741
        ACTION_MENUITEM_ADD(_("Forward"),               PF_ACTION_FORWARD);
742
        ACTION_MENUITEM_ADD(_("Forward as attachment"), PF_ACTION_FORWARD_AS_ATTACHMENT);
743
        ACTION_MENUITEM_ADD(_("Redirect"),              PF_ACTION_REDIRECT);
744
#endif
745
746
        MENUITEM_ADD(menu, menuitem, NULL,              PF_ACTION_SEPARATOR);
747
        ACTION_MENUITEM_ADD(_("Execute command"),       PF_ACTION_EXEC);
748
749
        MENUITEM_ADD(menu, menuitem, NULL,              PF_ACTION_SEPARATOR);
750
        ACTION_MENUITEM_ADD(_("Stop rule evaluation"),  PF_ACTION_STOP_EVAL);
751
752
        gtk_option_menu_set_menu(GTK_OPTION_MENU(action_type_optmenu), menu);
753
754
#undef ACTION_MENUITEM_ADD
755
756
        label = gtk_label_new(_("folder:"));
757
        gtk_widget_show(label);
758
        gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
759
760
        folder_entry = gtk_entry_new();
761
        gtk_widget_show(folder_entry);
762
        gtk_box_pack_start(GTK_BOX(hbox), folder_entry, TRUE, TRUE, 0);
763
764
        folder_img = stock_pixmap_widget(rule_edit_window.window,
765
                                         STOCK_PIXMAP_DIR_OPEN);
766
        gtk_widget_show(folder_img);
767
768
        folder_sel_btn = gtk_button_new();
769
        gtk_container_add(GTK_CONTAINER(folder_sel_btn), folder_img);
770
        gtk_widget_show(folder_sel_btn);
771
        gtk_box_pack_start(GTK_BOX(hbox), folder_sel_btn, FALSE, FALSE, 0);
772
773
        cmd_entry = gtk_entry_new();
774
        gtk_widget_show(cmd_entry);
775
        gtk_box_pack_start(GTK_BOX(hbox), cmd_entry, TRUE, TRUE, 0);
776
777
        address_entry = gtk_entry_new();
778
        gtk_widget_show(address_entry);
779
        gtk_box_pack_start(GTK_BOX(hbox), address_entry, TRUE, TRUE, 0);
780
781
        clabel_optmenu = gtk_option_menu_new();
782
        gtk_widget_show(clabel_optmenu);
783
        gtk_box_pack_start(GTK_BOX(hbox), clabel_optmenu, FALSE, FALSE, 0);
784
785
        menu = colorlabel_create_color_menu();
786
        gtk_option_menu_set_menu(GTK_OPTION_MENU(clabel_optmenu), menu);
787
788
        del_img = gtk_image_new_from_stock
789
                (GTK_STOCK_REMOVE, GTK_ICON_SIZE_MENU);
790
        gtk_widget_show(del_img);
791
792
        del_btn = gtk_button_new();
793
        gtk_container_add(GTK_CONTAINER(del_btn), del_img);
794
        gtk_widget_show(del_btn);
795
        gtk_box_pack_end(GTK_BOX(hbox), del_btn, FALSE, FALSE, 0);
796
797
        add_img = gtk_image_new_from_stock(GTK_STOCK_ADD, GTK_ICON_SIZE_MENU);
798
        gtk_widget_show(add_img);
799
800
        add_btn = gtk_button_new();
801
        gtk_container_add(GTK_CONTAINER(add_btn), add_img);
802
        gtk_widget_show(add_btn);
803
        gtk_box_pack_end(GTK_BOX(hbox), add_btn, FALSE, FALSE, 0);
804
805
        g_signal_connect(G_OBJECT(folder_sel_btn), "clicked",
806
                         G_CALLBACK(prefs_filter_action_select_dest_cb),
807
                         action_hbox);
808
        g_signal_connect(G_OBJECT(del_btn), "clicked",
809
                         G_CALLBACK(prefs_filter_action_del_cb), action_hbox);
810
        g_signal_connect(G_OBJECT(add_btn), "clicked",
811
                         G_CALLBACK(prefs_filter_action_add_cb), action_hbox);
812
813
        action_hbox->hbox = hbox;
814
        action_hbox->action_type_optmenu = action_type_optmenu;
815
        action_hbox->label = label;
816
        action_hbox->folder_entry = folder_entry;
817
        action_hbox->cmd_entry = cmd_entry;
818
        action_hbox->address_entry = address_entry;
819
        action_hbox->folder_sel_btn = folder_sel_btn;
820
        action_hbox->clabel_optmenu = clabel_optmenu;
821
        action_hbox->del_btn = del_btn;
822
        action_hbox->add_btn = add_btn;
823
824
        return action_hbox;
825
}
826
827
static void prefs_filter_edit_cond_hbox_set(CondHBox *hbox, FilterCond *cond)
828
{
829
        GtkOptionMenu *cond_type_optmenu =
830
                GTK_OPTION_MENU(hbox->cond_type_optmenu);
831
        GtkOptionMenu *match_type_optmenu =
832
                GTK_OPTION_MENU(hbox->match_type_optmenu);
833
        gint cond_index = -1;
834
        gint match_index = -1;
835
        CondMenuType cond_type = PF_COND_NONE;
836
        MatchMenuType match_type = PF_MATCH_NONE;
837
        SizeMatchType size_type = PF_SIZE_LARGER;
838
        AgeMatchType age_type = PF_AGE_LONGER;
839
840
        switch (cond->type) {
841
        case FLT_COND_HEADER:
842
                cond_type = PF_COND_HEADER; break;
843
        case FLT_COND_TO_OR_CC:
844
                cond_type = PF_COND_TO_OR_CC; break;
845
        case FLT_COND_ANY_HEADER:
846
                cond_type = PF_COND_ANY_HEADER; break;
847
        case FLT_COND_BODY:
848
                cond_type = PF_COND_BODY; break;
849
        case FLT_COND_CMD_TEST:
850
                cond_type = PF_COND_CMD_TEST; break;
851
        case FLT_COND_SIZE_GREATER:
852
                cond_type = PF_COND_SIZE;
853
                if (FLT_IS_NOT_MATCH(cond->match_flag))
854
                        size_type = PF_SIZE_SMALLER;
855
                else
856
                        size_type = PF_SIZE_LARGER;
857
                break;
858
        case FLT_COND_AGE_GREATER:
859
                cond_type = PF_COND_AGE;
860
                if (FLT_IS_NOT_MATCH(cond->match_flag))
861
                        age_type = PF_AGE_SHORTER;
862
                else
863
                        age_type = PF_AGE_LONGER;
864
                break;
865
        case FLT_COND_ACCOUNT:
866
                cond_type = PF_COND_ACCOUNT; break;
867
        default:
868
                break;
869
        }
870
871
        switch (cond->type) {
872
        case FLT_COND_HEADER:
873
        case FLT_COND_TO_OR_CC:
874
        case FLT_COND_ANY_HEADER:
875
        case FLT_COND_BODY:
876
                switch (cond->match_type) {
877
                case FLT_CONTAIN:
878
                        if (FLT_IS_NOT_MATCH(cond->match_flag))
879
                                match_type = PF_MATCH_NOT_CONTAIN;
880
                        else
881
                                match_type = PF_MATCH_CONTAIN;
882
                        break;
883
                case FLT_EQUAL:
884
                        if (FLT_IS_NOT_MATCH(cond->match_flag))
885
                                match_type = PF_MATCH_NOT_EQUAL;
886
                        else
887
                                match_type = PF_MATCH_EQUAL;
888
                        break;
889
                case FLT_REGEX:
890
                        if (FLT_IS_NOT_MATCH(cond->match_flag))
891
                                match_type = PF_MATCH_NOT_REGEX;
892
                        else
893
                                match_type = PF_MATCH_REGEX;
894
                        break;
895
                }
896
                break;
897
        default:
898
                break;
899
        }
900
901
        if (cond_type == PF_COND_HEADER)
902
                cond_index = procheader_find_header_list
903
                        (rule_edit_window.hdr_list, cond->header_name);
904
        else
905
                cond_index = menu_find_option_menu_index
906
                        (cond_type_optmenu, GINT_TO_POINTER(cond_type), NULL);
907
        if (cond_index >= 0) {
908
                GtkWidget *menuitem;
909
910
                if (cond_type == PF_COND_SIZE || cond_type == PF_COND_AGE) {
911
                        gtk_spin_button_set_value
912
                                (GTK_SPIN_BUTTON(hbox->spin_btn),
913
                                 (gfloat)cond->int_value);
914
                } else {
915
                        gtk_entry_set_text(GTK_ENTRY(hbox->key_entry),
916
                                           cond->str_value ?
917
                                           cond->str_value : "");
918
                }
919
                gtk_option_menu_set_history(cond_type_optmenu, cond_index);
920
                menuitem = gtk_menu_get_active
921
                        (GTK_MENU(gtk_option_menu_get_menu(cond_type_optmenu)));
922
                gtk_menu_item_activate(GTK_MENU_ITEM(menuitem));
923
        }
924
925
        match_index = menu_find_option_menu_index
926
                (match_type_optmenu, GINT_TO_POINTER(match_type), NULL);
927
        if (match_index >= 0)
928
                gtk_option_menu_set_history(match_type_optmenu, match_index);
929
        if (cond_type == PF_COND_SIZE)
930
                gtk_option_menu_set_history
931
                        (GTK_OPTION_MENU(hbox->size_match_optmenu), size_type);
932
        else if (cond_type == PF_COND_AGE)
933
                gtk_option_menu_set_history
934
                        (GTK_OPTION_MENU(hbox->age_match_optmenu), age_type);
935
}
936
937
static void prefs_filter_edit_action_hbox_set(ActionHBox *hbox,
938
                                              FilterAction *action)
939
{
940
        GtkOptionMenu *type_optmenu = GTK_OPTION_MENU(hbox->action_type_optmenu);
941
        GtkWidget *menu;
942
        ActionMenuType type = PF_ACTION_NONE;
943
944
        menu = gtk_option_menu_get_menu(type_optmenu);
945
946
        switch (action->type) {
947
        case FLT_ACTION_MOVE:
948
                type = PF_ACTION_MOVE; break;
949
        case FLT_ACTION_COPY:
950
                type = PF_ACTION_COPY; break;
951
        case FLT_ACTION_NOT_RECEIVE:
952
                type = PF_ACTION_NOT_RECEIVE; break;
953
        case FLT_ACTION_DELETE:
954
                type = PF_ACTION_DELETE; break;
955
        case FLT_ACTION_EXEC:
956
                type = PF_ACTION_EXEC; break;
957
        case FLT_ACTION_MARK:
958
                type = PF_ACTION_MARK; break;
959
        case FLT_ACTION_COLOR_LABEL:
960
                type = PF_ACTION_COLOR_LABEL; break;
961
        case FLT_ACTION_MARK_READ:
962
                type = PF_ACTION_MARK_READ; break;
963
        case FLT_ACTION_STOP_EVAL:
964
                type = PF_ACTION_STOP_EVAL; break;
965
        default:
966
                break;
967
        }
968
969
        switch (type) {
970
        case PF_ACTION_MOVE:
971
        case PF_ACTION_COPY:
972
                gtk_entry_set_text(GTK_ENTRY(hbox->folder_entry),
973
                                   action->str_value ? action->str_value : "");
974
                break;
975
        case PF_ACTION_EXEC:
976
                gtk_entry_set_text(GTK_ENTRY(hbox->cmd_entry),
977
                                   action->str_value ? action->str_value : "");
978
                break;
979
        case PF_ACTION_COLOR_LABEL:
980
                gtk_option_menu_set_history
981
                        (GTK_OPTION_MENU(hbox->clabel_optmenu),
982
                         action->int_value - 1);
983
                break;
984
        default:
985
                break;
986
        }
987
988
        prefs_filter_edit_set_action_hbox_widgets(hbox, type);
989
}
990
991
static void prefs_filter_edit_cond_hbox_select(CondHBox *hbox,
992
                                               CondMenuType type,
993
                                               const gchar *header_name)
994
{
995
        gint index;
996
        GtkOptionMenu *cond_type_optmenu =
997
                GTK_OPTION_MENU(hbox->cond_type_optmenu);
998
        GtkWidget *menu;
999
1000
        if (type == PF_COND_HEADER) {
1001
                if (header_name)
1002
                        index = procheader_find_header_list
1003
                                (rule_edit_window.hdr_list, header_name);
1004
                else
1005
                        index = 0;
1006
        } else
1007
                index = menu_find_option_menu_index
1008
                        (cond_type_optmenu, GINT_TO_POINTER(type), NULL);
1009
1010
        if (index < 0)
1011
                index = 0;
1012
1013
        menu = gtk_option_menu_get_menu(cond_type_optmenu);
1014
        gtk_option_menu_set_history(cond_type_optmenu, index);
1015
}
1016
1017
static void prefs_filter_edit_set_cond_hbox_widgets(CondHBox *hbox,
1018
                                                    CondMenuType type)
1019
{
1020
        switch (type) {
1021
        case PF_COND_HEADER:
1022
        case PF_COND_TO_OR_CC:
1023
        case PF_COND_ANY_HEADER:
1024
        case PF_COND_BODY:
1025
                gtk_widget_show(hbox->match_type_optmenu);
1026
                gtk_widget_hide(hbox->size_match_optmenu);
1027
                gtk_widget_hide(hbox->age_match_optmenu);
1028
                gtk_widget_show(hbox->key_entry);
1029
                gtk_widget_hide(hbox->spin_btn);
1030
                gtk_widget_hide(hbox->label);
1031
                break;
1032
        case PF_COND_CMD_TEST:
1033
                gtk_widget_hide(hbox->match_type_optmenu);
1034
                gtk_widget_hide(hbox->size_match_optmenu);
1035
                gtk_widget_hide(hbox->age_match_optmenu);
1036
                gtk_widget_show(hbox->key_entry);
1037
                gtk_widget_hide(hbox->spin_btn);
1038
                gtk_widget_hide(hbox->label);
1039
                break;
1040
        case PF_COND_SIZE:
1041
                gtk_widget_hide(hbox->match_type_optmenu);
1042
                gtk_widget_show(hbox->size_match_optmenu);
1043
                gtk_widget_hide(hbox->age_match_optmenu);
1044
                gtk_widget_hide(hbox->key_entry);
1045
                gtk_widget_show(hbox->spin_btn);
1046
                gtk_widget_show(hbox->label);
1047
                break;
1048
        case PF_COND_AGE:
1049
                gtk_widget_hide(hbox->match_type_optmenu);
1050
                gtk_widget_hide(hbox->size_match_optmenu);
1051
                gtk_widget_show(hbox->age_match_optmenu);
1052
                gtk_widget_hide(hbox->key_entry);
1053
                gtk_widget_show(hbox->spin_btn);
1054
                gtk_widget_hide(hbox->label);
1055
                break;
1056
        case PF_COND_ACCOUNT:
1057
                gtk_widget_hide(hbox->match_type_optmenu);
1058
                gtk_widget_hide(hbox->size_match_optmenu);
1059
                gtk_widget_hide(hbox->age_match_optmenu);
1060
                gtk_widget_hide(hbox->key_entry);
1061
                /* gtk_widget_show(hbox->account_optmenu); */
1062
                gtk_widget_hide(hbox->spin_btn);
1063
                gtk_widget_hide(hbox->label);
1064
                break;
1065
        default:
1066
                break;
1067
        }
1068
}
1069
1070
static void prefs_filter_edit_set_action_hbox_widgets(ActionHBox *hbox,
1071
                                                      ActionMenuType type)
1072
{
1073
        GtkOptionMenu *type_optmenu = GTK_OPTION_MENU(hbox->action_type_optmenu);
1074
        gint index;
1075
1076
        switch (type) {
1077
        case PF_ACTION_MOVE:
1078
        case PF_ACTION_COPY:
1079
                gtk_widget_show(hbox->label);
1080
                gtk_label_set_text(GTK_LABEL(hbox->label), _("folder:"));
1081
                gtk_widget_show(hbox->folder_entry);
1082
                gtk_widget_show(hbox->folder_sel_btn);
1083
                gtk_widget_hide(hbox->cmd_entry);
1084
                gtk_widget_hide(hbox->address_entry);
1085
                gtk_widget_hide(hbox->clabel_optmenu);
1086
                break;
1087
        case PF_ACTION_NOT_RECEIVE:
1088
        case PF_ACTION_DELETE:
1089
        case PF_ACTION_MARK:
1090
        case PF_ACTION_MARK_READ:
1091
        case PF_ACTION_STOP_EVAL:
1092
                gtk_widget_hide(hbox->label);
1093
                gtk_widget_hide(hbox->folder_entry);
1094
                gtk_widget_hide(hbox->folder_sel_btn);
1095
                gtk_widget_hide(hbox->cmd_entry);
1096
                gtk_widget_hide(hbox->address_entry);
1097
                gtk_widget_hide(hbox->clabel_optmenu);
1098
                break;
1099
        case PF_ACTION_EXEC:
1100
        case PF_ACTION_EXEC_ASYNC:
1101
                gtk_widget_hide(hbox->label);
1102
                gtk_widget_hide(hbox->folder_entry);
1103
                gtk_widget_hide(hbox->folder_sel_btn);
1104
                gtk_widget_show(hbox->cmd_entry);
1105
                gtk_widget_hide(hbox->address_entry);
1106
                gtk_widget_hide(hbox->clabel_optmenu);
1107
                break;
1108
        case PF_ACTION_COLOR_LABEL:
1109
                gtk_widget_hide(hbox->label);
1110
                gtk_widget_hide(hbox->folder_entry);
1111
                gtk_widget_hide(hbox->folder_sel_btn);
1112
                gtk_widget_hide(hbox->cmd_entry);
1113
                gtk_widget_hide(hbox->address_entry);
1114
                gtk_widget_show(hbox->clabel_optmenu);
1115
                break;
1116
        case PF_ACTION_FORWARD:
1117
        case PF_ACTION_FORWARD_AS_ATTACHMENT:
1118
        case PF_ACTION_REDIRECT:
1119
                gtk_widget_show(hbox->label);
1120
                gtk_label_set_text(GTK_LABEL(hbox->label), _("address:"));
1121
                gtk_widget_hide(hbox->folder_entry);
1122
                gtk_widget_hide(hbox->folder_sel_btn);
1123
                gtk_widget_hide(hbox->cmd_entry);
1124
                gtk_widget_show(hbox->address_entry);
1125
                gtk_widget_hide(hbox->clabel_optmenu);
1126
                break;
1127
        default:
1128
                break;
1129
        }
1130
1131
        index = menu_find_option_menu_index(type_optmenu, GINT_TO_POINTER(type),
1132
                                            NULL);
1133
        gtk_option_menu_set_history(type_optmenu, index);
1134
        prefs_filter_edit_set_action_hbox_menus_sensitive();
1135
}
1136
1137
static void prefs_filter_edit_set_action_hbox_menu_sensitive
1138
        (ActionHBox *hbox, ActionMenuType type, gboolean sensitive)
1139
{
1140
        GtkWidget *menuitem;
1141
1142
        menuitem = hbox->action_type_menu_items[type];
1143
        if (menuitem)
1144
                gtk_widget_set_sensitive(menuitem, sensitive);
1145
}
1146
1147
static void prefs_filter_edit_set_action_hbox_menus_sensitive(void)
1148
{
1149
        GSList *cur;
1150
        ActionHBox *cur_hbox;
1151
        ActionMenuType menu_type;
1152
        ActionMenuType cur_type;
1153
        gboolean action_menu_selection[PF_ACTION_NONE];
1154
        gboolean action_menu_sensitive[PF_ACTION_NONE];
1155
1156
        prefs_filter_edit_get_action_hbox_menus_selection
1157
                (action_menu_selection);
1158
1159
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1160
             cur = cur->next) {
1161
                cur_hbox = (ActionHBox *)cur->data;
1162
                menu_type = prefs_filter_edit_get_action_hbox_type(cur_hbox);
1163
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1164
                     cur_type++)
1165
                        action_menu_sensitive[cur_type] = TRUE;
1166
1167
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1168
                     cur_type++) {
1169
                        switch (cur_type) {
1170
                        case PF_ACTION_MOVE:
1171
                        case PF_ACTION_NOT_RECEIVE:
1172
                        case PF_ACTION_DELETE:
1173
                                if (action_menu_selection[cur_type] == TRUE &&
1174
                                    menu_type != cur_type) {
1175
                                        action_menu_sensitive[PF_ACTION_MOVE] = FALSE;
1176
                                        action_menu_sensitive[PF_ACTION_NOT_RECEIVE] = FALSE;
1177
                                        action_menu_sensitive[PF_ACTION_DELETE] = FALSE;
1178
                                }
1179
                                break;
1180
                        case PF_ACTION_MARK:
1181
                        case PF_ACTION_COLOR_LABEL:
1182
                        case PF_ACTION_MARK_READ:
1183
                        case PF_ACTION_STOP_EVAL:
1184
                                if (action_menu_selection[cur_type] == TRUE &&
1185
                                    menu_type != cur_type)
1186
                                        action_menu_sensitive[cur_type] = FALSE;
1187
                                break;
1188
                        default:
1189
                                break;
1190
                        }
1191
                }
1192
1193
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1194
                     cur_type++) {
1195
                        prefs_filter_edit_set_action_hbox_menu_sensitive
1196
                                (cur_hbox, cur_type,
1197
                                 action_menu_sensitive[cur_type]);
1198
                }
1199
        }
1200
}
1201
1202
static void prefs_filter_edit_get_action_hbox_menus_selection(gboolean *selection)
1203
{
1204
        GSList *cur;
1205
        ActionHBox *cur_hbox;
1206
        ActionMenuType menu_type;
1207
        ActionMenuType cur_type;
1208
1209
        for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE; cur_type++)
1210
                selection[cur_type] = FALSE;
1211
1212
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1213
             cur = cur->next) {
1214
                cur_hbox = (ActionHBox *)cur->data;
1215
                menu_type = prefs_filter_edit_get_action_hbox_type(cur_hbox);
1216
                if (menu_type >= PF_ACTION_MOVE && menu_type < PF_ACTION_NONE)
1217
                        selection[menu_type] = TRUE;
1218
        }
1219
}
1220
1221
static ActionMenuType prefs_filter_edit_get_action_hbox_type(ActionHBox *hbox)
1222
{
1223
        GtkWidget *menuitem;
1224
        ActionMenuType type;
1225
1226
        g_return_val_if_fail(hbox != NULL, PF_ACTION_NONE);
1227
1228
        menuitem = gtk_menu_get_active
1229
                (GTK_MENU(gtk_option_menu_get_menu
1230
                        (GTK_OPTION_MENU(hbox->action_type_optmenu))));
1231
        type = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menuitem),
1232
                                                 MENU_VAL_ID));
1233
1234
        return type;
1235
}
1236
1237
static void prefs_filter_edit_insert_cond_hbox(CondHBox *hbox, gint pos)
1238
{
1239
        g_return_if_fail(hbox != NULL);
1240
1241
        if (!rule_edit_window.cond_hbox_list) {
1242
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1243
        } else if (rule_edit_window.cond_hbox_list &&
1244
                   !rule_edit_window.cond_hbox_list->next) {
1245
                CondHBox *top_hbox =
1246
                        (CondHBox *)rule_edit_window.cond_hbox_list->data;
1247
                gtk_widget_set_sensitive(top_hbox->del_btn, TRUE);
1248
        }
1249
1250
        gtk_box_pack_start(GTK_BOX(rule_edit_window.cond_vbox),
1251
                           hbox->hbox, FALSE, FALSE, 0);
1252
        if (pos >= 0) {
1253
                gtk_box_reorder_child(GTK_BOX(rule_edit_window.cond_vbox),
1254
                                      hbox->hbox, pos);
1255
        }
1256
1257
        rule_edit_window.cond_hbox_list =
1258
                g_slist_insert(rule_edit_window.cond_hbox_list, hbox, pos);
1259
}
1260
1261
static void prefs_filter_edit_insert_action_hbox(ActionHBox *hbox, gint pos)
1262
{
1263
        g_return_if_fail(hbox != NULL);
1264
1265
        if (!rule_edit_window.action_hbox_list) {
1266
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1267
        } else if (rule_edit_window.action_hbox_list &&
1268
                   !rule_edit_window.action_hbox_list->next) {
1269
                ActionHBox *top_hbox =
1270
                        (ActionHBox *)rule_edit_window.action_hbox_list->data;
1271
                gtk_widget_set_sensitive(top_hbox->del_btn, TRUE);
1272
        }
1273
1274
        gtk_box_pack_start(GTK_BOX(rule_edit_window.action_vbox),
1275
                           hbox->hbox, FALSE, FALSE, 0);
1276
        if (pos >= 0) {
1277
                gtk_box_reorder_child(GTK_BOX(rule_edit_window.action_vbox),
1278
                                      hbox->hbox, pos);
1279
        }
1280
1281
        rule_edit_window.action_hbox_list =
1282
                g_slist_insert(rule_edit_window.action_hbox_list, hbox, pos);
1283
}
1284
1285
static void prefs_filter_edit_remove_cond_hbox(CondHBox *hbox)
1286
{
1287
        g_return_if_fail(hbox != NULL);
1288
        g_return_if_fail(rule_edit_window.cond_hbox_list != NULL);
1289
1290
        rule_edit_window.cond_hbox_list =
1291
                g_slist_remove(rule_edit_window.cond_hbox_list, hbox);
1292
        gtk_widget_destroy(hbox->hbox);
1293
        g_free(hbox);
1294
1295
        if (rule_edit_window.cond_hbox_list &&
1296
            !rule_edit_window.cond_hbox_list->next) {
1297
                hbox = (CondHBox *)rule_edit_window.cond_hbox_list->data;
1298
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1299
        }
1300
}
1301
1302
static void prefs_filter_edit_remove_action_hbox(ActionHBox *hbox)
1303
{
1304
        g_return_if_fail(hbox != NULL);
1305
        g_return_if_fail(rule_edit_window.action_hbox_list != NULL);
1306
1307
        rule_edit_window.action_hbox_list =
1308
                g_slist_remove(rule_edit_window.action_hbox_list, hbox);
1309
        gtk_widget_destroy(hbox->hbox);
1310
        g_free(hbox);
1311
1312
        prefs_filter_edit_set_action_hbox_menus_sensitive();
1313
1314
        if (rule_edit_window.action_hbox_list &&
1315
            !rule_edit_window.action_hbox_list->next) {
1316
                hbox = (ActionHBox *)rule_edit_window.action_hbox_list->data;
1317
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1318
        }
1319
}
1320
1321
static void prefs_filter_edit_add_rule_cond(FilterRule *rule)
1322
{
1323
        CondHBox *hbox;
1324
        GSList *cur;
1325
        FilterCond *cond;
1326
1327
        if (!rule || !rule->cond_list) {
1328
                hbox = prefs_filter_edit_cond_hbox_create();
1329
                prefs_filter_edit_set_cond_hbox_widgets(hbox, PF_COND_HEADER);
1330
                prefs_filter_edit_insert_cond_hbox(hbox, -1);
1331
                return;
1332
        }
1333
1334
        for (cur = rule->cond_list; cur != NULL; cur = cur->next) {
1335
                cond = (FilterCond *)cur->data;
1336
1337
                hbox = prefs_filter_edit_cond_hbox_create();
1338
                prefs_filter_edit_cond_hbox_set(hbox, cond);
1339
                prefs_filter_edit_insert_cond_hbox(hbox, -1);
1340
        }
1341
}
1342
1343
static void prefs_filter_edit_add_rule_action(FilterRule *rule)
1344
{
1345
        ActionHBox *hbox;
1346
        GSList *cur;
1347
1348
        if (!rule || !rule->action_list) {
1349
                hbox = prefs_filter_edit_action_hbox_create();
1350
                prefs_filter_edit_insert_action_hbox(hbox, -1);
1351
                prefs_filter_edit_set_action_hbox_widgets(hbox, PF_ACTION_MOVE);
1352
                return;
1353
        }
1354
1355
        for (cur = rule->action_list; cur != NULL; cur = cur->next) {
1356
                FilterAction *action = (FilterAction *)cur->data;
1357
1358
                hbox = prefs_filter_edit_action_hbox_create();
1359
                prefs_filter_edit_insert_action_hbox(hbox, -1);
1360
                prefs_filter_edit_action_hbox_set(hbox, action);
1361
        }
1362
}
1363
1364
static void prefs_filter_edit_set_cond_header_menu(CondHBox *hbox)
1365
{
1366
        GSList *cur;
1367
        GtkWidget *menu;
1368
        GtkWidget *menuitem;
1369
        gint pos = 0;
1370
        GList *child;
1371
1372
        menu = gtk_option_menu_get_menu
1373
                (GTK_OPTION_MENU(hbox->cond_type_optmenu));
1374
1375
        /* destroy header items */
1376
        child = GTK_MENU_SHELL(menu)->children;
1377
        while (child != NULL) {
1378
                GList *next = child->next;
1379
                menuitem = GTK_WIDGET(child->data);
1380
                if (!g_object_get_data(G_OBJECT(menuitem), "header_str"))
1381
                        break;
1382
                gtk_widget_destroy(menuitem);
1383
                child = next;
1384
        }
1385
1386
        for (cur = rule_edit_window.hdr_list; cur != NULL;
1387
             cur = cur->next, pos++) {
1388
                Header *header = (Header *)cur->data;
1389
1390
                menuitem = gtk_menu_item_new_with_label(header->name);
1391
                gtk_widget_show(menuitem);
1392
                gtk_menu_insert(GTK_MENU(menu), menuitem, pos);
1393
                g_object_set_data(G_OBJECT(menuitem), MENU_VAL_ID,
1394
                                  GINT_TO_POINTER(PF_COND_HEADER));
1395
                g_object_set_data(G_OBJECT(menuitem), "header_str",
1396
                                  header->name);
1397
                g_signal_connect(G_OBJECT(menuitem), "activate",
1398
                                 G_CALLBACK(prefs_filter_cond_activated_cb),
1399
                                 hbox);
1400
        }
1401
1402
        if (hbox->cur_type == PF_COND_HEADER)
1403
                prefs_filter_edit_cond_hbox_select
1404
                        (hbox, hbox->cur_type, hbox->cur_header_name);
1405
}
1406
1407
static void prefs_filter_edit_activate_cond_header(const gchar *header)
1408
{
1409
        gint index;
1410
        CondHBox *hbox;
1411
        GtkWidget *menu;
1412
        GtkWidget *menuitem;
1413
        GList *cur;
1414
        gchar *menu_header;
1415
1416
        g_return_if_fail(header != NULL);
1417
        g_return_if_fail(rule_edit_window.cond_hbox_list != NULL);
1418
1419
        hbox = (CondHBox *)rule_edit_window.cond_hbox_list->data;
1420
        menu = gtk_option_menu_get_menu
1421
                (GTK_OPTION_MENU(hbox->cond_type_optmenu));
1422
1423
        for (cur = GTK_MENU_SHELL(menu)->children, index = 0;
1424
             cur != NULL; cur = cur->next, index++) {
1425
                menuitem = GTK_WIDGET(cur->data);
1426
                menu_header = g_object_get_data(G_OBJECT(menuitem),
1427
                                                "header_str");
1428
                if (!menu_header)
1429
                        break;
1430
                if (!g_strcasecmp(menu_header, header)) {
1431
                        gtk_option_menu_set_history
1432
                                (GTK_OPTION_MENU(hbox->cond_type_optmenu),
1433
                                 index);
1434
                        gtk_menu_item_activate(GTK_MENU_ITEM(menuitem));
1435
                        break;
1436
                }
1437
        }
1438
}
1439
1440
static gint edit_header_list_dialog_deleted(GtkWidget *widget,
1441
                                            GdkEventAny *event, gpointer data)
1442
{
1443
        edit_header_list_dialog.finished = TRUE;
1444
        return TRUE;
1445
}
1446
1447
static gboolean edit_header_list_dialog_key_pressed(GtkWidget *widget,
1448
                                                    GdkEventKey *event,
1449
                                                    gpointer data)
1450
{
1451
        if (event && event->keyval == GDK_Escape)
1452
                edit_header_list_dialog.finished = TRUE;
1453
        return FALSE;
1454
}
1455
1456
static void edit_header_list_dialog_add(void)
1457
{
1458
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1459
        const gchar *text;
1460
        gchar *ctext[1];
1461
        gint row;
1462
        gchar *row_text;
1463
1464
        text = gtk_entry_get_text(GTK_ENTRY(edit_header_list_dialog.entry));
1465
        if (text[0] == '\0') return;
1466
1467
        for (row = 0; gtk_clist_get_text(clist, row, 0, &row_text) != 0;
1468
             row++) {
1469
                if (g_strcasecmp(row_text, text) == 0) return;
1470
        }
1471
1472
        ctext[0] = (gchar *)text;
1473
        gtk_clist_append(clist, ctext);
1474
}
1475
1476
static void edit_header_list_dialog_delete(void)
1477
{
1478
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1479
        gint row;
1480
1481
        if (!clist->selection) return;
1482
1483
        row = GPOINTER_TO_INT(clist->selection->data);
1484
        gtk_clist_remove(clist, row);
1485
}
1486
1487
static void edit_header_list_dialog_ok(void)
1488
{
1489
        edit_header_list_dialog.finished = TRUE;
1490
        edit_header_list_dialog.ok = TRUE;
1491
}
1492
1493
static void edit_header_list_dialog_cancel(void)
1494
{
1495
        edit_header_list_dialog.finished = TRUE;
1496
}
1497
1498
static void prefs_filter_edit_edit_header_list_dialog_create(void)
1499
{
1500
        GtkWidget *window;
1501
        GtkWidget *vbox;
1502
        GtkWidget *hbox;
1503
1504
        GtkWidget *vbox2;
1505
        GtkWidget *scrwin;
1506
        GtkWidget *clist;
1507
1508
        GtkWidget *entry_hbox;
1509
        GtkWidget *label;
1510
        GtkWidget *entry;
1511
1512
        GtkWidget *btn_vbox;
1513
        GtkWidget *add_btn;
1514
        GtkWidget *del_btn;
1515
1516
        GtkWidget *confirm_area;
1517
        GtkWidget *ok_btn;
1518
        GtkWidget *cancel_btn;
1519
1520
        gchar *title[1];
1521
1522
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1523
        gtk_container_set_border_width(GTK_CONTAINER(window), 8);
1524
        gtk_window_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
1525
        gtk_window_set_modal(GTK_WINDOW(window), TRUE);
1526
        gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, FALSE);
1527
        gtk_window_set_title(GTK_WINDOW(window), _("Edit header list"));
1528
1529
        g_signal_connect(G_OBJECT(window), "delete_event",
1530
                         G_CALLBACK(edit_header_list_dialog_deleted), NULL);
1531
        g_signal_connect(G_OBJECT(window), "key_press_event",
1532
                         G_CALLBACK(edit_header_list_dialog_key_pressed), NULL);
1533
1534
        vbox = gtk_vbox_new(FALSE, 6);
1535
        gtk_container_add(GTK_CONTAINER(window), vbox);
1536
1537
        hbox = gtk_hbox_new(FALSE, 8);
1538
        gtk_box_pack_start(GTK_BOX(vbox), hbox, TRUE, TRUE, 0);
1539
1540
        vbox2 = gtk_vbox_new(FALSE, 8);
1541
        gtk_box_pack_start(GTK_BOX(hbox), vbox2, TRUE, TRUE, 0);
1542
1543
        scrwin = gtk_scrolled_window_new(NULL, NULL);
1544
        gtk_widget_set_size_request(scrwin, 120, 160);
1545
        gtk_box_pack_start(GTK_BOX(vbox2), scrwin, TRUE, TRUE, 0);
1546
        gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrwin),
1547
                                       GTK_POLICY_AUTOMATIC,
1548
                                       GTK_POLICY_AUTOMATIC);
1549
1550
        title[0] = _("Headers");
1551
        clist = gtk_clist_new_with_titles(1, title);
1552
        gtk_container_add(GTK_CONTAINER(scrwin), clist);
1553
        gtk_clist_set_column_width(GTK_CLIST(clist), 0, 80);
1554
        gtk_clist_set_selection_mode(GTK_CLIST(clist), GTK_SELECTION_BROWSE);
1555
        GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist)->column[0].button,
1556
                               GTK_CAN_FOCUS);
1557
1558
        entry_hbox = gtk_hbox_new(FALSE, 8);
1559
        gtk_box_pack_start(GTK_BOX(vbox), entry_hbox, FALSE, TRUE, 0);
1560
1561
        label = gtk_label_new(_("Header:"));
1562
        gtk_box_pack_start(GTK_BOX(entry_hbox), label, FALSE, FALSE, 0);
1563
1564
        entry = gtk_entry_new();
1565
        gtk_box_pack_start(GTK_BOX(entry_hbox), entry, TRUE, TRUE, 0);
1566
1567
        btn_vbox = gtk_vbox_new(FALSE, 8);
1568
        gtk_box_pack_start(GTK_BOX(hbox), btn_vbox, FALSE, FALSE, 0);
1569
1570
        add_btn = gtk_button_new_with_label(_("Add"));
1571
        gtk_box_pack_start(GTK_BOX(btn_vbox), add_btn, FALSE, FALSE, 0);
1572
1573
        del_btn = gtk_button_new_with_label(_(" Delete "));
1574
        gtk_box_pack_start(GTK_BOX(btn_vbox), del_btn, FALSE, FALSE, 0);
1575
1576
        gtkut_stock_button_set_create(&confirm_area, &ok_btn, GTK_STOCK_OK,
1577
                                      &cancel_btn, GTK_STOCK_CANCEL,
1578
                                      NULL, NULL);
1579
        gtk_box_pack_end(GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
1580
        gtk_widget_grab_default(ok_btn);
1581
1582
        g_signal_connect(G_OBJECT(add_btn), "clicked",
1583
                         G_CALLBACK(edit_header_list_dialog_add), NULL);
1584
        g_signal_connect(G_OBJECT(del_btn), "clicked",
1585
                         G_CALLBACK(edit_header_list_dialog_delete), NULL);
1586
        g_signal_connect(G_OBJECT(ok_btn), "clicked",
1587
                         G_CALLBACK(edit_header_list_dialog_ok), NULL);
1588
        g_signal_connect(G_OBJECT(cancel_btn), "clicked",
1589
                         G_CALLBACK(edit_header_list_dialog_cancel), NULL);
1590
1591
        manage_window_set_transient(GTK_WINDOW(window));
1592
1593
        gtk_widget_show_all(window);
1594
1595
        edit_header_list_dialog.window = window;
1596
        edit_header_list_dialog.clist = clist;
1597
        edit_header_list_dialog.entry = entry;
1598
        edit_header_list_dialog.finished = FALSE;
1599
        edit_header_list_dialog.ok = FALSE;
1600
}
1601
1602
static void prefs_filter_edit_edit_header_list_dialog_set(void)
1603
{
1604
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1605
        GSList *list;
1606
        GSList *cur;
1607
        gchar *text[1];
1608
1609
        gtk_clist_freeze(clist);
1610
1611
        list = prefs_filter_get_user_header_list();
1612
        for (cur = list; cur != NULL; cur = cur->next) {
1613
                Header *header = (Header *)cur->data;
1614
                text[0] = header->name;
1615
                gtk_clist_append(clist, text);
1616
        }
1617
1618
        gtk_clist_thaw(clist);
1619
}
1620
1621
static GSList *prefs_filter_edit_edit_header_list_dialog_get(void)
1622
{
1623
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1624
        gint row;
1625
        gchar *text;
1626
        GSList *list = NULL;
1627
1628
        for (row = 0; gtk_clist_get_text(clist, row, 0, &text) != 0; row++)
1629
                list = procheader_add_header_list(list, text, NULL);
1630
1631
        return list;
1632
}
1633
1634
static void prefs_filter_edit_edit_header_list(void)
1635
{
1636
        GSList *list;
1637
        GSList *cur;
1638
1639
        prefs_filter_edit_edit_header_list_dialog_create();
1640
        prefs_filter_edit_edit_header_list_dialog_set();
1641
1642
        while (edit_header_list_dialog.finished == FALSE)
1643
                gtk_main_iteration();
1644
1645
        if (edit_header_list_dialog.ok == TRUE) {
1646
                list = prefs_filter_edit_edit_header_list_dialog_get();
1647
                prefs_filter_set_user_header_list(list);
1648
                prefs_filter_edit_update_header_list();
1649
                for (cur = rule_edit_window.cond_hbox_list; cur != NULL;
1650
                     cur = cur->next) {
1651
                        CondHBox *hbox = (CondHBox *)cur->data;
1652
                        prefs_filter_edit_set_cond_header_menu(hbox);
1653
                }
1654
        }
1655
1656
        gtk_widget_destroy(edit_header_list_dialog.window);
1657
        edit_header_list_dialog.window = NULL;
1658
        edit_header_list_dialog.clist = NULL;
1659
        edit_header_list_dialog.entry = NULL;
1660
        edit_header_list_dialog.finished = FALSE;
1661
        edit_header_list_dialog.ok = FALSE;
1662
}
1663
1664
static FilterRule *prefs_filter_edit_dialog_to_rule(void)
1665
{
1666
        FilterRule *rule = NULL;
1667
        GSList *cur;
1668
        const gchar *rule_name;
1669
        FilterBoolOp bool_op = FLT_OR;
1670
        GSList *cond_list = NULL;
1671
        GSList *action_list = NULL;
1672
        GtkWidget *bool_op_menuitem;
1673
        gchar *error_msg = NULL;
1674
1675
        rule_name = gtk_entry_get_text(GTK_ENTRY(rule_edit_window.name_entry));
1676
        if (!rule_name || *rule_name == '\0') {
1677
                error_msg = _("Rule name is not specified.");
1678
                goto error;
1679
        }
1680
1681
        bool_op_menuitem = gtk_menu_get_active
1682
                (GTK_MENU(gtk_option_menu_get_menu
1683
                        (GTK_OPTION_MENU(rule_edit_window.bool_op_optmenu))));
1684
        bool_op = GPOINTER_TO_INT
1685
                (g_object_get_data(G_OBJECT(bool_op_menuitem), MENU_VAL_ID));
1686
1687
        for (cur = rule_edit_window.cond_hbox_list; cur != NULL;
1688
             cur = cur->next) {
1689
                CondHBox *hbox = (CondHBox *)cur->data;
1690
                GtkWidget *cond_type_menuitem;
1691
                CondMenuType cond_menu_type;
1692
                MatchMenuType match_menu_type;
1693
                const gchar *header_name;
1694
                const gchar *key_str;
1695
                gint int_value;
1696
                FilterCond *cond = NULL;
1697
                FilterMatchType match_type = FLT_CONTAIN;
1698
                FilterMatchFlag match_flag = 0;
1699
                SizeMatchType size_type;
1700
                AgeMatchType age_type;
1701
1702
                cond_type_menuitem = gtk_menu_get_active
1703
                        (GTK_MENU(gtk_option_menu_get_menu
1704
                                (GTK_OPTION_MENU(hbox->cond_type_optmenu))));
1705
                cond_menu_type = GPOINTER_TO_INT
1706
                        (g_object_get_data
1707
                                (G_OBJECT(cond_type_menuitem), MENU_VAL_ID));
1708
1709
                match_menu_type = menu_get_option_menu_active_index
1710
                        (GTK_OPTION_MENU(hbox->match_type_optmenu));
1711
1712
                key_str = gtk_entry_get_text(GTK_ENTRY(hbox->key_entry));
1713
1714
                switch (match_menu_type) {
1715
                case PF_MATCH_CONTAIN:
1716
                        match_type = FLT_CONTAIN;
1717
                        break;
1718
                case PF_MATCH_NOT_CONTAIN:
1719
                        match_type = FLT_CONTAIN;
1720
                        match_flag |= FLT_NOT_MATCH;
1721
                        break;
1722
                case PF_MATCH_EQUAL:
1723
                        match_type = FLT_EQUAL;
1724
                        break;
1725
                case PF_MATCH_NOT_EQUAL:
1726
                        match_type = FLT_EQUAL;
1727
                        match_flag |= FLT_NOT_MATCH;
1728
                        break;
1729
                case PF_MATCH_REGEX:
1730
                        match_type = FLT_REGEX;
1731
                        break;
1732
                case PF_MATCH_NOT_REGEX:
1733
                        match_type = FLT_REGEX;
1734
                        match_flag |= FLT_NOT_MATCH;
1735
                        break;
1736
                default:
1737
                        break;
1738
                }
1739
1740
                switch (cond_menu_type) {
1741
                case PF_COND_HEADER:
1742
                        header_name = g_object_get_data
1743
                                (G_OBJECT(cond_type_menuitem), "header_str");
1744
                        cond = filter_cond_new(FLT_COND_HEADER,
1745
                                               match_type, match_flag,
1746
                                               header_name, key_str);
1747
                        break;
1748
                case PF_COND_TO_OR_CC:
1749
                        cond = filter_cond_new(FLT_COND_TO_OR_CC, match_type,
1750
                                               match_flag, NULL, key_str);
1751
                        break;
1752
                case PF_COND_ANY_HEADER:
1753
                        cond = filter_cond_new(FLT_COND_ANY_HEADER, match_type,
1754
                                               match_flag, NULL, key_str);
1755
                        break;
1756
                case PF_COND_BODY:
1757
                        cond = filter_cond_new(FLT_COND_BODY, match_type,
1758
                                               match_flag, NULL, key_str);
1759
                        break;
1760
                case PF_COND_CMD_TEST:
1761
                        if (key_str && *key_str)
1762
                                cond = filter_cond_new(FLT_COND_CMD_TEST,
1763
                                                       0, 0, NULL, key_str);
1764
                        else
1765
                                error_msg = _("Command is not specified.");
1766
                        break;
1767
                case PF_COND_SIZE:
1768
                        size_type = menu_get_option_menu_active_index
1769
                                 (GTK_OPTION_MENU(hbox->size_match_optmenu));
1770
                        match_flag = size_type == PF_SIZE_LARGER
1771
                                ? 0 : FLT_NOT_MATCH;
1772
                        int_value = gtk_spin_button_get_value_as_int
1773
                                (GTK_SPIN_BUTTON(hbox->spin_btn));
1774
                        cond = filter_cond_new(FLT_COND_SIZE_GREATER,
1775
                                               0, match_flag, NULL,
1776
                                               itos(int_value));
1777
                        break;
1778
                case PF_COND_AGE:
1779
                        age_type = menu_get_option_menu_active_index
1780
                                 (GTK_OPTION_MENU(hbox->age_match_optmenu));
1781
                        match_flag = age_type == PF_AGE_LONGER
1782
                                ? 0 : FLT_NOT_MATCH;
1783
                        int_value = gtk_spin_button_get_value_as_int
1784
                                (GTK_SPIN_BUTTON(hbox->spin_btn));
1785
                        cond = filter_cond_new(FLT_COND_AGE_GREATER,
1786
                                               0, match_flag, NULL,
1787
                                               itos(int_value));
1788
                        break;
1789
                case PF_COND_ACCOUNT:
1790
                case PF_COND_EDIT_HEADER:
1791
                default:
1792
                        break;
1793
                }
1794
1795
                if (cond)
1796
                        cond_list = g_slist_append(cond_list, cond);
1797
                else {
1798
                        if (!error_msg)
1799
                                error_msg = _("Invalid condition exists.");
1800
                        goto error;
1801
                }
1802
        }
1803
1804
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1805
             cur = cur->next) {
1806
                ActionHBox *hbox = (ActionHBox *)cur->data;
1807
                ActionMenuType action_menu_type;
1808
                const gchar *str;
1809
                guint color;
1810
                FilterAction *action = NULL;
1811
1812
                action_menu_type = prefs_filter_edit_get_action_hbox_type(hbox);
1813
1814
                switch (action_menu_type) {
1815
                case PF_ACTION_MOVE:
1816
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->folder_entry));
1817
                        if (str && *str)
1818
                                action = filter_action_new(FLT_ACTION_MOVE,
1819
                                                           str);
1820
                        else
1821
                                error_msg = _("Destination folder is not specified.");
1822
                        break;
1823
                case PF_ACTION_COPY:
1824
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->folder_entry));
1825
                        if (str && *str)
1826
                                action = filter_action_new(FLT_ACTION_COPY,
1827
                                                           str);
1828
                        else
1829
                                error_msg = _("Destination folder is not specified.");
1830
                        break;
1831
                case PF_ACTION_NOT_RECEIVE:
1832
                        action = filter_action_new(FLT_ACTION_NOT_RECEIVE,
1833
                                                   NULL);
1834
                        break;
1835
                case PF_ACTION_DELETE:
1836
                        action = filter_action_new(FLT_ACTION_DELETE, NULL);
1837
                        break;
1838
                case PF_ACTION_EXEC:
1839
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->cmd_entry));
1840
                        if (str && *str)
1841
                                action = filter_action_new(FLT_ACTION_EXEC,
1842
                                                           str);
1843
                        else
1844
                                error_msg = _("Command is not specified.");
1845
                        break;
1846
                case PF_ACTION_EXEC_ASYNC:
1847
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->cmd_entry));
1848
                        if (str && *str)
1849
                                action = filter_action_new
1850
                                        (FLT_ACTION_EXEC_ASYNC, str);
1851
                        else
1852
                                error_msg = _("Command is not specified.");
1853
                        break;
1854
                case PF_ACTION_MARK:
1855
                        action = filter_action_new(FLT_ACTION_MARK, NULL);
1856
                        break;
1857
                case PF_ACTION_COLOR_LABEL:
1858
                        color = colorlabel_get_color_menu_active_item
1859
                                (gtk_option_menu_get_menu
1860
                                 (GTK_OPTION_MENU(hbox->clabel_optmenu)));
1861
                        action = filter_action_new(FLT_ACTION_COLOR_LABEL,
1862
                                                   itos(color));
1863
                        break;
1864
                case PF_ACTION_MARK_READ:
1865
                        action = filter_action_new(FLT_ACTION_MARK_READ, NULL);
1866
                        break;
1867
                case PF_ACTION_FORWARD:
1868
                case PF_ACTION_FORWARD_AS_ATTACHMENT:
1869
                case PF_ACTION_REDIRECT:
1870
                        break;
1871
                case PF_ACTION_STOP_EVAL:
1872
                        action = filter_action_new(FLT_ACTION_STOP_EVAL, NULL);
1873
                        break;
1874
                case PF_ACTION_SEPARATOR:
1875
                default:
1876
                        break;
1877
                }
1878
1879
                if (action)
1880
                        action_list = g_slist_append(action_list, action);
1881
                else {
1882
                        if (!error_msg)
1883
                                error_msg = _("Invalid action exists.");
1884
                        goto error;
1885
                }
1886
        }
1887
1888
error:
1889
        if (error_msg || !cond_list || !action_list) {
1890
                if (!error_msg) {
1891
                        if (!cond_list)
1892
                                error_msg = _("Condition not exist.");
1893
                        else
1894
                                error_msg = _("Action not exist.");
1895
                }
1896
                alertpanel_error("%s", error_msg);
1897
                if (cond_list)
1898
                        filter_cond_list_free(cond_list);
1899
                if (action_list)
1900
                        filter_action_list_free(action_list);
1901
                return NULL;
1902
        }
1903
1904
        rule = filter_rule_new(rule_name, bool_op, cond_list, action_list);
1905
1906
        return rule;
1907
}
1908
1909
/* callback functions */
1910
1911
static gint prefs_filter_edit_deleted(GtkWidget *widget, GdkEventAny *event,
1912
                                      gpointer data)
1913
{
1914
        prefs_filter_edit_cancel();
1915
        return TRUE;
1916
}
1917
1918
static gboolean prefs_filter_edit_key_pressed(GtkWidget *widget,
1919
                                              GdkEventKey *event,
1920
                                              gpointer data)
1921
{
1922
        if (event && event->keyval == GDK_Escape)
1923
                prefs_filter_edit_cancel();
1924
        return FALSE;
1925
}
1926
1927
static void prefs_filter_edit_ok(void)
1928
{
1929
        FilterRule *rule;
1930
1931
        rule = prefs_filter_edit_dialog_to_rule();
1932
        if (rule) {
1933
                rule_edit_window.new_rule = rule;
1934
                rule_edit_window.edit_finished = TRUE;
1935
        }
1936
}
1937
1938
static void prefs_filter_edit_cancel(void)
1939
{
1940
        rule_edit_window.new_rule = NULL;
1941
        rule_edit_window.edit_finished = TRUE;
1942
}
1943
1944
static void prefs_filter_cond_activated_cb(GtkWidget *widget, gpointer data)
1945
{
1946
        CondHBox *hbox = (CondHBox *)data;
1947
        CondMenuType type;
1948
1949
        type = GPOINTER_TO_INT
1950
                (g_object_get_data(G_OBJECT(widget), MENU_VAL_ID));
1951
1952
        if (type == PF_COND_EDIT_HEADER) {
1953
                prefs_filter_edit_edit_header_list();
1954
                prefs_filter_edit_cond_hbox_select
1955
                        (hbox, hbox->cur_type, hbox->cur_header_name);
1956
        } else {
1957
                hbox->cur_type = type;
1958
                g_free(hbox->cur_header_name);
1959
                hbox->cur_header_name = NULL;
1960
1961
                prefs_filter_edit_set_cond_hbox_widgets(hbox, type);
1962
                if (type == PF_COND_HEADER) {
1963
                        gchar *header_name;
1964
                        gchar *header_field;
1965
1966
                        header_name = (gchar *)g_object_get_data
1967
                                (G_OBJECT(widget), "header_str");
1968
                        header_field = prefs_filter_get_msg_header_field
1969
                                (header_name);
1970
                        if (header_field)
1971
                                gtk_entry_set_text(GTK_ENTRY(hbox->key_entry),
1972
                                                   header_field);
1973
                        hbox->cur_header_name = g_strdup(header_name);
1974
                }
1975
        }
1976
}
1977
1978
static void prefs_filter_action_activated_cb(GtkWidget *widget, gpointer data)
1979
{
1980
        ActionHBox *hbox = (ActionHBox *)data;
1981
        ActionMenuType type;
1982
1983
        type = GPOINTER_TO_INT
1984
                (g_object_get_data(G_OBJECT(widget), MENU_VAL_ID));
1985
        prefs_filter_edit_set_action_hbox_widgets(hbox, type);
1986
}
1987
1988
static void prefs_filter_action_select_dest_cb(GtkWidget *widget, gpointer data)
1989
{
1990
        ActionHBox *hbox = (ActionHBox *)data;
1991
1992
        FolderItem *dest;
1993
        gchar *id;
1994
1995
        dest = foldersel_folder_sel(NULL, FOLDER_SEL_COPY, NULL);
1996
        if (!dest || !dest->path) return;
1997
1998
        id = folder_item_get_identifier(dest);
1999
        if (id) {
2000
                gtk_entry_set_text(GTK_ENTRY(hbox->folder_entry), id);
2001
                g_free(id);
2002
        }
2003
}
2004
2005
static void prefs_filter_cond_del_cb(GtkWidget *widget, gpointer data)
2006
{
2007
        CondHBox *hbox = (CondHBox *)data;
2008
2009
        if (rule_edit_window.cond_hbox_list &&
2010
            rule_edit_window.cond_hbox_list->next)
2011
                prefs_filter_edit_remove_cond_hbox(hbox);
2012
}
2013
2014
static void prefs_filter_cond_add_cb(GtkWidget *widget, gpointer data)
2015
{
2016
        CondHBox *hbox = (CondHBox *)data;
2017
        CondHBox *new_hbox;
2018
        gint index;
2019
2020
        index = g_slist_index(rule_edit_window.cond_hbox_list, hbox);
2021
        g_return_if_fail(index >= 0);
2022
        new_hbox = prefs_filter_edit_cond_hbox_create();
2023
        prefs_filter_edit_set_cond_hbox_widgets(new_hbox, PF_COND_HEADER);
2024
        prefs_filter_edit_insert_cond_hbox(new_hbox, index + 1);
2025
}
2026
2027
static void prefs_filter_action_del_cb(GtkWidget *widget, gpointer data)
2028
{
2029
        ActionHBox *hbox = (ActionHBox *)data;
2030
2031
        if (rule_edit_window.action_hbox_list &&
2032
            rule_edit_window.action_hbox_list->next)
2033
                prefs_filter_edit_remove_action_hbox(hbox);
2034
}
2035
2036
static void prefs_filter_action_add_cb(GtkWidget *widget, gpointer data)
2037
{
2038
        ActionHBox *hbox = (ActionHBox *)data;
2039
        ActionHBox *new_hbox;
2040
        gboolean action_menu_selection[PF_ACTION_NONE];
2041
        gint index;
2042
2043
        prefs_filter_edit_get_action_hbox_menus_selection(action_menu_selection);
2044
2045
        index = g_slist_index(rule_edit_window.action_hbox_list, hbox);
2046
        g_return_if_fail(index >= 0);
2047
        new_hbox = prefs_filter_edit_action_hbox_create();
2048
        prefs_filter_edit_insert_action_hbox(new_hbox, index + 1);
2049
        if (action_menu_selection[PF_ACTION_MOVE] == TRUE ||
2050
            action_menu_selection[PF_ACTION_NOT_RECEIVE] == TRUE ||
2051
            action_menu_selection[PF_ACTION_DELETE] == TRUE)
2052
                prefs_filter_edit_set_action_hbox_widgets(new_hbox,
2053
                                                          PF_ACTION_COPY);
2054
        else
2055
                prefs_filter_edit_set_action_hbox_widgets(new_hbox,
2056
                                                          PF_ACTION_MOVE);
2057
}