Statistics
| Revision:

root / src / prefs_filter_edit.c @ 424

History | View | Annotate | Download (58 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
                gtk_label_set_text(GTK_LABEL(hbox->label), _("KB"));
1048
                break;
1049
        case PF_COND_AGE:
1050
                gtk_widget_hide(hbox->match_type_optmenu);
1051
                gtk_widget_hide(hbox->size_match_optmenu);
1052
                gtk_widget_show(hbox->age_match_optmenu);
1053
                gtk_widget_hide(hbox->key_entry);
1054
                gtk_widget_show(hbox->spin_btn);
1055
                gtk_widget_show(hbox->label);
1056
                gtk_label_set_text(GTK_LABEL(hbox->label), _("day(s)"));
1057
                break;
1058
        case PF_COND_ACCOUNT:
1059
                gtk_widget_hide(hbox->match_type_optmenu);
1060
                gtk_widget_hide(hbox->size_match_optmenu);
1061
                gtk_widget_hide(hbox->age_match_optmenu);
1062
                gtk_widget_hide(hbox->key_entry);
1063
                /* gtk_widget_show(hbox->account_optmenu); */
1064
                gtk_widget_hide(hbox->spin_btn);
1065
                gtk_widget_hide(hbox->label);
1066
                break;
1067
        default:
1068
                break;
1069
        }
1070
}
1071
1072
static void prefs_filter_edit_set_action_hbox_widgets(ActionHBox *hbox,
1073
                                                      ActionMenuType type)
1074
{
1075
        GtkOptionMenu *type_optmenu = GTK_OPTION_MENU(hbox->action_type_optmenu);
1076
        gint index;
1077
1078
        switch (type) {
1079
        case PF_ACTION_MOVE:
1080
        case PF_ACTION_COPY:
1081
                gtk_widget_show(hbox->label);
1082
                gtk_label_set_text(GTK_LABEL(hbox->label), _("folder:"));
1083
                gtk_widget_show(hbox->folder_entry);
1084
                gtk_widget_show(hbox->folder_sel_btn);
1085
                gtk_widget_hide(hbox->cmd_entry);
1086
                gtk_widget_hide(hbox->address_entry);
1087
                gtk_widget_hide(hbox->clabel_optmenu);
1088
                break;
1089
        case PF_ACTION_NOT_RECEIVE:
1090
        case PF_ACTION_DELETE:
1091
        case PF_ACTION_MARK:
1092
        case PF_ACTION_MARK_READ:
1093
        case PF_ACTION_STOP_EVAL:
1094
                gtk_widget_hide(hbox->label);
1095
                gtk_widget_hide(hbox->folder_entry);
1096
                gtk_widget_hide(hbox->folder_sel_btn);
1097
                gtk_widget_hide(hbox->cmd_entry);
1098
                gtk_widget_hide(hbox->address_entry);
1099
                gtk_widget_hide(hbox->clabel_optmenu);
1100
                break;
1101
        case PF_ACTION_EXEC:
1102
        case PF_ACTION_EXEC_ASYNC:
1103
                gtk_widget_hide(hbox->label);
1104
                gtk_widget_hide(hbox->folder_entry);
1105
                gtk_widget_hide(hbox->folder_sel_btn);
1106
                gtk_widget_show(hbox->cmd_entry);
1107
                gtk_widget_hide(hbox->address_entry);
1108
                gtk_widget_hide(hbox->clabel_optmenu);
1109
                break;
1110
        case PF_ACTION_COLOR_LABEL:
1111
                gtk_widget_hide(hbox->label);
1112
                gtk_widget_hide(hbox->folder_entry);
1113
                gtk_widget_hide(hbox->folder_sel_btn);
1114
                gtk_widget_hide(hbox->cmd_entry);
1115
                gtk_widget_hide(hbox->address_entry);
1116
                gtk_widget_show(hbox->clabel_optmenu);
1117
                break;
1118
        case PF_ACTION_FORWARD:
1119
        case PF_ACTION_FORWARD_AS_ATTACHMENT:
1120
        case PF_ACTION_REDIRECT:
1121
                gtk_widget_show(hbox->label);
1122
                gtk_label_set_text(GTK_LABEL(hbox->label), _("address:"));
1123
                gtk_widget_hide(hbox->folder_entry);
1124
                gtk_widget_hide(hbox->folder_sel_btn);
1125
                gtk_widget_hide(hbox->cmd_entry);
1126
                gtk_widget_show(hbox->address_entry);
1127
                gtk_widget_hide(hbox->clabel_optmenu);
1128
                break;
1129
        default:
1130
                break;
1131
        }
1132
1133
        index = menu_find_option_menu_index(type_optmenu, GINT_TO_POINTER(type),
1134
                                            NULL);
1135
        gtk_option_menu_set_history(type_optmenu, index);
1136
        prefs_filter_edit_set_action_hbox_menus_sensitive();
1137
}
1138
1139
static void prefs_filter_edit_set_action_hbox_menu_sensitive
1140
        (ActionHBox *hbox, ActionMenuType type, gboolean sensitive)
1141
{
1142
        GtkWidget *menuitem;
1143
1144
        menuitem = hbox->action_type_menu_items[type];
1145
        if (menuitem)
1146
                gtk_widget_set_sensitive(menuitem, sensitive);
1147
}
1148
1149
static void prefs_filter_edit_set_action_hbox_menus_sensitive(void)
1150
{
1151
        GSList *cur;
1152
        ActionHBox *cur_hbox;
1153
        ActionMenuType menu_type;
1154
        ActionMenuType cur_type;
1155
        gboolean action_menu_selection[PF_ACTION_NONE];
1156
        gboolean action_menu_sensitive[PF_ACTION_NONE];
1157
1158
        prefs_filter_edit_get_action_hbox_menus_selection
1159
                (action_menu_selection);
1160
1161
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1162
             cur = cur->next) {
1163
                cur_hbox = (ActionHBox *)cur->data;
1164
                menu_type = prefs_filter_edit_get_action_hbox_type(cur_hbox);
1165
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1166
                     cur_type++)
1167
                        action_menu_sensitive[cur_type] = TRUE;
1168
1169
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1170
                     cur_type++) {
1171
                        switch (cur_type) {
1172
                        case PF_ACTION_MOVE:
1173
                        case PF_ACTION_NOT_RECEIVE:
1174
                        case PF_ACTION_DELETE:
1175
                                if (action_menu_selection[cur_type] == TRUE &&
1176
                                    menu_type != cur_type) {
1177
                                        action_menu_sensitive[PF_ACTION_MOVE] = FALSE;
1178
                                        action_menu_sensitive[PF_ACTION_NOT_RECEIVE] = FALSE;
1179
                                        action_menu_sensitive[PF_ACTION_DELETE] = FALSE;
1180
                                }
1181
                                break;
1182
                        case PF_ACTION_MARK:
1183
                        case PF_ACTION_COLOR_LABEL:
1184
                        case PF_ACTION_MARK_READ:
1185
                        case PF_ACTION_STOP_EVAL:
1186
                                if (action_menu_selection[cur_type] == TRUE &&
1187
                                    menu_type != cur_type)
1188
                                        action_menu_sensitive[cur_type] = FALSE;
1189
                                break;
1190
                        default:
1191
                                break;
1192
                        }
1193
                }
1194
1195
                for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE;
1196
                     cur_type++) {
1197
                        prefs_filter_edit_set_action_hbox_menu_sensitive
1198
                                (cur_hbox, cur_type,
1199
                                 action_menu_sensitive[cur_type]);
1200
                }
1201
        }
1202
}
1203
1204
static void prefs_filter_edit_get_action_hbox_menus_selection(gboolean *selection)
1205
{
1206
        GSList *cur;
1207
        ActionHBox *cur_hbox;
1208
        ActionMenuType menu_type;
1209
        ActionMenuType cur_type;
1210
1211
        for (cur_type = PF_ACTION_MOVE; cur_type < PF_ACTION_NONE; cur_type++)
1212
                selection[cur_type] = FALSE;
1213
1214
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1215
             cur = cur->next) {
1216
                cur_hbox = (ActionHBox *)cur->data;
1217
                menu_type = prefs_filter_edit_get_action_hbox_type(cur_hbox);
1218
                if (menu_type >= PF_ACTION_MOVE && menu_type < PF_ACTION_NONE)
1219
                        selection[menu_type] = TRUE;
1220
        }
1221
}
1222
1223
static ActionMenuType prefs_filter_edit_get_action_hbox_type(ActionHBox *hbox)
1224
{
1225
        GtkWidget *menuitem;
1226
        ActionMenuType type;
1227
1228
        g_return_val_if_fail(hbox != NULL, PF_ACTION_NONE);
1229
1230
        menuitem = gtk_menu_get_active
1231
                (GTK_MENU(gtk_option_menu_get_menu
1232
                        (GTK_OPTION_MENU(hbox->action_type_optmenu))));
1233
        type = GPOINTER_TO_INT(g_object_get_data(G_OBJECT(menuitem),
1234
                                                 MENU_VAL_ID));
1235
1236
        return type;
1237
}
1238
1239
static void prefs_filter_edit_insert_cond_hbox(CondHBox *hbox, gint pos)
1240
{
1241
        g_return_if_fail(hbox != NULL);
1242
1243
        if (!rule_edit_window.cond_hbox_list) {
1244
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1245
        } else if (rule_edit_window.cond_hbox_list &&
1246
                   !rule_edit_window.cond_hbox_list->next) {
1247
                CondHBox *top_hbox =
1248
                        (CondHBox *)rule_edit_window.cond_hbox_list->data;
1249
                gtk_widget_set_sensitive(top_hbox->del_btn, TRUE);
1250
        }
1251
1252
        gtk_box_pack_start(GTK_BOX(rule_edit_window.cond_vbox),
1253
                           hbox->hbox, FALSE, FALSE, 0);
1254
        if (pos >= 0) {
1255
                gtk_box_reorder_child(GTK_BOX(rule_edit_window.cond_vbox),
1256
                                      hbox->hbox, pos);
1257
        }
1258
1259
        rule_edit_window.cond_hbox_list =
1260
                g_slist_insert(rule_edit_window.cond_hbox_list, hbox, pos);
1261
}
1262
1263
static void prefs_filter_edit_insert_action_hbox(ActionHBox *hbox, gint pos)
1264
{
1265
        g_return_if_fail(hbox != NULL);
1266
1267
        if (!rule_edit_window.action_hbox_list) {
1268
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1269
        } else if (rule_edit_window.action_hbox_list &&
1270
                   !rule_edit_window.action_hbox_list->next) {
1271
                ActionHBox *top_hbox =
1272
                        (ActionHBox *)rule_edit_window.action_hbox_list->data;
1273
                gtk_widget_set_sensitive(top_hbox->del_btn, TRUE);
1274
        }
1275
1276
        gtk_box_pack_start(GTK_BOX(rule_edit_window.action_vbox),
1277
                           hbox->hbox, FALSE, FALSE, 0);
1278
        if (pos >= 0) {
1279
                gtk_box_reorder_child(GTK_BOX(rule_edit_window.action_vbox),
1280
                                      hbox->hbox, pos);
1281
        }
1282
1283
        rule_edit_window.action_hbox_list =
1284
                g_slist_insert(rule_edit_window.action_hbox_list, hbox, pos);
1285
}
1286
1287
static void prefs_filter_edit_remove_cond_hbox(CondHBox *hbox)
1288
{
1289
        g_return_if_fail(hbox != NULL);
1290
        g_return_if_fail(rule_edit_window.cond_hbox_list != NULL);
1291
1292
        rule_edit_window.cond_hbox_list =
1293
                g_slist_remove(rule_edit_window.cond_hbox_list, hbox);
1294
        gtk_widget_destroy(hbox->hbox);
1295
        g_free(hbox);
1296
1297
        if (rule_edit_window.cond_hbox_list &&
1298
            !rule_edit_window.cond_hbox_list->next) {
1299
                hbox = (CondHBox *)rule_edit_window.cond_hbox_list->data;
1300
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1301
        }
1302
}
1303
1304
static void prefs_filter_edit_remove_action_hbox(ActionHBox *hbox)
1305
{
1306
        g_return_if_fail(hbox != NULL);
1307
        g_return_if_fail(rule_edit_window.action_hbox_list != NULL);
1308
1309
        rule_edit_window.action_hbox_list =
1310
                g_slist_remove(rule_edit_window.action_hbox_list, hbox);
1311
        gtk_widget_destroy(hbox->hbox);
1312
        g_free(hbox);
1313
1314
        prefs_filter_edit_set_action_hbox_menus_sensitive();
1315
1316
        if (rule_edit_window.action_hbox_list &&
1317
            !rule_edit_window.action_hbox_list->next) {
1318
                hbox = (ActionHBox *)rule_edit_window.action_hbox_list->data;
1319
                gtk_widget_set_sensitive(hbox->del_btn, FALSE);
1320
        }
1321
}
1322
1323
static void prefs_filter_edit_add_rule_cond(FilterRule *rule)
1324
{
1325
        CondHBox *hbox;
1326
        GSList *cur;
1327
        FilterCond *cond;
1328
1329
        if (!rule || !rule->cond_list) {
1330
                hbox = prefs_filter_edit_cond_hbox_create();
1331
                prefs_filter_edit_set_cond_hbox_widgets(hbox, PF_COND_HEADER);
1332
                prefs_filter_edit_insert_cond_hbox(hbox, -1);
1333
                return;
1334
        }
1335
1336
        for (cur = rule->cond_list; cur != NULL; cur = cur->next) {
1337
                cond = (FilterCond *)cur->data;
1338
1339
                hbox = prefs_filter_edit_cond_hbox_create();
1340
                prefs_filter_edit_cond_hbox_set(hbox, cond);
1341
                prefs_filter_edit_insert_cond_hbox(hbox, -1);
1342
        }
1343
}
1344
1345
static void prefs_filter_edit_add_rule_action(FilterRule *rule)
1346
{
1347
        ActionHBox *hbox;
1348
        GSList *cur;
1349
1350
        if (!rule || !rule->action_list) {
1351
                hbox = prefs_filter_edit_action_hbox_create();
1352
                prefs_filter_edit_insert_action_hbox(hbox, -1);
1353
                prefs_filter_edit_set_action_hbox_widgets(hbox, PF_ACTION_MOVE);
1354
                return;
1355
        }
1356
1357
        for (cur = rule->action_list; cur != NULL; cur = cur->next) {
1358
                FilterAction *action = (FilterAction *)cur->data;
1359
1360
                hbox = prefs_filter_edit_action_hbox_create();
1361
                prefs_filter_edit_insert_action_hbox(hbox, -1);
1362
                prefs_filter_edit_action_hbox_set(hbox, action);
1363
        }
1364
}
1365
1366
static void prefs_filter_edit_set_cond_header_menu(CondHBox *hbox)
1367
{
1368
        GSList *cur;
1369
        GtkWidget *menu;
1370
        GtkWidget *menuitem;
1371
        gint pos = 0;
1372
        GList *child;
1373
1374
        menu = gtk_option_menu_get_menu
1375
                (GTK_OPTION_MENU(hbox->cond_type_optmenu));
1376
1377
        /* destroy header items */
1378
        child = GTK_MENU_SHELL(menu)->children;
1379
        while (child != NULL) {
1380
                GList *next = child->next;
1381
                menuitem = GTK_WIDGET(child->data);
1382
                if (!g_object_get_data(G_OBJECT(menuitem), "header_str"))
1383
                        break;
1384
                gtk_widget_destroy(menuitem);
1385
                child = next;
1386
        }
1387
1388
        for (cur = rule_edit_window.hdr_list; cur != NULL;
1389
             cur = cur->next, pos++) {
1390
                Header *header = (Header *)cur->data;
1391
1392
                menuitem = gtk_menu_item_new_with_label(header->name);
1393
                gtk_widget_show(menuitem);
1394
                gtk_menu_insert(GTK_MENU(menu), menuitem, pos);
1395
                g_object_set_data(G_OBJECT(menuitem), MENU_VAL_ID,
1396
                                  GINT_TO_POINTER(PF_COND_HEADER));
1397
                g_object_set_data(G_OBJECT(menuitem), "header_str",
1398
                                  header->name);
1399
                g_signal_connect(G_OBJECT(menuitem), "activate",
1400
                                 G_CALLBACK(prefs_filter_cond_activated_cb),
1401
                                 hbox);
1402
        }
1403
1404
        if (hbox->cur_type == PF_COND_HEADER)
1405
                prefs_filter_edit_cond_hbox_select
1406
                        (hbox, hbox->cur_type, hbox->cur_header_name);
1407
}
1408
1409
static void prefs_filter_edit_activate_cond_header(const gchar *header)
1410
{
1411
        gint index;
1412
        CondHBox *hbox;
1413
        GtkWidget *menu;
1414
        GtkWidget *menuitem;
1415
        GList *cur;
1416
        gchar *menu_header;
1417
1418
        g_return_if_fail(header != NULL);
1419
        g_return_if_fail(rule_edit_window.cond_hbox_list != NULL);
1420
1421
        hbox = (CondHBox *)rule_edit_window.cond_hbox_list->data;
1422
        menu = gtk_option_menu_get_menu
1423
                (GTK_OPTION_MENU(hbox->cond_type_optmenu));
1424
1425
        for (cur = GTK_MENU_SHELL(menu)->children, index = 0;
1426
             cur != NULL; cur = cur->next, index++) {
1427
                menuitem = GTK_WIDGET(cur->data);
1428
                menu_header = g_object_get_data(G_OBJECT(menuitem),
1429
                                                "header_str");
1430
                if (!menu_header)
1431
                        break;
1432
                if (!g_ascii_strcasecmp(menu_header, header)) {
1433
                        gtk_option_menu_set_history
1434
                                (GTK_OPTION_MENU(hbox->cond_type_optmenu),
1435
                                 index);
1436
                        gtk_menu_item_activate(GTK_MENU_ITEM(menuitem));
1437
                        break;
1438
                }
1439
        }
1440
}
1441
1442
static gint edit_header_list_dialog_deleted(GtkWidget *widget,
1443
                                            GdkEventAny *event, gpointer data)
1444
{
1445
        edit_header_list_dialog.finished = TRUE;
1446
        return TRUE;
1447
}
1448
1449
static gboolean edit_header_list_dialog_key_pressed(GtkWidget *widget,
1450
                                                    GdkEventKey *event,
1451
                                                    gpointer data)
1452
{
1453
        if (event && event->keyval == GDK_Escape)
1454
                edit_header_list_dialog.finished = TRUE;
1455
        return FALSE;
1456
}
1457
1458
static void edit_header_list_dialog_add(void)
1459
{
1460
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1461
        const gchar *text;
1462
        gchar *ctext[1];
1463
        gint row;
1464
        gchar *row_text;
1465
1466
        text = gtk_entry_get_text(GTK_ENTRY(edit_header_list_dialog.entry));
1467
        if (text[0] == '\0') return;
1468
1469
        for (row = 0; gtk_clist_get_text(clist, row, 0, &row_text) != 0;
1470
             row++) {
1471
                if (g_ascii_strcasecmp(row_text, text) == 0) return;
1472
        }
1473
1474
        ctext[0] = (gchar *)text;
1475
        gtk_clist_append(clist, ctext);
1476
}
1477
1478
static void edit_header_list_dialog_delete(void)
1479
{
1480
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1481
        gint row;
1482
1483
        if (!clist->selection) return;
1484
1485
        row = GPOINTER_TO_INT(clist->selection->data);
1486
        gtk_clist_remove(clist, row);
1487
}
1488
1489
static void edit_header_list_dialog_ok(void)
1490
{
1491
        edit_header_list_dialog.finished = TRUE;
1492
        edit_header_list_dialog.ok = TRUE;
1493
}
1494
1495
static void edit_header_list_dialog_cancel(void)
1496
{
1497
        edit_header_list_dialog.finished = TRUE;
1498
}
1499
1500
static void prefs_filter_edit_edit_header_list_dialog_create(void)
1501
{
1502
        GtkWidget *window;
1503
        GtkWidget *vbox;
1504
        GtkWidget *hbox;
1505
1506
        GtkWidget *vbox2;
1507
        GtkWidget *scrwin;
1508
        GtkWidget *clist;
1509
1510
        GtkWidget *entry_hbox;
1511
        GtkWidget *label;
1512
        GtkWidget *entry;
1513
1514
        GtkWidget *btn_vbox;
1515
        GtkWidget *add_btn;
1516
        GtkWidget *del_btn;
1517
1518
        GtkWidget *confirm_area;
1519
        GtkWidget *ok_btn;
1520
        GtkWidget *cancel_btn;
1521
1522
        gchar *title[1];
1523
1524
        window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1525
        gtk_container_set_border_width(GTK_CONTAINER(window), 8);
1526
        gtk_window_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER);
1527
        gtk_window_set_modal(GTK_WINDOW(window), TRUE);
1528
        gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, FALSE);
1529
        gtk_window_set_title(GTK_WINDOW(window), _("Edit header list"));
1530
1531
        g_signal_connect(G_OBJECT(window), "delete_event",
1532
                         G_CALLBACK(edit_header_list_dialog_deleted), NULL);
1533
        g_signal_connect(G_OBJECT(window), "key_press_event",
1534
                         G_CALLBACK(edit_header_list_dialog_key_pressed), NULL);
1535
1536
        vbox = gtk_vbox_new(FALSE, 6);
1537
        gtk_container_add(GTK_CONTAINER(window), vbox);
1538
1539
        hbox = gtk_hbox_new(FALSE, 8);
1540
        gtk_box_pack_start(GTK_BOX(vbox), hbox, TRUE, TRUE, 0);
1541
1542
        vbox2 = gtk_vbox_new(FALSE, 8);
1543
        gtk_box_pack_start(GTK_BOX(hbox), vbox2, TRUE, TRUE, 0);
1544
1545
        scrwin = gtk_scrolled_window_new(NULL, NULL);
1546
        gtk_widget_set_size_request(scrwin, 120, 160);
1547
        gtk_box_pack_start(GTK_BOX(vbox2), scrwin, TRUE, TRUE, 0);
1548
        gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrwin),
1549
                                       GTK_POLICY_AUTOMATIC,
1550
                                       GTK_POLICY_AUTOMATIC);
1551
1552
        title[0] = _("Headers");
1553
        clist = gtk_clist_new_with_titles(1, title);
1554
        gtk_container_add(GTK_CONTAINER(scrwin), clist);
1555
        gtk_clist_set_column_width(GTK_CLIST(clist), 0, 80);
1556
        gtk_clist_set_selection_mode(GTK_CLIST(clist), GTK_SELECTION_BROWSE);
1557
        GTK_WIDGET_UNSET_FLAGS(GTK_CLIST(clist)->column[0].button,
1558
                               GTK_CAN_FOCUS);
1559
1560
        entry_hbox = gtk_hbox_new(FALSE, 8);
1561
        gtk_box_pack_start(GTK_BOX(vbox), entry_hbox, FALSE, TRUE, 0);
1562
1563
        label = gtk_label_new(_("Header:"));
1564
        gtk_box_pack_start(GTK_BOX(entry_hbox), label, FALSE, FALSE, 0);
1565
1566
        entry = gtk_entry_new();
1567
        gtk_box_pack_start(GTK_BOX(entry_hbox), entry, TRUE, TRUE, 0);
1568
1569
        btn_vbox = gtk_vbox_new(FALSE, 8);
1570
        gtk_box_pack_start(GTK_BOX(hbox), btn_vbox, FALSE, FALSE, 0);
1571
1572
        add_btn = gtk_button_new_with_label(_("Add"));
1573
        gtk_box_pack_start(GTK_BOX(btn_vbox), add_btn, FALSE, FALSE, 0);
1574
1575
        del_btn = gtk_button_new_with_label(_(" Delete "));
1576
        gtk_box_pack_start(GTK_BOX(btn_vbox), del_btn, FALSE, FALSE, 0);
1577
1578
        gtkut_stock_button_set_create(&confirm_area, &ok_btn, GTK_STOCK_OK,
1579
                                      &cancel_btn, GTK_STOCK_CANCEL,
1580
                                      NULL, NULL);
1581
        gtk_box_pack_end(GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
1582
        gtk_widget_grab_default(ok_btn);
1583
1584
        g_signal_connect(G_OBJECT(add_btn), "clicked",
1585
                         G_CALLBACK(edit_header_list_dialog_add), NULL);
1586
        g_signal_connect(G_OBJECT(del_btn), "clicked",
1587
                         G_CALLBACK(edit_header_list_dialog_delete), NULL);
1588
        g_signal_connect(G_OBJECT(ok_btn), "clicked",
1589
                         G_CALLBACK(edit_header_list_dialog_ok), NULL);
1590
        g_signal_connect(G_OBJECT(cancel_btn), "clicked",
1591
                         G_CALLBACK(edit_header_list_dialog_cancel), NULL);
1592
1593
        manage_window_set_transient(GTK_WINDOW(window));
1594
1595
        gtk_widget_show_all(window);
1596
1597
        edit_header_list_dialog.window = window;
1598
        edit_header_list_dialog.clist = clist;
1599
        edit_header_list_dialog.entry = entry;
1600
        edit_header_list_dialog.finished = FALSE;
1601
        edit_header_list_dialog.ok = FALSE;
1602
}
1603
1604
static void prefs_filter_edit_edit_header_list_dialog_set(void)
1605
{
1606
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1607
        GSList *list;
1608
        GSList *cur;
1609
        gchar *text[1];
1610
1611
        gtk_clist_freeze(clist);
1612
1613
        list = prefs_filter_get_user_header_list();
1614
        for (cur = list; cur != NULL; cur = cur->next) {
1615
                Header *header = (Header *)cur->data;
1616
                text[0] = header->name;
1617
                gtk_clist_append(clist, text);
1618
        }
1619
1620
        gtk_clist_thaw(clist);
1621
}
1622
1623
static GSList *prefs_filter_edit_edit_header_list_dialog_get(void)
1624
{
1625
        GtkCList *clist = GTK_CLIST(edit_header_list_dialog.clist);
1626
        gint row;
1627
        gchar *text;
1628
        GSList *list = NULL;
1629
1630
        for (row = 0; gtk_clist_get_text(clist, row, 0, &text) != 0; row++)
1631
                list = procheader_add_header_list(list, text, NULL);
1632
1633
        return list;
1634
}
1635
1636
static void prefs_filter_edit_edit_header_list(void)
1637
{
1638
        GSList *list;
1639
        GSList *cur;
1640
1641
        prefs_filter_edit_edit_header_list_dialog_create();
1642
        prefs_filter_edit_edit_header_list_dialog_set();
1643
1644
        while (edit_header_list_dialog.finished == FALSE)
1645
                gtk_main_iteration();
1646
1647
        if (edit_header_list_dialog.ok == TRUE) {
1648
                list = prefs_filter_edit_edit_header_list_dialog_get();
1649
                prefs_filter_set_user_header_list(list);
1650
                prefs_filter_edit_update_header_list();
1651
                for (cur = rule_edit_window.cond_hbox_list; cur != NULL;
1652
                     cur = cur->next) {
1653
                        CondHBox *hbox = (CondHBox *)cur->data;
1654
                        prefs_filter_edit_set_cond_header_menu(hbox);
1655
                }
1656
        }
1657
1658
        gtk_widget_destroy(edit_header_list_dialog.window);
1659
        edit_header_list_dialog.window = NULL;
1660
        edit_header_list_dialog.clist = NULL;
1661
        edit_header_list_dialog.entry = NULL;
1662
        edit_header_list_dialog.finished = FALSE;
1663
        edit_header_list_dialog.ok = FALSE;
1664
}
1665
1666
static FilterRule *prefs_filter_edit_dialog_to_rule(void)
1667
{
1668
        FilterRule *rule = NULL;
1669
        GSList *cur;
1670
        const gchar *rule_name;
1671
        FilterBoolOp bool_op = FLT_OR;
1672
        GSList *cond_list = NULL;
1673
        GSList *action_list = NULL;
1674
        GtkWidget *bool_op_menuitem;
1675
        gchar *error_msg = NULL;
1676
1677
        rule_name = gtk_entry_get_text(GTK_ENTRY(rule_edit_window.name_entry));
1678
        if (!rule_name || *rule_name == '\0') {
1679
                error_msg = _("Rule name is not specified.");
1680
                goto error;
1681
        }
1682
1683
        bool_op_menuitem = gtk_menu_get_active
1684
                (GTK_MENU(gtk_option_menu_get_menu
1685
                        (GTK_OPTION_MENU(rule_edit_window.bool_op_optmenu))));
1686
        bool_op = GPOINTER_TO_INT
1687
                (g_object_get_data(G_OBJECT(bool_op_menuitem), MENU_VAL_ID));
1688
1689
        for (cur = rule_edit_window.cond_hbox_list; cur != NULL;
1690
             cur = cur->next) {
1691
                CondHBox *hbox = (CondHBox *)cur->data;
1692
                GtkWidget *cond_type_menuitem;
1693
                CondMenuType cond_menu_type;
1694
                MatchMenuType match_menu_type;
1695
                const gchar *header_name;
1696
                const gchar *key_str;
1697
                gint int_value;
1698
                FilterCond *cond = NULL;
1699
                FilterMatchType match_type = FLT_CONTAIN;
1700
                FilterMatchFlag match_flag = 0;
1701
                SizeMatchType size_type;
1702
                AgeMatchType age_type;
1703
1704
                cond_type_menuitem = gtk_menu_get_active
1705
                        (GTK_MENU(gtk_option_menu_get_menu
1706
                                (GTK_OPTION_MENU(hbox->cond_type_optmenu))));
1707
                cond_menu_type = GPOINTER_TO_INT
1708
                        (g_object_get_data
1709
                                (G_OBJECT(cond_type_menuitem), MENU_VAL_ID));
1710
1711
                match_menu_type = menu_get_option_menu_active_index
1712
                        (GTK_OPTION_MENU(hbox->match_type_optmenu));
1713
1714
                key_str = gtk_entry_get_text(GTK_ENTRY(hbox->key_entry));
1715
1716
                switch (match_menu_type) {
1717
                case PF_MATCH_CONTAIN:
1718
                        match_type = FLT_CONTAIN;
1719
                        break;
1720
                case PF_MATCH_NOT_CONTAIN:
1721
                        match_type = FLT_CONTAIN;
1722
                        match_flag |= FLT_NOT_MATCH;
1723
                        break;
1724
                case PF_MATCH_EQUAL:
1725
                        match_type = FLT_EQUAL;
1726
                        break;
1727
                case PF_MATCH_NOT_EQUAL:
1728
                        match_type = FLT_EQUAL;
1729
                        match_flag |= FLT_NOT_MATCH;
1730
                        break;
1731
                case PF_MATCH_REGEX:
1732
                        match_type = FLT_REGEX;
1733
                        break;
1734
                case PF_MATCH_NOT_REGEX:
1735
                        match_type = FLT_REGEX;
1736
                        match_flag |= FLT_NOT_MATCH;
1737
                        break;
1738
                default:
1739
                        break;
1740
                }
1741
1742
                switch (cond_menu_type) {
1743
                case PF_COND_HEADER:
1744
                        header_name = g_object_get_data
1745
                                (G_OBJECT(cond_type_menuitem), "header_str");
1746
                        cond = filter_cond_new(FLT_COND_HEADER,
1747
                                               match_type, match_flag,
1748
                                               header_name, key_str);
1749
                        break;
1750
                case PF_COND_TO_OR_CC:
1751
                        cond = filter_cond_new(FLT_COND_TO_OR_CC, match_type,
1752
                                               match_flag, NULL, key_str);
1753
                        break;
1754
                case PF_COND_ANY_HEADER:
1755
                        cond = filter_cond_new(FLT_COND_ANY_HEADER, match_type,
1756
                                               match_flag, NULL, key_str);
1757
                        break;
1758
                case PF_COND_BODY:
1759
                        cond = filter_cond_new(FLT_COND_BODY, match_type,
1760
                                               match_flag, NULL, key_str);
1761
                        break;
1762
                case PF_COND_CMD_TEST:
1763
                        if (key_str && *key_str)
1764
                                cond = filter_cond_new(FLT_COND_CMD_TEST,
1765
                                                       0, 0, NULL, key_str);
1766
                        else
1767
                                error_msg = _("Command is not specified.");
1768
                        break;
1769
                case PF_COND_SIZE:
1770
                        size_type = menu_get_option_menu_active_index
1771
                                 (GTK_OPTION_MENU(hbox->size_match_optmenu));
1772
                        match_flag = size_type == PF_SIZE_LARGER
1773
                                ? 0 : FLT_NOT_MATCH;
1774
                        int_value = gtk_spin_button_get_value_as_int
1775
                                (GTK_SPIN_BUTTON(hbox->spin_btn));
1776
                        cond = filter_cond_new(FLT_COND_SIZE_GREATER,
1777
                                               0, match_flag, NULL,
1778
                                               itos(int_value));
1779
                        break;
1780
                case PF_COND_AGE:
1781
                        age_type = menu_get_option_menu_active_index
1782
                                 (GTK_OPTION_MENU(hbox->age_match_optmenu));
1783
                        match_flag = age_type == PF_AGE_LONGER
1784
                                ? 0 : FLT_NOT_MATCH;
1785
                        int_value = gtk_spin_button_get_value_as_int
1786
                                (GTK_SPIN_BUTTON(hbox->spin_btn));
1787
                        cond = filter_cond_new(FLT_COND_AGE_GREATER,
1788
                                               0, match_flag, NULL,
1789
                                               itos(int_value));
1790
                        break;
1791
                case PF_COND_ACCOUNT:
1792
                case PF_COND_EDIT_HEADER:
1793
                default:
1794
                        break;
1795
                }
1796
1797
                if (cond)
1798
                        cond_list = g_slist_append(cond_list, cond);
1799
                else {
1800
                        if (!error_msg)
1801
                                error_msg = _("Invalid condition exists.");
1802
                        goto error;
1803
                }
1804
        }
1805
1806
        for (cur = rule_edit_window.action_hbox_list; cur != NULL;
1807
             cur = cur->next) {
1808
                ActionHBox *hbox = (ActionHBox *)cur->data;
1809
                ActionMenuType action_menu_type;
1810
                const gchar *str;
1811
                guint color;
1812
                FilterAction *action = NULL;
1813
1814
                action_menu_type = prefs_filter_edit_get_action_hbox_type(hbox);
1815
1816
                switch (action_menu_type) {
1817
                case PF_ACTION_MOVE:
1818
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->folder_entry));
1819
                        if (str && *str)
1820
                                action = filter_action_new(FLT_ACTION_MOVE,
1821
                                                           str);
1822
                        else
1823
                                error_msg = _("Destination folder is not specified.");
1824
                        break;
1825
                case PF_ACTION_COPY:
1826
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->folder_entry));
1827
                        if (str && *str)
1828
                                action = filter_action_new(FLT_ACTION_COPY,
1829
                                                           str);
1830
                        else
1831
                                error_msg = _("Destination folder is not specified.");
1832
                        break;
1833
                case PF_ACTION_NOT_RECEIVE:
1834
                        action = filter_action_new(FLT_ACTION_NOT_RECEIVE,
1835
                                                   NULL);
1836
                        break;
1837
                case PF_ACTION_DELETE:
1838
                        action = filter_action_new(FLT_ACTION_DELETE, NULL);
1839
                        break;
1840
                case PF_ACTION_EXEC:
1841
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->cmd_entry));
1842
                        if (str && *str)
1843
                                action = filter_action_new(FLT_ACTION_EXEC,
1844
                                                           str);
1845
                        else
1846
                                error_msg = _("Command is not specified.");
1847
                        break;
1848
                case PF_ACTION_EXEC_ASYNC:
1849
                        str = gtk_entry_get_text(GTK_ENTRY(hbox->cmd_entry));
1850
                        if (str && *str)
1851
                                action = filter_action_new
1852
                                        (FLT_ACTION_EXEC_ASYNC, str);
1853
                        else
1854
                                error_msg = _("Command is not specified.");
1855
                        break;
1856
                case PF_ACTION_MARK:
1857
                        action = filter_action_new(FLT_ACTION_MARK, NULL);
1858
                        break;
1859
                case PF_ACTION_COLOR_LABEL:
1860
                        color = colorlabel_get_color_menu_active_item
1861
                                (gtk_option_menu_get_menu
1862
                                 (GTK_OPTION_MENU(hbox->clabel_optmenu)));
1863
                        action = filter_action_new(FLT_ACTION_COLOR_LABEL,
1864
                                                   itos(color));
1865
                        break;
1866
                case PF_ACTION_MARK_READ:
1867
                        action = filter_action_new(FLT_ACTION_MARK_READ, NULL);
1868
                        break;
1869
                case PF_ACTION_FORWARD:
1870
                case PF_ACTION_FORWARD_AS_ATTACHMENT:
1871
                case PF_ACTION_REDIRECT:
1872
                        break;
1873
                case PF_ACTION_STOP_EVAL:
1874
                        action = filter_action_new(FLT_ACTION_STOP_EVAL, NULL);
1875
                        break;
1876
                case PF_ACTION_SEPARATOR:
1877
                default:
1878
                        break;
1879
                }
1880
1881
                if (action)
1882
                        action_list = g_slist_append(action_list, action);
1883
                else {
1884
                        if (!error_msg)
1885
                                error_msg = _("Invalid action exists.");
1886
                        goto error;
1887
                }
1888
        }
1889
1890
error:
1891
        if (error_msg || !cond_list || !action_list) {
1892
                if (!error_msg) {
1893
                        if (!cond_list)
1894
                                error_msg = _("Condition not exist.");
1895
                        else
1896
                                error_msg = _("Action not exist.");
1897
                }
1898
                alertpanel_error("%s", error_msg);
1899
                if (cond_list)
1900
                        filter_cond_list_free(cond_list);
1901
                if (action_list)
1902
                        filter_action_list_free(action_list);
1903
                return NULL;
1904
        }
1905
1906
        rule = filter_rule_new(rule_name, bool_op, cond_list, action_list);
1907
1908
        return rule;
1909
}
1910
1911
/* callback functions */
1912
1913
static gint prefs_filter_edit_deleted(GtkWidget *widget, GdkEventAny *event,
1914
                                      gpointer data)
1915
{
1916
        prefs_filter_edit_cancel();
1917
        return TRUE;
1918
}
1919
1920
static gboolean prefs_filter_edit_key_pressed(GtkWidget *widget,
1921
                                              GdkEventKey *event,
1922
                                              gpointer data)
1923
{
1924
        if (event && event->keyval == GDK_Escape)
1925
                prefs_filter_edit_cancel();
1926
        return FALSE;
1927
}
1928
1929
static void prefs_filter_edit_ok(void)
1930
{
1931
        FilterRule *rule;
1932
1933
        rule = prefs_filter_edit_dialog_to_rule();
1934
        if (rule) {
1935
                rule_edit_window.new_rule = rule;
1936
                rule_edit_window.edit_finished = TRUE;
1937
        }
1938
}
1939
1940
static void prefs_filter_edit_cancel(void)
1941
{
1942
        rule_edit_window.new_rule = NULL;
1943
        rule_edit_window.edit_finished = TRUE;
1944
}
1945
1946
static void prefs_filter_cond_activated_cb(GtkWidget *widget, gpointer data)
1947
{
1948
        CondHBox *hbox = (CondHBox *)data;
1949
        CondMenuType type;
1950
1951
        type = GPOINTER_TO_INT
1952
                (g_object_get_data(G_OBJECT(widget), MENU_VAL_ID));
1953
1954
        if (type == PF_COND_EDIT_HEADER) {
1955
                prefs_filter_edit_edit_header_list();
1956
                prefs_filter_edit_cond_hbox_select
1957
                        (hbox, hbox->cur_type, hbox->cur_header_name);
1958
        } else {
1959
                hbox->cur_type = type;
1960
                g_free(hbox->cur_header_name);
1961
                hbox->cur_header_name = NULL;
1962
1963
                prefs_filter_edit_set_cond_hbox_widgets(hbox, type);
1964
                if (type == PF_COND_HEADER) {
1965
                        gchar *header_name;
1966
                        gchar *header_field;
1967
1968
                        header_name = (gchar *)g_object_get_data
1969
                                (G_OBJECT(widget), "header_str");
1970
                        header_field = prefs_filter_get_msg_header_field
1971
                                (header_name);
1972
                        if (header_field)
1973
                                gtk_entry_set_text(GTK_ENTRY(hbox->key_entry),
1974
                                                   header_field);
1975
                        hbox->cur_header_name = g_strdup(header_name);
1976
                }
1977
        }
1978
}
1979
1980
static void prefs_filter_action_activated_cb(GtkWidget *widget, gpointer data)
1981
{
1982
        ActionHBox *hbox = (ActionHBox *)data;
1983
        ActionMenuType type;
1984
1985
        type = GPOINTER_TO_INT
1986
                (g_object_get_data(G_OBJECT(widget), MENU_VAL_ID));
1987
        prefs_filter_edit_set_action_hbox_widgets(hbox, type);
1988
}
1989
1990
static void prefs_filter_action_select_dest_cb(GtkWidget *widget, gpointer data)
1991
{
1992
        ActionHBox *hbox = (ActionHBox *)data;
1993
1994
        FolderItem *dest;
1995
        gchar *id;
1996
1997
        dest = foldersel_folder_sel(NULL, FOLDER_SEL_COPY, NULL);
1998
        if (!dest || !dest->path) return;
1999
2000
        id = folder_item_get_identifier(dest);
2001
        if (id) {
2002
                gtk_entry_set_text(GTK_ENTRY(hbox->folder_entry), id);
2003
                g_free(id);
2004
        }
2005
}
2006
2007
static void prefs_filter_cond_del_cb(GtkWidget *widget, gpointer data)
2008
{
2009
        CondHBox *hbox = (CondHBox *)data;
2010
2011
        if (rule_edit_window.cond_hbox_list &&
2012
            rule_edit_window.cond_hbox_list->next)
2013
                prefs_filter_edit_remove_cond_hbox(hbox);
2014
}
2015
2016
static void prefs_filter_cond_add_cb(GtkWidget *widget, gpointer data)
2017
{
2018
        CondHBox *hbox = (CondHBox *)data;
2019
        CondHBox *new_hbox;
2020
        gint index;
2021
2022
        index = g_slist_index(rule_edit_window.cond_hbox_list, hbox);
2023
        g_return_if_fail(index >= 0);
2024
        new_hbox = prefs_filter_edit_cond_hbox_create();
2025
        prefs_filter_edit_set_cond_hbox_widgets(new_hbox, PF_COND_HEADER);
2026
        prefs_filter_edit_insert_cond_hbox(new_hbox, index + 1);
2027
}
2028
2029
static void prefs_filter_action_del_cb(GtkWidget *widget, gpointer data)
2030
{
2031
        ActionHBox *hbox = (ActionHBox *)data;
2032
2033
        if (rule_edit_window.action_hbox_list &&
2034
            rule_edit_window.action_hbox_list->next)
2035
                prefs_filter_edit_remove_action_hbox(hbox);
2036
}
2037
2038
static void prefs_filter_action_add_cb(GtkWidget *widget, gpointer data)
2039
{
2040
        ActionHBox *hbox = (ActionHBox *)data;
2041
        ActionHBox *new_hbox;
2042
        gboolean action_menu_selection[PF_ACTION_NONE];
2043
        gint index;
2044
2045
        prefs_filter_edit_get_action_hbox_menus_selection(action_menu_selection);
2046
2047
        index = g_slist_index(rule_edit_window.action_hbox_list, hbox);
2048
        g_return_if_fail(index >= 0);
2049
        new_hbox = prefs_filter_edit_action_hbox_create();
2050
        prefs_filter_edit_insert_action_hbox(new_hbox, index + 1);
2051
        if (action_menu_selection[PF_ACTION_MOVE] == TRUE ||
2052
            action_menu_selection[PF_ACTION_NOT_RECEIVE] == TRUE ||
2053
            action_menu_selection[PF_ACTION_DELETE] == TRUE)
2054
                prefs_filter_edit_set_action_hbox_widgets(new_hbox,
2055
                                                          PF_ACTION_COPY);
2056
        else
2057
                prefs_filter_edit_set_action_hbox_widgets(new_hbox,
2058
                                                          PF_ACTION_MOVE);
2059
}