1 /*
2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2017 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 <gdk/gdkkeysyms.h>
29 #include <gtk/gtkwidget.h>
30 #include <gtk/gtkwindow.h>
31 #include <gtk/gtkvbox.h>
32 #include <gtk/gtktable.h>
33 #include <gtk/gtkoptionmenu.h>
34 #include <gtk/gtklabel.h>
35 #include <gtk/gtkentry.h>
36 #include <gtk/gtkhbox.h>
37 #include <gtk/gtkcheckbutton.h>
38 #include <gtk/gtkhbbox.h>
39 #include <gtk/gtkbutton.h>
40 #include <gtk/gtkmenuitem.h>
41 #include <gtk/gtkstock.h>
42 #include <gtk/gtktreemodel.h>
43 #include <gtk/gtkliststore.h>
44 #include <gtk/gtktreeview.h>
45 #include <gtk/gtktreeselection.h>
46 #include <gtk/gtkcellrenderertext.h>
47 #include <gtk/gtkuimanager.h>
48 #include <stdio.h>
49 #include <stdlib.h>
50 #include <string.h>
51
52 #include "query_search.h"
53 #include "summaryview.h"
54 #include "messageview.h"
55 #include "mainwindow.h"
56 #include "folderview.h"
57 #include "menu.h"
58 #include "utils.h"
59 #include "gtkutils.h"
60 #include "manage_window.h"
61 #include "alertpanel.h"
62 #include "foldersel.h"
63 #include "statusbar.h"
64 #include "procmsg.h"
65 #include "procheader.h"
66 #include "folder.h"
67 #include "filter.h"
68 #include "prefs_common.h"
69 #include "prefs_filter.h"
70 #include "prefs_filter_edit.h"
71 #include "compose.h"
72 #include "sourcewindow.h"
73 #include "printing.h"
74
75 enum
76 {
77 COL_FOLDER,
78 COL_SUBJECT,
79 COL_FROM,
80 COL_DATE,
81 COL_MSGINFO,
82 N_COLS
83 };
84
85 static struct QuerySearchWindow {
86 GtkWidget *window;
87
88 GtkWidget *bool_optmenu;
89
90 FilterCondEdit *cond_edit;
91
92 GtkWidget *folder_entry;
93 GtkWidget *folder_btn;
94
95 GtkWidget *subfolder_checkbtn;
96 GtkWidget *case_checkbtn;
97
98 GtkWidget *treeview;
99 GtkListStore *store;
100
101 GtkWidget *status_label;
102
103 GtkWidget *clear_btn;
104 GtkWidget *search_btn;
105 GtkWidget *save_btn;
106 GtkWidget *close_btn;
107
108 GtkActionGroup *group;
109 GtkUIManager *ui;
110
111 FilterRule *rule;
112 gboolean requires_full_headers;
113
114 gboolean exclude_trash;
115
116 gint n_found;
117
118 gboolean on_search;
119 gboolean cancelled;
120 } search_window;
121
122 typedef struct {
123 GtkWidget *window;
124
125 GtkWidget *folder_entry;
126 GtkWidget *name_entry;
127
128 GtkWidget *ok_btn;
129 GtkWidget *cancel_btn;
130
131 gboolean cancelled;
132 gboolean finished;
133 } QuerySearchSaveDialog;
134
135 static const gchar *ui_def =
136 "<ui>"
137 " <popup name='PopUpMenu'>"
138 " <menuitem name='Reply' action='ReplyAction'/>"
139 " <menu name='ReplyTo' action='ReplyToAction'>"
140 " <menuitem name='ReplyAll' action='ReplyAllAction'/>"
141 " <menuitem name='ReplySender' action='ReplySenderAction'/>"
142 " <menuitem name='ReplyList' action='ReplyListAction'/>"
143 " </menu>"
144 " <separator />"
145 " <menuitem name='Forward' action='ForwardAction'/>"
146 " <menuitem name='ForwardAsAttach' action='ForwardAsAttachAction'/>"
147 " <menuitem name='Redirect' action='RedirectAction'/>"
148 " <separator />"
149 " <menuitem name='MoveMsg' action='MoveMsgAction'/>"
150 " <menuitem name='CopyMsg' action='CopyMsgAction'/>"
151 " <separator />"
152 " <menuitem name='AddAddress' action='AddAddressAction'/>"
153 " <menu name='CreateFilter' action='CreateFilterAction'>"
154 " <menuitem name='Auto' action='FilterAutoAction'/>"
155 " <menuitem name='From' action='FilterFromAction'/>"
156 " <menuitem name='To' action='FilterToAction'/>"
157 " <menuitem name='Subject' action='FilterSubjectAction'/>"
158 " </menu>"
159 " <separator />"
160 " <menuitem name='Open' action='OpenAction'/>"
161 " <menuitem name='OpenSource' action='OpenSourceAction'/>"
162 " <separator />"
163 " <menuitem name='Print' action='PrintAction'/>"
164 " </popup>"
165 "</ui>";
166
167 static void query_search_create (void);
168
169 static FilterRule *query_search_dialog_to_rule (const gchar *name,
170 FolderItem **item);
171
172 static void query_search_query (void);
173 static void query_search_folder (FolderItem *item);
174
175 static gboolean query_search_recursive_func (GNode *node,
176 gpointer data);
177
178 static void query_search_append_msg (MsgInfo *msginfo);
179 static void query_search_clear_list (void);
180
181 static void query_search_hbox_added (CondHBox *hbox);
182
183 static void row_activated (GtkTreeView *treeview,
184 GtkTreePath *path,
185 GtkTreeViewColumn *column,
186 gpointer data);
187
188 static gboolean row_selected (GtkTreeSelection *selection,
189 GtkTreeModel *model,
190 GtkTreePath *path,
191 gboolean cur_selected,
192 gpointer data);
193
194 static gboolean treeview_button_pressed (GtkWidget *widget,
195 GdkEventButton *event,
196 gpointer data);
197
198 static void query_search_clear (GtkButton *button,
199 gpointer data);
200 static void query_select_folder (GtkButton *button,
201 gpointer data);
202 static void query_search_clicked (GtkButton *button,
203 gpointer data);
204 static void query_search_save (GtkButton *button,
205 gpointer data);
206 static void query_search_close (GtkButton *button,
207 gpointer data);
208
209 static void query_search_entry_activated(GtkWidget *widget,
210 gpointer data);
211
212 static gint query_search_deleted (GtkWidget *widget,
213 GdkEventAny *event,
214 gpointer data);
215 static gboolean key_pressed (GtkWidget *widget,
216 GdkEventKey *event,
217 gpointer data);
218
219 static void query_search_open_msg (gboolean new_window);
220
221 /* Popup menu callbacks */
222 static void reply_cb(void);
223 static void reply_all_cb(void);
224 static void reply_sender_cb(void);
225 static void reply_list_cb(void);
226 static void forward_cb(void);
227 static void forward_attach_cb(void);
228 static void redirect_cb(void);
229 static void move_cb(void);
230 static void copy_cb(void);
231 static void add_address_cb(void);
232 static void create_filter_auto_cb(void);
233 static void create_filter_from_cb(void);
234 static void create_filter_to_cb(void);
235 static void create_filter_subject_cb(void);
236 static void open_cb (void);
237 static void open_source_cb (void);
238 static void print_cb (void);
239
240 static gint query_search_cmp_by_folder (GtkTreeModel *model,
241 GtkTreeIter *a,
242 GtkTreeIter *b,
243 gpointer data);
244 static gint query_search_cmp_by_subject (GtkTreeModel *model,
245 GtkTreeIter *a,
246 GtkTreeIter *b,
247 gpointer data);
248 static gint query_search_cmp_by_from (GtkTreeModel *model,
249 GtkTreeIter *a,
250 GtkTreeIter *b,
251 gpointer data);
252 static gint query_search_cmp_by_date (GtkTreeModel *model,
253 GtkTreeIter *a,
254 GtkTreeIter *b,
255 gpointer data);
256
257 static GtkActionEntry action_entries[] = {
258 {"ReplyAction", NULL, N_("_Reply"), NULL, NULL, reply_cb},
259 {"ReplyToAction", NULL, N_("Repl_y to"), NULL, NULL, NULL},
260 {"ReplyAllAction", NULL, N_("Reply to _all"), NULL, NULL, reply_all_cb},
261 {"ReplySenderAction", NULL, N_("Reply to _sender"), NULL, NULL, reply_sender_cb},
262 {"ReplyListAction", NULL, N_("Reply to _list"), NULL, NULL, reply_list_cb},
263
264 {"ForwardAction", NULL, N_("_Forward"), NULL, NULL, forward_cb},
265 {"ForwardAsAttachAction", NULL, N_("For_ward as attachment"), NULL, NULL, forward_attach_cb},
266 {"RedirectAction", NULL, N_("Redirec_t"), NULL, NULL, redirect_cb},
267
268 {"MoveMsgAction", NULL, N_("M_ove messages to..."), NULL, NULL, move_cb},
269 {"CopyMsgAction", NULL, N_("_Copy messages to..."), NULL, NULL, copy_cb},
270
271 {"AddAddressAction", NULL, N_("Add sender to address boo_k..."), NULL, NULL, add_address_cb},
272 {"CreateFilterAction", NULL, N_("Create f_ilter rule"), NULL, NULL, NULL},
273 {"FilterAutoAction", NULL, N_("_Automatically"), NULL, NULL, create_filter_auto_cb},
274 {"FilterFromAction", NULL, N_("by _From"), NULL, NULL, create_filter_from_cb},
275 {"FilterToAction", NULL, N_("by _To"), NULL, NULL, create_filter_to_cb},
276 {"FilterSubjectAction", NULL, N_("by _Subject"), NULL, NULL, create_filter_subject_cb},
277
278 {"OpenAction", GTK_STOCK_OPEN, N_("Open in _new window"), NULL, NULL, open_cb},
279 {"OpenSourceAction", NULL, N_("View mess_age source"), NULL, NULL, open_source_cb},
280
281 {"PrintAction", GTK_STOCK_PRINT, N_("_Print..."), NULL, NULL, print_cb},
282 };
283
284
query_search(FolderItem * item)285 void query_search(FolderItem *item)
286 {
287 gchar *id;
288
289 if (!search_window.window)
290 query_search_create();
291 else
292 gtk_window_present(GTK_WINDOW(search_window.window));
293
294 if (item && item->stype != F_VIRTUAL) {
295 id = folder_item_get_identifier(item);
296 gtk_entry_set_text(GTK_ENTRY(search_window.folder_entry), id);
297 g_free(id);
298 } else
299 gtk_entry_set_text(GTK_ENTRY(search_window.folder_entry), "");
300
301 gtk_widget_grab_focus(search_window.search_btn);
302 gtk_widget_show(search_window.window);
303 }
304
query_search_create(void)305 static void query_search_create(void)
306 {
307 GtkWidget *window;
308 GtkWidget *vbox1;
309 GtkWidget *bool_hbox;
310 GtkWidget *bool_optmenu;
311 GtkWidget *bool_menu;
312 GtkWidget *menuitem;
313 GtkWidget *clear_btn;
314 GtkWidget *search_btn;
315
316 GtkWidget *scrolledwin;
317 FilterCondEdit *cond_edit;
318 CondHBox *cond_hbox;
319
320 GtkWidget *folder_hbox;
321 GtkWidget *folder_label;
322 GtkWidget *folder_entry;
323 GtkWidget *folder_btn;
324
325 GtkWidget *checkbtn_hbox;
326 GtkWidget *subfolder_checkbtn;
327 GtkWidget *case_checkbtn;
328
329 GtkWidget *treeview;
330 GtkListStore *store;
331 GtkTreeViewColumn *column;
332 GtkCellRenderer *renderer;
333 GtkTreeSelection *selection;
334
335 GtkWidget *confirm_area;
336
337 GtkWidget *status_label;
338
339 GtkWidget *btn_hbox;
340 GtkWidget *hbbox;
341 GtkWidget *save_btn;
342 GtkWidget *close_btn;
343
344 GtkActionGroup *group;
345 GtkUIManager *ui;
346
347 window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
348 gtk_window_set_title(GTK_WINDOW (window), _("Search messages"));
349 gtk_widget_set_size_request(window, 600 * gtkut_get_dpi_multiplier(), -1);
350 gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, TRUE);
351 gtk_container_set_border_width(GTK_CONTAINER (window), 8);
352 g_signal_connect(G_OBJECT(window), "delete_event",
353 G_CALLBACK(query_search_deleted), NULL);
354 g_signal_connect(G_OBJECT(window), "key_press_event",
355 G_CALLBACK(key_pressed), NULL);
356 MANAGE_WINDOW_SIGNALS_CONNECT(window);
357
358 vbox1 = gtk_vbox_new (FALSE, 6);
359 gtk_widget_show (vbox1);
360 gtk_container_add (GTK_CONTAINER (window), vbox1);
361
362 bool_hbox = gtk_hbox_new(FALSE, 12);
363 gtk_widget_show(bool_hbox);
364 gtk_box_pack_start(GTK_BOX(vbox1), bool_hbox, FALSE, FALSE, 0);
365
366 bool_optmenu = gtk_option_menu_new();
367 gtk_widget_show(bool_optmenu);
368 gtk_box_pack_start(GTK_BOX(bool_hbox), bool_optmenu, FALSE, FALSE, 0);
369
370 bool_menu = gtk_menu_new();
371 MENUITEM_ADD(bool_menu, menuitem, _("Match any of the following"),
372 FLT_OR);
373 MENUITEM_ADD(bool_menu, menuitem, _("Match all of the following"),
374 FLT_AND);
375 gtk_option_menu_set_menu(GTK_OPTION_MENU(bool_optmenu), bool_menu);
376 gtk_option_menu_set_history(GTK_OPTION_MENU(bool_optmenu), FLT_AND);
377
378 hbbox = gtk_hbutton_box_new();
379 gtk_widget_show(hbbox);
380 gtk_button_box_set_layout(GTK_BUTTON_BOX(hbbox), GTK_BUTTONBOX_END);
381 gtk_box_set_spacing(GTK_BOX(hbbox), 6);
382 gtk_box_pack_end(GTK_BOX(bool_hbox), hbbox, FALSE, FALSE, 0);
383
384 clear_btn = gtk_button_new_from_stock(GTK_STOCK_CLEAR);
385 gtk_widget_show(clear_btn);
386 gtk_box_pack_start(GTK_BOX(hbbox), clear_btn, FALSE, FALSE, 0);
387
388 search_btn = gtk_button_new_from_stock(GTK_STOCK_FIND);
389 GTK_WIDGET_SET_FLAGS(search_btn, GTK_CAN_DEFAULT);
390 gtk_widget_show(search_btn);
391 gtk_box_pack_start(GTK_BOX(hbbox), search_btn, FALSE, FALSE, 0);
392 gtk_widget_grab_default(search_btn);
393
394 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
395 gtk_widget_show(scrolledwin);
396 gtk_box_pack_start(GTK_BOX(vbox1), scrolledwin, FALSE, FALSE, 0);
397 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
398 GTK_POLICY_AUTOMATIC,
399 GTK_POLICY_AUTOMATIC);
400 gtk_widget_set_size_request(scrolledwin, -1, 120 * gtkut_get_dpi_multiplier());
401
402 cond_edit = prefs_filter_edit_cond_edit_create();
403 cond_edit->add_hbox = query_search_hbox_added;
404 gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(scrolledwin),
405 cond_edit->cond_vbox);
406 prefs_filter_set_header_list(NULL);
407 prefs_filter_edit_set_header_list(cond_edit, NULL);
408 cond_hbox = prefs_filter_edit_cond_hbox_create(cond_edit);
409 prefs_filter_edit_set_cond_hbox_widgets(cond_hbox, PF_COND_HEADER);
410 prefs_filter_edit_insert_cond_hbox(cond_edit, cond_hbox, -1);
411 if (cond_edit->add_hbox)
412 cond_edit->add_hbox(cond_hbox);
413
414 folder_hbox = gtk_hbox_new (FALSE, 8);
415 gtk_widget_show (folder_hbox);
416 gtk_box_pack_start (GTK_BOX (vbox1), folder_hbox, FALSE, FALSE, 0);
417
418 folder_label = gtk_label_new (_("Folder:"));
419 gtk_widget_show (folder_label);
420 gtk_box_pack_start (GTK_BOX (folder_hbox), folder_label,
421 FALSE, FALSE, 0);
422
423 folder_entry = gtk_entry_new ();
424 gtk_widget_show (folder_entry);
425 gtk_box_pack_start (GTK_BOX (folder_hbox), folder_entry, TRUE, TRUE, 0);
426
427 folder_btn = gtk_button_new_with_label("...");
428 gtk_widget_show (folder_btn);
429 gtk_box_pack_start (GTK_BOX (folder_hbox), folder_btn, FALSE, FALSE, 0);
430
431 checkbtn_hbox = gtk_hbox_new (FALSE, 12);
432 gtk_widget_show (checkbtn_hbox);
433 gtk_box_pack_start (GTK_BOX (vbox1), checkbtn_hbox, FALSE, FALSE, 0);
434
435 subfolder_checkbtn =
436 gtk_check_button_new_with_label (_("Search subfolders"));
437 gtk_widget_show (subfolder_checkbtn);
438 gtk_box_pack_start (GTK_BOX (checkbtn_hbox), subfolder_checkbtn,
439 FALSE, FALSE, 0);
440
441 case_checkbtn = gtk_check_button_new_with_label (_("Case sensitive"));
442 gtk_widget_show (case_checkbtn);
443 gtk_box_pack_start (GTK_BOX (checkbtn_hbox), case_checkbtn,
444 FALSE, FALSE, 0);
445
446 scrolledwin = gtk_scrolled_window_new(NULL, NULL);
447 gtk_box_pack_start(GTK_BOX(vbox1), scrolledwin, TRUE, TRUE, 0);
448 gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolledwin),
449 GTK_POLICY_AUTOMATIC,
450 GTK_POLICY_AUTOMATIC);
451 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(scrolledwin),
452 GTK_SHADOW_IN);
453 gtk_widget_set_size_request(scrolledwin, -1, 150 * gtkut_get_dpi_multiplier());
454
455 store = gtk_list_store_new(N_COLS,
456 G_TYPE_STRING, G_TYPE_STRING, G_TYPE_STRING,
457 G_TYPE_STRING, G_TYPE_POINTER);
458 treeview = gtk_tree_view_new_with_model(GTK_TREE_MODEL(store));
459 g_object_unref(store);
460 gtk_tree_view_set_rules_hint(GTK_TREE_VIEW(treeview), TRUE);
461 g_signal_connect(G_OBJECT(treeview), "row-activated",
462 G_CALLBACK(row_activated), NULL);
463 g_signal_connect(G_OBJECT(treeview), "button-press-event",
464 G_CALLBACK(treeview_button_pressed), NULL);
465
466 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(store), COL_FOLDER,
467 query_search_cmp_by_folder, NULL, NULL);
468 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(store), COL_SUBJECT,
469 query_search_cmp_by_subject,
470 NULL, NULL);
471 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(store), COL_FROM,
472 query_search_cmp_by_from, NULL, NULL);
473 gtk_tree_sortable_set_sort_func(GTK_TREE_SORTABLE(store), COL_DATE,
474 query_search_cmp_by_date, NULL, NULL);
475
476 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(treeview));
477 gtk_tree_selection_set_mode(selection, GTK_SELECTION_MULTIPLE);
478 gtk_tree_selection_set_select_function(selection, row_selected,
479 NULL, NULL);
480
481 gtk_container_add(GTK_CONTAINER(scrolledwin), treeview);
482
483 #define APPEND_COLUMN(label, col, width) \
484 { \
485 renderer = gtk_cell_renderer_text_new(); \
486 column = gtk_tree_view_column_new_with_attributes \
487 (label, renderer, "text", col, NULL); \
488 gtk_tree_view_column_set_resizable(column, TRUE); \
489 if (width) { \
490 gtk_tree_view_column_set_sizing \
491 (column, GTK_TREE_VIEW_COLUMN_FIXED); \
492 gtk_tree_view_column_set_fixed_width(column, width); \
493 } \
494 gtk_tree_view_column_set_sort_column_id(column, col); \
495 gtk_tree_view_append_column(GTK_TREE_VIEW(treeview), column); \
496 }
497
498 APPEND_COLUMN(_("Folder"), COL_FOLDER, 0);
499 APPEND_COLUMN(_("Subject"), COL_SUBJECT, 200);
500 #if GTK_CHECK_VERSION(2, 14, 0)
501 gtk_tree_view_column_set_expand(column, TRUE);
502 #endif
503 APPEND_COLUMN(_("From"), COL_FROM, 180);
504 APPEND_COLUMN(_("Date"), COL_DATE, 0);
505
506 gtk_widget_show_all(scrolledwin);
507
508 confirm_area = gtk_hbox_new(FALSE, 12);
509 gtk_widget_show(confirm_area);
510 gtk_box_pack_start(GTK_BOX(vbox1), confirm_area, FALSE, FALSE, 0);
511
512 status_label = gtk_label_new("");
513 gtk_widget_show(status_label);
514 gtk_box_pack_start(GTK_BOX(confirm_area), status_label,
515 FALSE, FALSE, 0);
516
517 btn_hbox = gtk_hbox_new(FALSE, 6);
518 gtk_widget_show(btn_hbox);
519 gtk_box_pack_end(GTK_BOX(confirm_area), btn_hbox, FALSE, FALSE, 0);
520
521 gtkut_stock_button_set_create(&hbbox, &close_btn, GTK_STOCK_CLOSE,
522 NULL, NULL, NULL, NULL);
523 gtk_widget_show(hbbox);
524 gtk_box_pack_end(GTK_BOX(btn_hbox), hbbox, FALSE, FALSE, 0);
525
526 save_btn = gtk_button_new_with_mnemonic(_("_Save as search folder"));
527 gtk_widget_show(save_btn);
528 gtk_box_pack_end(GTK_BOX(btn_hbox), save_btn, FALSE, FALSE, 0);
529
530 g_signal_connect(G_OBJECT(clear_btn), "clicked",
531 G_CALLBACK(query_search_clear), NULL);
532 g_signal_connect(G_OBJECT(folder_btn), "clicked",
533 G_CALLBACK(query_select_folder), NULL);
534 g_signal_connect(G_OBJECT(search_btn), "clicked",
535 G_CALLBACK(query_search_clicked), NULL);
536 g_signal_connect(G_OBJECT(save_btn), "clicked",
537 G_CALLBACK(query_search_save), NULL);
538 g_signal_connect(G_OBJECT(close_btn), "clicked",
539 G_CALLBACK(query_search_close), NULL);
540
541 group = gtk_action_group_new("main");
542 gtk_action_group_set_translation_domain(group, GETTEXT_PACKAGE);
543 gtk_action_group_add_actions(group, action_entries,
544 sizeof(action_entries) /
545 sizeof(action_entries[0]), NULL);
546
547 ui = gtk_ui_manager_new();
548 gtk_ui_manager_insert_action_group(ui, group, 0);
549 gtk_ui_manager_add_ui_from_string(ui, ui_def, -1, NULL);
550
551 search_window.window = window;
552 search_window.bool_optmenu = bool_optmenu;
553
554 search_window.cond_edit = cond_edit;
555
556 search_window.folder_entry = folder_entry;
557 search_window.folder_btn = folder_btn;
558 search_window.subfolder_checkbtn = subfolder_checkbtn;
559 search_window.case_checkbtn = case_checkbtn;
560
561 search_window.treeview = treeview;
562 search_window.store = store;
563
564 search_window.status_label = status_label;
565
566 search_window.clear_btn = clear_btn;
567 search_window.search_btn = search_btn;
568 search_window.save_btn = save_btn;
569 search_window.close_btn = close_btn;
570
571 search_window.group = group;
572 search_window.ui = ui;
573 }
574
query_search_dialog_to_rule(const gchar * name,FolderItem ** item)575 static FilterRule *query_search_dialog_to_rule(const gchar *name,
576 FolderItem **item)
577 {
578 const gchar *id;
579 FolderItem *item_;
580 FilterBoolOp bool_op;
581 gboolean recursive;
582 gboolean case_sens;
583 GSList *cond_list;
584 FilterRule *rule;
585
586 id = gtk_entry_get_text(GTK_ENTRY(search_window.folder_entry));
587 item_ = folder_find_item_from_identifier(id);
588 if (!item_)
589 return NULL;
590 if (item)
591 *item = item_;
592
593 bool_op = menu_get_option_menu_active_index
594 (GTK_OPTION_MENU(search_window.bool_optmenu));
595 recursive = gtk_toggle_button_get_active
596 (GTK_TOGGLE_BUTTON(search_window.subfolder_checkbtn));
597 case_sens = gtk_toggle_button_get_active
598 (GTK_TOGGLE_BUTTON(search_window.case_checkbtn));
599
600 cond_list = prefs_filter_edit_cond_edit_to_list(search_window.cond_edit,
601 case_sens);
602 if (!cond_list)
603 return NULL;
604
605 rule = filter_rule_new(name, bool_op, cond_list, NULL);
606 rule->target_folder = g_strdup(id);
607 rule->recursive = recursive;
608
609 return rule;
610 }
611
query_search_query(void)612 static void query_search_query(void)
613 {
614 FolderItem *item;
615 gchar *msg;
616
617 if (search_window.on_search)
618 return;
619
620 search_window.on_search = TRUE;
621
622 search_window.rule = query_search_dialog_to_rule("Query rule", &item);
623 if (!search_window.rule) {
624 search_window.on_search = FALSE;
625 return;
626 }
627 search_window.requires_full_headers =
628 filter_rule_requires_full_headers(search_window.rule);
629
630 if (search_window.rule->recursive) {
631 if (item->stype == F_TRASH)
632 search_window.exclude_trash = FALSE;
633 else
634 search_window.exclude_trash = TRUE;
635 } else
636 search_window.exclude_trash = FALSE;
637
638 search_window.n_found = 0;
639 search_window.cancelled = FALSE;
640
641 gtk_widget_set_sensitive(search_window.clear_btn, FALSE);
642 gtk_button_set_label(GTK_BUTTON(search_window.search_btn),
643 GTK_STOCK_STOP);
644 query_search_clear_list();
645
646 if (search_window.rule->recursive)
647 g_node_traverse(item->node, G_PRE_ORDER, G_TRAVERSE_ALL, -1,
648 query_search_recursive_func, NULL);
649 else
650 query_search_folder(item);
651
652 filter_rule_free(search_window.rule);
653 search_window.rule = NULL;
654 search_window.requires_full_headers = FALSE;
655 search_window.exclude_trash = FALSE;
656
657 gtk_widget_set_sensitive(search_window.clear_btn, TRUE);
658 gtk_button_set_label(GTK_BUTTON(search_window.search_btn),
659 GTK_STOCK_FIND);
660 if (search_window.n_found == 0)
661 msg = g_strdup_printf(_("Message not found."));
662 else if (search_window.n_found == 1)
663 msg = g_strdup_printf(_("1 message found."));
664 else
665 msg = g_strdup_printf(_("%d messages found."),
666 search_window.n_found);
667 gtk_label_set_text(GTK_LABEL(search_window.status_label), msg);
668 g_free(msg);
669 statusbar_pop_all();
670
671 if (search_window.cancelled)
672 debug_print("* query search cancelled.\n");
673 debug_print("query search finished.\n");
674
675 search_window.n_found = 0;
676 search_window.on_search = FALSE;
677 search_window.cancelled = FALSE;
678 }
679
680 typedef struct _QueryData
681 {
682 FolderItem *item;
683 gchar *folder_name;
684 gint count;
685 gint total;
686 gint flag;
687 GTimeVal tv_prev;
688 GSList *mlist;
689 #if USE_THREADS
690 GAsyncQueue *queue;
691 guint timer_tag;
692 #endif
693 } QueryData;
694
query_search_folder_show_progress(const gchar * name,gint count,gint total)695 static void query_search_folder_show_progress(const gchar *name, gint count,
696 gint total)
697 {
698 gchar *str;
699
700 str = g_strdup_printf(_("Searching %s (%d / %d)..."),
701 name, count, total);
702 gtk_label_set_text(GTK_LABEL(search_window.status_label), str);
703 g_free(str);
704 #ifndef USE_THREADS
705 ui_update();
706 #endif
707 }
708
709 #if USE_THREADS
query_search_progress_func(gpointer data)710 static gboolean query_search_progress_func(gpointer data)
711 {
712 QueryData *qdata = (QueryData *)data;
713 MsgInfo *msginfo;
714
715 gdk_threads_enter();
716 query_search_folder_show_progress(qdata->folder_name,
717 g_atomic_int_get(&qdata->count),
718 qdata->total);
719 while ((msginfo = g_async_queue_try_pop(qdata->queue)))
720 query_search_append_msg(msginfo);
721 gdk_threads_leave();
722
723 return TRUE;
724 }
725 #endif
726
query_search_folder_func(gpointer data)727 static gpointer query_search_folder_func(gpointer data)
728 {
729 QueryData *qdata = (QueryData *)data;
730 GSList *mlist, *cur;
731 FilterInfo fltinfo;
732 GTimeVal tv_cur;
733
734 debug_print("query_search_folder_func start\n");
735
736 #if USE_THREADS
737 g_async_queue_ref(qdata->queue);
738 #endif
739
740 mlist = qdata->mlist;
741
742 memset(&fltinfo, 0, sizeof(FilterInfo));
743
744 debug_print("requires_full_headers: %d\n",
745 search_window.requires_full_headers);
746 debug_print("start query search: %s\n",
747 qdata->item->path ? qdata->item->path : "");
748
749 for (cur = mlist; cur != NULL; cur = cur->next) {
750 MsgInfo *msginfo = (MsgInfo *)cur->data;
751 GSList *hlist;
752
753 g_atomic_int_add(&qdata->count, 1);
754
755 g_get_current_time(&tv_cur);
756 if ((tv_cur.tv_sec - qdata->tv_prev.tv_sec) * G_USEC_PER_SEC +
757 tv_cur.tv_usec - qdata->tv_prev.tv_usec >
758 PROGRESS_UPDATE_INTERVAL * 1000) {
759 #ifndef USE_THREADS
760 query_search_folder_show_progress(qdata->folder_name,
761 qdata->count,
762 qdata->total);
763 #endif
764 qdata->tv_prev = tv_cur;
765 }
766
767 if (search_window.cancelled)
768 break;
769
770 fltinfo.flags = msginfo->flags;
771 if (search_window.requires_full_headers) {
772 gchar *file;
773
774 file = procmsg_get_message_file(msginfo);
775 hlist = procheader_get_header_list_from_file(file);
776 g_free(file);
777 } else
778 hlist = procheader_get_header_list_from_msginfo
779 (msginfo);
780 if (!hlist)
781 continue;
782
783 if (filter_match_rule(search_window.rule, msginfo, hlist,
784 &fltinfo)) {
785 #if USE_THREADS
786 g_async_queue_push(qdata->queue, msginfo);
787 #else
788 query_search_append_msg(msginfo);
789 #endif
790 cur->data = NULL;
791 search_window.n_found++;
792 }
793
794 procheader_header_list_destroy(hlist);
795 }
796
797 #if USE_THREADS
798 g_async_queue_unref(qdata->queue);
799 #endif
800
801 g_atomic_int_set(&qdata->flag, 1);
802 g_main_context_wakeup(NULL);
803
804 debug_print("query_search_folder_func end\n");
805
806 return GINT_TO_POINTER(0);
807 }
808
query_search_folder(FolderItem * item)809 static void query_search_folder(FolderItem *item)
810 {
811 gchar *str;
812 QueryData data = {item};
813 #if USE_THREADS
814 GThread *thread;
815 MsgInfo *msginfo;
816 #endif
817
818 if (!item->path || item->stype == F_VIRTUAL)
819 return;
820
821 data.folder_name = g_path_get_basename(item->path);
822 str = g_strdup_printf(_("Searching %s ..."), data.folder_name);
823 gtk_label_set_text(GTK_LABEL(search_window.status_label), str);
824 g_free(str);
825 g_get_current_time(&data.tv_prev);
826 #ifndef USE_THREADS
827 ui_update();
828 #endif
829
830 if (search_window.cancelled) {
831 g_free(data.folder_name);
832 return;
833 }
834
835 if (item->opened)
836 summary_write_cache(main_window_get()->summaryview);
837
838 procmsg_set_auto_decrypt_message(FALSE);
839
840 data.mlist = folder_item_get_msg_list(item, TRUE);
841 data.total = g_slist_length(data.mlist);
842
843 #if USE_THREADS
844 data.queue = g_async_queue_new();
845 data.timer_tag = g_timeout_add(PROGRESS_UPDATE_INTERVAL,
846 query_search_progress_func, &data);
847 thread = g_thread_create(query_search_folder_func, &data, TRUE, NULL);
848
849 debug_print("query_search_folder: thread started\n");
850 while (g_atomic_int_get(&data.flag) == 0)
851 gtk_main_iteration();
852 log_window_flush();
853
854 while ((msginfo = g_async_queue_try_pop(data.queue)))
855 query_search_append_msg(msginfo);
856
857 g_source_remove(data.timer_tag);
858 g_thread_join(thread);
859 debug_print("query_search_folder: thread exited\n");
860
861 g_async_queue_unref(data.queue);
862 #else /* !USE_THREADS */
863 query_search_folder_func(&data);
864 #endif
865
866 procmsg_msg_list_free(data.mlist);
867 procmsg_set_auto_decrypt_message(TRUE);
868 g_free(data.folder_name);
869 }
870
query_search_recursive_func(GNode * node,gpointer data)871 static gboolean query_search_recursive_func(GNode *node, gpointer data)
872 {
873 FolderItem *item;
874
875 g_return_val_if_fail(node->data != NULL, FALSE);
876
877 item = FOLDER_ITEM(node->data);
878
879 if (!item->path)
880 return FALSE;
881 if (search_window.exclude_trash && item->stype == F_TRASH)
882 return FALSE;
883
884 query_search_folder(item);
885
886 if (search_window.cancelled)
887 return TRUE;
888
889 return FALSE;
890 }
891
query_search_append_msg(MsgInfo * msginfo)892 static void query_search_append_msg(MsgInfo *msginfo)
893 {
894 GtkListStore *store = search_window.store;
895 GtkTreeIter iter;
896 gchar *folder;
897 gchar date_buf[80];
898 const gchar *subject, *from, *date;
899 gchar *id;
900
901 id = folder_item_get_identifier(msginfo->folder);
902 folder = g_path_get_basename(id);
903 g_free(id);
904 subject = msginfo->subject ? msginfo->subject : _("(No Subject)");
905 from = msginfo->from ? msginfo->from : _("(No From)");
906 if (msginfo->date_t) {
907 procheader_date_get_localtime(date_buf, sizeof(date_buf),
908 msginfo->date_t);
909 date = date_buf;
910 } else if (msginfo->date)
911 date = msginfo->date;
912 else
913 date = _("(No Date)");
914
915 gtk_list_store_append(store, &iter);
916 gtk_list_store_set(store, &iter,
917 COL_FOLDER, folder,
918 COL_SUBJECT, subject,
919 COL_FROM, from,
920 COL_DATE, date,
921 COL_MSGINFO, msginfo,
922 -1);
923
924 g_free(folder);
925 }
926
query_search_clear_list(void)927 static void query_search_clear_list(void)
928 {
929 GtkTreeIter iter;
930 GtkTreeModel *model = GTK_TREE_MODEL(search_window.store);
931 MsgInfo *msginfo;
932
933 gtkut_tree_sortable_unset_sort_column_id
934 (GTK_TREE_SORTABLE(search_window.store));
935
936 if (!gtk_tree_model_get_iter_first(model, &iter))
937 return;
938
939 do {
940 gtk_tree_model_get(model, &iter, COL_MSGINFO, &msginfo, -1);
941 procmsg_msginfo_free(msginfo);
942 } while (gtk_tree_model_iter_next(model, &iter));
943
944 gtk_list_store_clear(search_window.store);
945 }
946
query_search_hbox_added(CondHBox * hbox)947 static void query_search_hbox_added(CondHBox *hbox)
948 {
949 g_signal_connect(hbox->key_entry, "activate",
950 G_CALLBACK(query_search_entry_activated), NULL);
951 }
952
row_activated(GtkTreeView * treeview,GtkTreePath * path,GtkTreeViewColumn * column,gpointer data)953 static void row_activated(GtkTreeView *treeview, GtkTreePath *path,
954 GtkTreeViewColumn *column, gpointer data)
955 {
956 query_search_open_msg(FALSE);
957 }
958
row_selected(GtkTreeSelection * selection,GtkTreeModel * model,GtkTreePath * path,gboolean cur_selected,gpointer data)959 static gboolean row_selected(GtkTreeSelection *selection,
960 GtkTreeModel *model, GtkTreePath *path,
961 gboolean cur_selected, gpointer data)
962 {
963 return TRUE;
964 }
965
treeview_button_pressed(GtkWidget * widget,GdkEventButton * event,gpointer data)966 static gboolean treeview_button_pressed(GtkWidget *widget,
967 GdkEventButton *event, gpointer data)
968 {
969 GtkTreeIter iter;
970 GtkTreeModel *model = GTK_TREE_MODEL(search_window.store);
971 GtkWidget *menu;
972 GtkTreeSelection *selection;
973 GtkTreePath *path;
974 GtkTreeViewColumn *column = NULL;
975 gboolean is_selected;
976 gint px, py;
977
978 debug_print("treeview_button_pressed\n");
979
980 if (!event)
981 return FALSE;
982
983 if (!gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(widget),
984 event->x, event->y, &path, &column,
985 NULL, NULL))
986 return FALSE;
987
988 // ??
989 gtk_widget_get_pointer(widget, &px, &py);
990 if (py == (gint)event->y) {
991 gtk_tree_path_free(path);
992 return FALSE;
993 }
994
995 if (!gtk_tree_model_get_iter(model, &iter, path)) {
996 gtk_tree_path_free(path);
997 return FALSE;
998 }
999
1000 selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget));
1001 is_selected = gtk_tree_selection_path_is_selected(selection, path);
1002 gtk_tree_path_free(path);
1003
1004 if (event->button == 3) {
1005 menu = gtk_ui_manager_get_widget(search_window.ui,
1006 "/PopUpMenu");
1007 gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL,
1008 event->button, event->time);
1009 if (is_selected)
1010 return TRUE;
1011 }
1012
1013 return FALSE;
1014 }
1015
query_search_clear(GtkButton * button,gpointer data)1016 static void query_search_clear(GtkButton *button, gpointer data)
1017 {
1018 CondHBox *cond_hbox;
1019
1020 if (search_window.on_search)
1021 return;
1022
1023 prefs_filter_edit_clear_cond_edit(search_window.cond_edit);
1024 prefs_filter_set_header_list(NULL);
1025 prefs_filter_edit_set_header_list(search_window.cond_edit, NULL);
1026 cond_hbox = prefs_filter_edit_cond_hbox_create(search_window.cond_edit);
1027 prefs_filter_edit_set_cond_hbox_widgets(cond_hbox, PF_COND_HEADER);
1028 prefs_filter_edit_insert_cond_hbox
1029 (search_window.cond_edit, cond_hbox, -1);
1030 if (search_window.cond_edit->add_hbox)
1031 search_window.cond_edit->add_hbox(cond_hbox);
1032
1033 gtk_label_set_text(GTK_LABEL(search_window.status_label), "");
1034
1035 query_search_clear_list();
1036 }
1037
query_select_folder(GtkButton * button,gpointer data)1038 static void query_select_folder(GtkButton *button, gpointer data)
1039 {
1040 FolderItem *item;
1041 gchar *id;
1042
1043 item = foldersel_folder_sel(NULL, FOLDER_SEL_ALL, NULL);
1044 if (!item || item->stype == F_VIRTUAL)
1045 return;
1046
1047 id = folder_item_get_identifier(item);
1048 if (id) {
1049 gtk_entry_set_text(GTK_ENTRY(search_window.folder_entry), id);
1050 g_free(id);
1051 }
1052 }
1053
query_search_clicked(GtkButton * button,gpointer data)1054 static void query_search_clicked(GtkButton *button, gpointer data)
1055 {
1056 if (search_window.on_search)
1057 search_window.cancelled = TRUE;
1058 else
1059 query_search_query();
1060 }
1061
query_search_save_dialog_deleted(GtkWidget * widget,GdkEventAny * event,gpointer data)1062 static gint query_search_save_dialog_deleted(GtkWidget *widget,
1063 GdkEventAny *event, gpointer data)
1064 {
1065 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1066
1067 dialog->cancelled = TRUE;
1068 dialog->finished = TRUE;
1069 return TRUE;
1070 }
1071
query_search_save_dialog_key_pressed(GtkWidget * widget,GdkEventKey * event,gpointer data)1072 static gint query_search_save_dialog_key_pressed(GtkWidget *widget,
1073 GdkEventKey *event,
1074 gpointer data)
1075 {
1076 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1077
1078 if (event && event->keyval == GDK_Escape) {
1079 dialog->cancelled = TRUE;
1080 dialog->finished = TRUE;
1081 }
1082 return FALSE;
1083 }
1084
query_search_save_dialog_select_folder(GtkButton * button,gpointer data)1085 static void query_search_save_dialog_select_folder(GtkButton *button,
1086 gpointer data)
1087 {
1088 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1089 FolderItem *item;
1090 gchar *id;
1091
1092 item = foldersel_folder_sel(NULL, FOLDER_SEL_ALL, NULL);
1093 if (!item || item->no_sub || item->stype == F_VIRTUAL)
1094 return;
1095
1096 id = folder_item_get_identifier(item);
1097 if (id) {
1098 gtk_entry_set_text(GTK_ENTRY(dialog->folder_entry), id);
1099 g_free(id);
1100 }
1101 }
1102
query_search_save_activated(GtkEditable * editable,gpointer data)1103 static void query_search_save_activated(GtkEditable *editable, gpointer data)
1104 {
1105 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1106
1107 gtk_button_clicked(GTK_BUTTON(dialog->ok_btn));
1108 }
1109
query_search_save_ok(GtkButton * button,gpointer data)1110 static void query_search_save_ok(GtkButton *button, gpointer data)
1111 {
1112 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1113
1114 dialog->finished = TRUE;
1115 }
1116
query_search_save_cancel(GtkButton * button,gpointer data)1117 static void query_search_save_cancel(GtkButton *button, gpointer data)
1118 {
1119 QuerySearchSaveDialog *dialog = (QuerySearchSaveDialog *)data;
1120
1121 dialog->cancelled = TRUE;
1122 dialog->finished = TRUE;
1123 }
1124
query_search_save_dialog_create(void)1125 static QuerySearchSaveDialog *query_search_save_dialog_create(void)
1126 {
1127 QuerySearchSaveDialog *dialog;
1128 GtkWidget *window;
1129 GtkWidget *vbox;
1130 GtkWidget *hbox;
1131 GtkWidget *label;
1132 GtkWidget *folder_entry;
1133 GtkWidget *folder_btn;
1134 GtkWidget *name_entry;
1135
1136 GtkWidget *confirm_area;
1137 GtkWidget *hbbox;
1138 GtkWidget *cancel_btn;
1139 GtkWidget *ok_btn;
1140
1141 dialog = g_new0(QuerySearchSaveDialog, 1);
1142
1143 window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
1144 gtk_window_set_title(GTK_WINDOW(window), _("Save as search folder"));
1145 gtk_widget_set_size_request(window, 400 * gtkut_get_dpi_multiplier(), -1);
1146 gtk_container_set_border_width(GTK_CONTAINER(window), 8);
1147 gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
1148 gtk_window_set_modal(GTK_WINDOW(window), TRUE);
1149 gtk_window_set_policy(GTK_WINDOW(window), FALSE, TRUE, FALSE);
1150 g_signal_connect(G_OBJECT(window), "delete_event",
1151 G_CALLBACK(query_search_save_dialog_deleted),
1152 dialog);
1153 g_signal_connect(G_OBJECT(window), "key_press_event",
1154 G_CALLBACK(query_search_save_dialog_key_pressed),
1155 dialog);
1156 MANAGE_WINDOW_SIGNALS_CONNECT(window);
1157 manage_window_set_transient(GTK_WINDOW(window));
1158
1159 vbox = gtk_vbox_new(FALSE, 8);
1160 gtk_container_add(GTK_CONTAINER(window), vbox);
1161
1162 hbox = gtk_hbox_new(FALSE, 8);
1163 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
1164
1165 label = gtk_label_new(_("Location:"));
1166 gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
1167
1168 folder_entry = gtk_entry_new();
1169 gtk_box_pack_start(GTK_BOX(hbox), folder_entry, TRUE, TRUE, 0);
1170
1171 folder_btn = gtk_button_new_with_label("...");
1172 gtk_box_pack_start(GTK_BOX(hbox), folder_btn, FALSE, FALSE, 0);
1173 g_signal_connect(G_OBJECT(folder_btn), "clicked",
1174 G_CALLBACK(query_search_save_dialog_select_folder),
1175 dialog);
1176
1177 hbox = gtk_hbox_new(FALSE, 8);
1178 gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0);
1179
1180 label = gtk_label_new(_("Folder name:"));
1181 gtk_box_pack_start(GTK_BOX(hbox), label, FALSE, FALSE, 0);
1182
1183 name_entry = gtk_entry_new();
1184 gtk_box_pack_start(GTK_BOX(hbox), name_entry, TRUE, TRUE, 0);
1185 g_signal_connect(G_OBJECT(name_entry), "activate",
1186 G_CALLBACK(query_search_save_activated), dialog);
1187
1188 confirm_area = gtk_hbox_new(FALSE, 12);
1189 gtk_box_pack_end(GTK_BOX(vbox), confirm_area, FALSE, FALSE, 0);
1190
1191 gtkut_stock_button_set_create(&hbbox,
1192 &ok_btn, GTK_STOCK_OK,
1193 &cancel_btn, GTK_STOCK_CANCEL,
1194 NULL, NULL);
1195 gtk_box_pack_end(GTK_BOX(confirm_area), hbbox, FALSE, FALSE, 0);
1196 GTK_WIDGET_SET_FLAGS(ok_btn, GTK_CAN_DEFAULT);
1197 gtk_widget_grab_default(ok_btn);
1198 g_signal_connect(G_OBJECT(ok_btn), "clicked",
1199 G_CALLBACK(query_search_save_ok), dialog);
1200 g_signal_connect(G_OBJECT(cancel_btn), "clicked",
1201 G_CALLBACK(query_search_save_cancel), dialog);
1202
1203 gtk_widget_grab_focus(name_entry);
1204
1205 gtk_widget_show_all(window);
1206
1207 dialog->window = window;
1208 dialog->folder_entry = folder_entry;
1209 dialog->name_entry = name_entry;
1210 dialog->ok_btn = ok_btn;
1211 dialog->cancel_btn = cancel_btn;
1212 dialog->cancelled = FALSE;
1213 dialog->finished = FALSE;
1214
1215 return dialog;
1216 }
1217
query_search_save_dialog_destroy(QuerySearchSaveDialog * dialog)1218 static void query_search_save_dialog_destroy(QuerySearchSaveDialog *dialog)
1219 {
1220 gtk_widget_destroy(dialog->window);
1221 g_free(dialog);
1222 }
1223
query_search_create_vfolder(FolderItem * parent,const gchar * name)1224 static FolderItem *query_search_create_vfolder(FolderItem *parent,
1225 const gchar *name)
1226 {
1227 gchar *path;
1228 gchar *fs_name;
1229 gchar *fullpath;
1230 FolderItem *item;
1231
1232 g_return_val_if_fail(parent != NULL, NULL);
1233 g_return_val_if_fail(name != NULL, NULL);
1234
1235 path = folder_item_get_path(parent);
1236 fs_name = g_filename_from_utf8(name, -1, NULL, NULL, NULL);
1237 fullpath = g_strconcat(path, G_DIR_SEPARATOR_S,
1238 fs_name ? fs_name : name, NULL);
1239 g_free(fs_name);
1240 g_free(path);
1241
1242 if (make_dir_hier(fullpath) < 0) {
1243 g_free(fullpath);
1244 return NULL;
1245 }
1246
1247 if (parent->path)
1248 path = g_strconcat(parent->path, "/", name, NULL);
1249 else
1250 path = g_strdup(name);
1251
1252 item = folder_item_new(name, path);
1253 item->stype = F_VIRTUAL;
1254 item->no_sub = TRUE;
1255 folder_item_append(parent, item);
1256
1257 g_free(path);
1258
1259 return item;
1260 }
1261
query_search_vfolder_update_rule(FolderItem * item)1262 static void query_search_vfolder_update_rule(FolderItem *item)
1263 {
1264 GSList list;
1265 FilterRule *rule;
1266 gchar *file;
1267 gchar *path;
1268
1269 rule = query_search_dialog_to_rule(item->name, NULL);
1270 list.data = rule;
1271 list.next = NULL;
1272
1273 path = folder_item_get_path(item);
1274 file = g_strconcat(path, G_DIR_SEPARATOR_S, FILTER_LIST, NULL);
1275 filter_write_file(&list, file);
1276 g_free(file);
1277 g_free(path);
1278
1279 filter_rule_free(rule);
1280 }
1281
query_search_save(GtkButton * button,gpointer data)1282 static void query_search_save(GtkButton *button, gpointer data)
1283 {
1284 QuerySearchSaveDialog *dialog;
1285 const gchar *id, *name;
1286 FolderItem *parent, *item;
1287
1288 dialog = query_search_save_dialog_create();
1289 id = gtk_entry_get_text(GTK_ENTRY(search_window.folder_entry));
1290 if (id && *id)
1291 gtk_entry_set_text(GTK_ENTRY(dialog->folder_entry), id);
1292
1293 while (!dialog->finished)
1294 gtk_main_iteration();
1295
1296 if (dialog->cancelled) {
1297 query_search_save_dialog_destroy(dialog);
1298 return;
1299 }
1300
1301 id = gtk_entry_get_text(GTK_ENTRY(dialog->folder_entry));
1302 parent = folder_find_item_from_identifier(id);
1303 name = gtk_entry_get_text(GTK_ENTRY(dialog->name_entry));
1304 if (parent && name && *name) {
1305 if (folder_find_child_item_by_name(parent, name)) {
1306 alertpanel_error(_("The folder `%s' already exists."),
1307 name);
1308 } else {
1309 item = query_search_create_vfolder(parent, name);
1310 if (item) {
1311 query_search_vfolder_update_rule(item);
1312 folderview_append_item(folderview_get(),
1313 NULL, item, TRUE);
1314 folder_write_list();
1315 }
1316 }
1317 }
1318
1319 query_search_save_dialog_destroy(dialog);
1320 }
1321
query_search_close(GtkButton * button,gpointer data)1322 static void query_search_close(GtkButton *button, gpointer data)
1323 {
1324 if (search_window.on_search)
1325 search_window.cancelled = TRUE;
1326 gtk_widget_hide(search_window.window);
1327 }
1328
query_search_entry_activated(GtkWidget * widget,gpointer data)1329 static void query_search_entry_activated(GtkWidget *widget, gpointer data)
1330 {
1331 gtk_button_clicked(GTK_BUTTON(search_window.search_btn));
1332 }
1333
query_search_deleted(GtkWidget * widget,GdkEventAny * event,gpointer data)1334 static gint query_search_deleted(GtkWidget *widget, GdkEventAny *event,
1335 gpointer data)
1336 {
1337 gtk_button_clicked(GTK_BUTTON(search_window.close_btn));
1338 return TRUE;
1339 }
1340
key_pressed(GtkWidget * widget,GdkEventKey * event,gpointer data)1341 static gboolean key_pressed(GtkWidget *widget, GdkEventKey *event,
1342 gpointer data)
1343 {
1344 if (event && event->keyval == GDK_Escape) {
1345 if (search_window.on_search)
1346 gtk_button_clicked
1347 (GTK_BUTTON(search_window.search_btn));
1348 else
1349 gtk_button_clicked(GTK_BUTTON(search_window.close_btn));
1350 return TRUE;
1351 }
1352 return FALSE;
1353 }
1354
query_search_get_selected_msg_list(void)1355 static GSList *query_search_get_selected_msg_list(void)
1356 {
1357 GtkTreeModel *model = GTK_TREE_MODEL(search_window.store);
1358 GtkTreeSelection *selection;
1359 GList *list, *cur;
1360 GtkTreePath *path;
1361 GtkTreeIter iter;
1362 GSList *mlist = NULL;
1363 MsgInfo *msginfo;
1364
1365 selection = gtk_tree_view_get_selection
1366 (GTK_TREE_VIEW(search_window.treeview));
1367 list = gtk_tree_selection_get_selected_rows(selection, NULL);
1368 if (!list)
1369 return NULL;
1370
1371 for (cur = list; cur != NULL; cur = cur->next) {
1372 path = cur->data;
1373 if (!gtk_tree_model_get_iter(model, &iter, path))
1374 break;
1375 msginfo = NULL;
1376 gtk_tree_model_get(model, &iter, COL_MSGINFO, &msginfo, -1);
1377 if (msginfo)
1378 mlist = g_slist_prepend(mlist, msginfo);
1379 }
1380 mlist = g_slist_reverse(mlist);
1381
1382 g_list_foreach(list, (GFunc)gtk_tree_path_free, NULL);
1383 g_list_free(list);
1384
1385 return mlist;
1386 }
1387
query_search_open_msg(gboolean new_window)1388 static void query_search_open_msg(gboolean new_window)
1389 {
1390 GSList *mlist;
1391 MsgInfo *msginfo;
1392 MessageView *msgview;
1393
1394 mlist = query_search_get_selected_msg_list();
1395 if (!mlist)
1396 return;
1397 msginfo = (MsgInfo *)mlist->data;
1398 g_slist_free(mlist);
1399
1400 if (new_window || !summary_select_by_msginfo(main_window_get()->summaryview, msginfo)) {
1401 msgview = messageview_create_with_new_window();
1402 messageview_show(msgview, msginfo, FALSE);
1403 statusbar_pop_all();
1404 }
1405 }
1406
query_search_reply_forward(ComposeMode mode)1407 static void query_search_reply_forward(ComposeMode mode)
1408 {
1409 GSList *mlist;
1410 MsgInfo *msginfo;
1411
1412 mlist = query_search_get_selected_msg_list();
1413 if (!mlist)
1414 return;
1415 msginfo = (MsgInfo *)mlist->data;
1416
1417 if (prefs_common.reply_with_quote)
1418 mode |= COMPOSE_WITH_QUOTE;
1419
1420 switch (COMPOSE_MODE(mode)) {
1421 case COMPOSE_REPLY:
1422 case COMPOSE_REPLY_TO_SENDER:
1423 case COMPOSE_REPLY_TO_ALL:
1424 case COMPOSE_REPLY_TO_LIST:
1425 compose_reply(msginfo, msginfo->folder, mode, NULL);
1426 break;
1427 case COMPOSE_FORWARD:
1428 compose_forward(mlist, NULL, FALSE, NULL);
1429 break;
1430 case COMPOSE_FORWARD_AS_ATTACH:
1431 compose_forward(mlist, NULL, TRUE, NULL);
1432 break;
1433 case COMPOSE_REDIRECT:
1434 compose_redirect(msginfo, msginfo->folder);
1435 break;
1436 default:
1437 g_warning("query_search_reply_forward: invalid mode: %d", mode);
1438 }
1439
1440 g_slist_free(mlist);
1441 }
1442
query_search_move_copy(gboolean is_copy)1443 static void query_search_move_copy(gboolean is_copy)
1444 {
1445 GSList *mlist, *cur;
1446 FolderItem *dest;
1447 MsgInfo *msginfo;
1448
1449 debug_print("query_search_move_copy: is_copy: %d\n", is_copy);
1450
1451 mlist = query_search_get_selected_msg_list();
1452 if (!mlist)
1453 return;
1454
1455 if (is_copy)
1456 dest = foldersel_folder_sel_full(NULL, FOLDER_SEL_COPY, NULL,
1457 _("Select folder to copy"));
1458 else
1459 dest = foldersel_folder_sel_full(NULL, FOLDER_SEL_MOVE, NULL,
1460 _("Select folder to move"));
1461 if (!dest || dest->stype == F_VIRTUAL || !FOLDER_ITEM_CAN_ADD(dest)) {
1462 g_slist_free(mlist);
1463 return;
1464 }
1465 if (summary_is_locked(main_window_get()->summaryview)) {
1466 g_slist_free(mlist);
1467 return;
1468 }
1469
1470 for (cur = mlist; cur != NULL; cur = cur->next) {
1471 msginfo = (MsgInfo *)cur->data;
1472 if (!msginfo)
1473 continue;
1474
1475 if (is_copy) {
1476 folder_item_copy_msg(dest, msginfo);
1477 } else {
1478 folder_item_move_msg(dest, msginfo);
1479 }
1480 }
1481
1482 folderview_update_all_updated(TRUE);
1483 g_slist_free(mlist);
1484 }
1485
query_search_create_filter(FilterCreateType type)1486 static void query_search_create_filter(FilterCreateType type)
1487 {
1488 GSList *mlist;
1489 MsgInfo *msginfo;
1490 gchar *header = NULL;
1491 gchar *key = NULL;
1492
1493 mlist = query_search_get_selected_msg_list();
1494 if (!mlist)
1495 return;
1496 msginfo = (MsgInfo *)mlist->data;
1497 g_slist_free(mlist);
1498
1499 filter_get_keyword_from_msg(msginfo, &header, &key, type);
1500 prefs_filter_open(msginfo, header, key);
1501
1502 g_free(header);
1503 g_free(key);
1504 }
1505
reply_cb(void)1506 static void reply_cb(void)
1507 {
1508 query_search_reply_forward(COMPOSE_REPLY);
1509 }
1510
reply_all_cb(void)1511 static void reply_all_cb(void)
1512 {
1513 query_search_reply_forward(COMPOSE_REPLY_TO_ALL);
1514 }
1515
reply_sender_cb(void)1516 static void reply_sender_cb(void)
1517 {
1518 query_search_reply_forward(COMPOSE_REPLY_TO_SENDER);
1519 }
1520
reply_list_cb(void)1521 static void reply_list_cb(void)
1522 {
1523 query_search_reply_forward(COMPOSE_REPLY_TO_LIST);
1524 }
1525
forward_cb(void)1526 static void forward_cb(void)
1527 {
1528 query_search_reply_forward(COMPOSE_FORWARD);
1529 }
1530
forward_attach_cb(void)1531 static void forward_attach_cb(void)
1532 {
1533 query_search_reply_forward(COMPOSE_FORWARD_AS_ATTACH);
1534 }
1535
redirect_cb(void)1536 static void redirect_cb(void)
1537 {
1538 query_search_reply_forward(COMPOSE_REDIRECT);
1539 }
1540
move_cb(void)1541 static void move_cb(void)
1542 {
1543 query_search_move_copy(FALSE);
1544 }
1545
copy_cb(void)1546 static void copy_cb(void)
1547 {
1548 query_search_move_copy(TRUE);
1549 }
1550
add_address_cb(void)1551 static void add_address_cb(void)
1552 {
1553 GSList *mlist;
1554 MsgInfo *msginfo;
1555 gchar from[BUFFSIZE];
1556
1557 mlist = query_search_get_selected_msg_list();
1558 if (!mlist)
1559 return;
1560 msginfo = (MsgInfo *)mlist->data;
1561 g_slist_free(mlist);
1562
1563 strncpy2(from, msginfo->from, sizeof(from));
1564 extract_address(from);
1565 addressbook_add_contact(msginfo->fromname, from, NULL);
1566 }
1567
create_filter_auto_cb(void)1568 static void create_filter_auto_cb(void)
1569 {
1570 query_search_create_filter(FLT_BY_AUTO);
1571 }
1572
create_filter_from_cb(void)1573 static void create_filter_from_cb(void)
1574 {
1575 query_search_create_filter(FLT_BY_FROM);
1576 }
1577
create_filter_to_cb(void)1578 static void create_filter_to_cb(void)
1579 {
1580 query_search_create_filter(FLT_BY_TO);
1581 }
1582
create_filter_subject_cb(void)1583 static void create_filter_subject_cb(void)
1584 {
1585 query_search_create_filter(FLT_BY_SUBJECT);
1586 }
1587
open_cb(void)1588 static void open_cb(void)
1589 {
1590 query_search_open_msg(TRUE);
1591 }
1592
open_source_cb(void)1593 static void open_source_cb(void)
1594 {
1595 GSList *mlist;
1596 MsgInfo *msginfo;
1597 SourceWindow *srcwin;
1598
1599 mlist = query_search_get_selected_msg_list();
1600 if (!mlist)
1601 return;
1602 msginfo = (MsgInfo *)mlist->data;
1603 g_slist_free(mlist);
1604
1605 srcwin = source_window_create();
1606 source_window_show_msg(srcwin, msginfo);
1607 source_window_show(srcwin);
1608 }
1609
print_cb(void)1610 static void print_cb(void)
1611 {
1612 GSList *mlist;
1613
1614 mlist = query_search_get_selected_msg_list();
1615 if (!mlist)
1616 return;
1617 printing_print_messages(mlist, FALSE);
1618 g_slist_free(mlist);
1619 }
1620
query_search_cmp_by_folder(GtkTreeModel * model,GtkTreeIter * a,GtkTreeIter * b,gpointer data)1621 static gint query_search_cmp_by_folder(GtkTreeModel *model,
1622 GtkTreeIter *a, GtkTreeIter *b,
1623 gpointer data)
1624 {
1625 gchar *folder_a = NULL, *folder_b = NULL;
1626 MsgInfo *msginfo_a = NULL, *msginfo_b = NULL;
1627 gint ret;
1628
1629 gtk_tree_model_get(model, a, COL_FOLDER, &folder_a, COL_MSGINFO,
1630 &msginfo_a, -1);
1631 gtk_tree_model_get(model, b, COL_FOLDER, &folder_b, COL_MSGINFO,
1632 &msginfo_b, -1);
1633
1634 if (!folder_a || !folder_b || !msginfo_a || !msginfo_b)
1635 return 0;
1636
1637 ret = g_ascii_strcasecmp(folder_a, folder_b);
1638 return (ret != 0) ? ret : (msginfo_a->date_t - msginfo_b->date_t);
1639 }
1640
query_search_cmp_by_subject(GtkTreeModel * model,GtkTreeIter * a,GtkTreeIter * b,gpointer data)1641 static gint query_search_cmp_by_subject(GtkTreeModel *model,
1642 GtkTreeIter *a, GtkTreeIter *b,
1643 gpointer data)
1644 {
1645 MsgInfo *msginfo_a = NULL, *msginfo_b = NULL;
1646 gint ret;
1647
1648 gtk_tree_model_get(model, a, COL_MSGINFO, &msginfo_a, -1);
1649 gtk_tree_model_get(model, b, COL_MSGINFO, &msginfo_b, -1);
1650
1651 if (!msginfo_a || !msginfo_b)
1652 return 0;
1653
1654 if (!msginfo_a->subject)
1655 return -(msginfo_b->subject != NULL);
1656 if (!msginfo_b->subject)
1657 return (msginfo_a->subject != NULL);
1658
1659 ret = subject_compare_for_sort(msginfo_a->subject, msginfo_b->subject);
1660 return (ret != 0) ? ret : (msginfo_a->date_t - msginfo_b->date_t);
1661 }
1662
query_search_cmp_by_from(GtkTreeModel * model,GtkTreeIter * a,GtkTreeIter * b,gpointer data)1663 static gint query_search_cmp_by_from(GtkTreeModel *model,
1664 GtkTreeIter *a, GtkTreeIter *b,
1665 gpointer data)
1666 {
1667 MsgInfo *msginfo_a = NULL, *msginfo_b = NULL;
1668 gint ret;
1669
1670 gtk_tree_model_get(model, a, COL_MSGINFO, &msginfo_a, -1);
1671 gtk_tree_model_get(model, b, COL_MSGINFO, &msginfo_b, -1);
1672
1673 if (!msginfo_a || !msginfo_b)
1674 return 0;
1675
1676 if (!msginfo_a->fromname)
1677 return -(msginfo_b->fromname != NULL);
1678 if (!msginfo_b->fromname)
1679 return (msginfo_a->fromname != NULL);
1680
1681 ret = g_ascii_strcasecmp(msginfo_a->fromname, msginfo_b->fromname);
1682 return (ret != 0) ? ret : (msginfo_a->date_t - msginfo_b->date_t);
1683 }
1684
query_search_cmp_by_date(GtkTreeModel * model,GtkTreeIter * a,GtkTreeIter * b,gpointer data)1685 static gint query_search_cmp_by_date(GtkTreeModel *model,
1686 GtkTreeIter *a, GtkTreeIter *b,
1687 gpointer data)
1688 {
1689 MsgInfo *msginfo_a = NULL, *msginfo_b = NULL;
1690
1691 gtk_tree_model_get(model, a, COL_MSGINFO, &msginfo_a, -1);
1692 gtk_tree_model_get(model, b, COL_MSGINFO, &msginfo_b, -1);
1693
1694 if (!msginfo_a || !msginfo_b)
1695 return 0;
1696
1697 return msginfo_a->date_t - msginfo_b->date_t;
1698 }
1699