2 * Dialog boxes for (display and capture) filter editing
6 * Wireshark - Network traffic analyzer
7 * By Gerald Combs <gerald@wireshark.org>
8 * Copyright 1998 Gerald Combs
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License
12 * as published by the Free Software Foundation; either version 2
13 * of the License, or (at your option) any later version.
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
31 #include <epan/filesystem.h>
32 #include <epan/prefs.h>
33 #include <epan/proto.h>
35 #include "../filters.h"
36 #include "ui/simple_dialog.h"
37 #include "ui/main_statusbar.h"
39 #include "ui/gtk/main.h"
40 #include "ui/gtk/filter_dlg.h"
41 #include "ui/gtk/dlg_utils.h"
42 #include "ui/gtk/gui_utils.h"
43 #include "ui/gtk/dfilter_expr_dlg.h"
44 #include "ui/gtk/stock_icons.h"
45 #include "ui/gtk/gtkglobals.h"
46 #include "ui/gtk/help_dlg.h"
47 #include "ui/gtk/filter_autocomplete.h"
49 #include "ui/gtk/old-gtk-compat.h"
51 #define E_FILT_DIALOG_PTR_KEY "filter_dialog_ptr"
52 #define E_FILT_BUTTON_PTR_KEY "filter_button_ptr"
53 #define E_FILT_PARENT_FILTER_TE_KEY "filter_parent_filter_te"
54 #define E_FILT_CONSTRUCT_ARGS_KEY "filter_construct_args"
55 #define E_FILT_LIST_ITEM_MODEL_KEY "filter_list_item_model"
56 #define E_FILT_LBL_KEY "filter_label"
57 #define E_FILT_FILTER_L_KEY "filter_filter_l"
58 #define E_FILT_CHG_BT_KEY "filter_chg_bt"
59 #define E_FILT_COPY_BT_KEY "filter_copy_bt"
60 #define E_FILT_DEL_BT_KEY "filter_del_bt"
61 #define E_FILT_NAME_TE_KEY "filter_name_te"
62 #define E_FILT_DBLFUNC_KEY "filter_dblfunc"
63 #define E_FILT_DBLARG_KEY "filter_dblarg"
64 #define E_FILT_DBLACTIVATE_KEY "filter_dblactivate"
66 typedef struct _filter_cb_data
{
71 static GtkWidget
*filter_dialog_new(GtkWidget
*button
, GtkWidget
*filter_te
,
72 filter_list_type_t list_type
,
73 construct_args_t
*construct_args
);
74 static void filter_dlg_dclick(GtkWidget
*dummy
, gpointer main_w_arg
,
76 static void filter_dlg_ok_cb(GtkWidget
*ok_bt
, gpointer data
);
77 static void filter_dlg_apply_cb(GtkWidget
*apply_bt
, gpointer data
);
78 static void filter_apply(GtkWidget
*main_w
, gboolean destroy
);
79 static void filter_dlg_save(filter_list_type_t list_type
);
80 static void filter_dlg_save_cb(GtkWidget
*save_bt
, gpointer parent_w
);
81 static void filter_dlg_destroy_cb(GtkWidget
*win
, gpointer data
);
84 filter_dlg_delete_event_cb(GtkWidget
*prefs_w
, GdkEvent
*event
, gpointer data
);
86 filter_dlg_cancel_cb(GtkWidget
*cancel_bt
, gpointer data
);
88 static gboolean
filter_sel_list_button_cb(GtkWidget
*, GdkEventButton
*,
90 static void filter_sel_list_cb(GtkTreeSelection
*, gpointer
);
91 static void filter_new_bt_clicked_cb(GtkWidget
*, gpointer
);
92 static void filter_del_bt_clicked_cb(GtkWidget
*, gpointer
);
93 static void filter_name_te_changed_cb(GtkWidget
*, gpointer
);
96 /* Create a filter dialog for constructing a capture filter.
98 This is to be used as a callback for a button next to a text entry box,
99 which, when clicked, pops up this dialog to allow you to construct a
100 display filter by browsing the list of saved filters (the dialog
101 for constructing expressions assumes display filter syntax, not
102 capture filter syntax). The "OK" button sets the text entry box to the
103 constructed filter and activates that text entry box (which should have
104 no effect in the main capture dialog); this dialog is then dismissed. */
106 capture_filter_construct_cb(GtkWidget
*w
, gpointer user_data _U_
)
108 GtkWidget
*filter_browse_w
;
109 GtkWidget
*parent_filter_te
;
110 /* No Apply button, and "OK" just sets our text widget, it doesn't
111 activate it (i.e., it doesn't cause us to try to open the file). */
112 static construct_args_t args
= {
113 "Wireshark: Capture Filter",
119 /* Has a filter dialog box already been opened for that button? */
120 filter_browse_w
= (GtkWidget
*)g_object_get_data(G_OBJECT(w
), E_FILT_DIALOG_PTR_KEY
);
122 if (filter_browse_w
!= NULL
) {
123 /* Yes. Just re-activate that dialog box. */
124 reactivate_window(filter_browse_w
);
128 /* No. Get the text entry attached to the button. */
129 parent_filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(w
), E_FILT_TE_PTR_KEY
);
131 /* Now create a new dialog, without an "Add Expression..." button. */
132 filter_dialog_new(w
, parent_filter_te
, CFILTER_LIST
, &args
);
136 /* Create a filter dialog for constructing a display filter.
138 This is to be used as a callback for a button next to a text entry box,
139 which, when clicked, pops up this dialog to allow you to construct a
140 display filter by browsing the list of saved filters and/or by adding
141 test expressions constructed with another dialog. The "OK" button
142 sets the text entry box to the constructed filter and activates that
143 text entry box, causing the filter to be used; this dialog is then
146 If "wants_apply_button" is non-null, we add an "Apply" button that
147 acts like "OK" but doesn't dismiss this dialog. */
149 display_filter_construct_cb(GtkWidget
*w
, gpointer construct_args_ptr
)
151 construct_args_t
*construct_args
= (construct_args_t
*)construct_args_ptr
;
152 GtkWidget
*filter_browse_w
;
153 GtkWidget
*parent_filter_te
;
155 /* Has a filter dialog box already been opened for the button? */
156 filter_browse_w
= (GtkWidget
*)g_object_get_data(G_OBJECT(w
), E_FILT_DIALOG_PTR_KEY
);
158 if (filter_browse_w
!= NULL
) {
159 /* Yes. Just re-activate that dialog box. */
160 reactivate_window(filter_browse_w
);
164 /* No. Get the text entry attached to the button. */
165 parent_filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(w
), E_FILT_TE_PTR_KEY
);
167 /* Now create a new dialog, possibly with an "Apply" button, and
168 definitely with an "Add Expression..." button. */
169 filter_dialog_new(w
, parent_filter_te
, DFILTER_LIST
, construct_args
);
172 /* Should be called when a button that creates filters is destroyed; it
173 destroys any filter dialog created by that button. */
175 filter_button_destroy_cb(GtkWidget
*button
, gpointer user_data _U_
)
179 /* Is there a filter edit/selection dialog associated with this
181 filter_w
= (GtkWidget
*)g_object_get_data(G_OBJECT(button
), E_FILT_DIALOG_PTR_KEY
);
183 if (filter_w
!= NULL
) {
184 /* Yes. Break the association, and destroy the dialog. */
185 g_object_set_data(G_OBJECT(button
), E_FILT_DIALOG_PTR_KEY
, NULL
);
186 window_destroy(filter_w
);
191 static GtkWidget
*global_cfilter_w
;
193 /* Create a filter dialog for editing capture filters; this is to be used
194 as a callback for menu items, toolbars, etc.. */
196 cfilter_dialog_cb(GtkWidget
*w _U_
)
198 /* No Apply button, and there's no text widget to set, much less
199 activate, on "OK". */
200 static construct_args_t args
= {
201 "Wireshark: Capture Filter",
207 /* Has a filter dialog box already been opened for editing
209 if (global_cfilter_w
!= NULL
) {
210 /* Yes. Just reactivate it. */
211 reactivate_window(global_cfilter_w
);
216 * No. Create one; we didn't pop this up as a result of pressing
217 * a button next to some text entry field, so don't associate it
218 * with a text entry field or button.
220 global_cfilter_w
= filter_dialog_new(NULL
, NULL
, CFILTER_LIST
, &args
);
224 /* Create a filter dialog for editing display filters; this is to be used
225 as a callback for menu items, toolbars, etc.. */
227 dfilter_dialog_cb(GtkWidget
*w _U_
)
229 static construct_args_t args
= {
230 "Wireshark: Display Filter",
236 display_filter_construct_cb((GtkWidget
*)g_object_get_data(G_OBJECT(top_level
), E_FILT_BT_PTR_KEY
), &args
);
239 /* List of capture filter dialogs, so that if the list of filters changes
240 (the model, if you will), we can update all of their lists displaying
241 the filters (the views). */
242 static GList
*cfilter_dialogs
;
244 /* List of display filter dialogs, so that if the list of filters changes
245 (the model, if you will), we can update all of their lists displaying
246 the filters (the views). */
247 static GList
*dfilter_dialogs
;
250 remember_filter_dialog(GtkWidget
*main_w
, GList
**filter_dialogs
)
252 *filter_dialogs
= g_list_append(*filter_dialogs
, main_w
);
255 /* Remove a filter dialog from the specified list of filter_dialogs. */
257 forget_filter_dialog(GtkWidget
*main_w
, filter_list_type_t list_type
)
261 case CFILTER_EDITED_LIST
:
262 cfilter_dialogs
= g_list_remove(cfilter_dialogs
, main_w
);
265 case DFILTER_EDITED_LIST
:
266 dfilter_dialogs
= g_list_remove(dfilter_dialogs
, main_w
);
270 g_assert_not_reached();
275 /* Get the dialog list corresponding to a particular filter list. */
277 get_filter_dialog_list(filter_list_type_t list_type
)
281 case CFILTER_EDITED_LIST
:
282 return cfilter_dialogs
;
284 case DFILTER_EDITED_LIST
:
285 return dfilter_dialogs
;
288 g_assert_not_reached();
295 fill_list(GtkWidget
*main_w
, filter_list_type_t list_type
, const gchar
*filter_te_str
)
299 GtkTreeView
*filter_l
;
302 GtkTreeIter
*l_select
= NULL
;
304 filter_l
= GTK_TREE_VIEW(g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
));
305 store
= GTK_LIST_STORE(gtk_tree_view_get_model(filter_l
));
308 fl_entry
= get_filter_list_first(list_type
);
309 while (fl_entry
!= NULL
) {
310 filt
= (filter_def
*) fl_entry
->data
;
311 gtk_list_store_append(store
, &iter
);
312 gtk_list_store_set(store
, &iter
, 0, filt
->name
,
315 if (filter_te_str
&& filt
->strval
) {
316 if (strcmp(filter_te_str
, filt
->strval
) == 0) {
318 * XXX - We're assuming that we can just copy a GtkTreeIter
319 * and use it later without any crashes. This may not be a
322 l_select
= (GtkTreeIter
*)g_memdup(&iter
, sizeof(iter
));
326 fl_entry
= fl_entry
->next
;
333 clear_list(GtkWidget
*main_w
) {
334 GtkWidget
*filter_l
= g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
335 GtkTreeModel
*model
= gtk_tree_view_get_model(GTK_TREE_VIEW(filter_l
));
337 gtk_list_store_clear(GTK_LIST_STORE(model
));
342 filter_dialog_new(GtkWidget
*button
, GtkWidget
*parent_filter_te
,
343 filter_list_type_t list_type
, construct_args_t
*construct_args
)
345 GtkWidget
*main_w
, /* main window */
346 *main_vb
, /* main container */
347 *bbox
, /* button container */
348 *ok_bt
, /* "OK" button */
349 *apply_bt
, /* "Apply" button */
350 *save_bt
, /* "Save" button */
351 *cancel_bt
, /* "Cancel" button */
352 *help_bt
; /* "Help" button */
353 GtkWidget
*filter_vb
, /* filter settings box */
372 static filter_list_type_t cfilter_list_type
= CFILTER_EDITED_LIST
;
373 static filter_list_type_t dfilter_list_type
= DFILTER_EDITED_LIST
;
374 filter_list_type_t
*filter_list_type_p
;
375 GList
**filter_dialogs
;
376 const gchar
*filter_te_str
= NULL
;
378 GtkCellRenderer
*renderer
;
379 GtkTreeViewColumn
*column
;
380 GtkTreeSelection
*sel
;
381 GtkTreeIter
*l_select
;
382 const gchar
*list_name
= NULL
;
384 /* Get a pointer to a static variable holding the type of filter on
385 which we're working, so we can pass that pointer to callback
390 filter_dialogs
= &cfilter_dialogs
;
391 filter_list_type_p
= &cfilter_list_type
;
392 list_type
= CFILTER_EDITED_LIST
;
393 list_name
= "Capture Filter";
397 filter_dialogs
= &dfilter_dialogs
;
398 filter_list_type_p
= &dfilter_list_type
;
399 list_type
= DFILTER_EDITED_LIST
;
400 list_name
= "Display Filter";
404 g_assert_not_reached();
405 filter_dialogs
= NULL
;
406 filter_list_type_p
= NULL
;
410 main_w
= dlg_conf_window_new(construct_args
->title
);
411 gtk_window_set_modal(GTK_WINDOW(main_w
), TRUE
);
412 gtk_window_set_default_size(GTK_WINDOW(main_w
), 400, 400);
413 g_object_set_data(G_OBJECT(main_w
), E_FILT_CONSTRUCT_ARGS_KEY
, construct_args
);
415 main_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 0, FALSE
);
416 gtk_container_set_border_width(GTK_CONTAINER(main_vb
), 5);
417 gtk_container_add(GTK_CONTAINER(main_w
), main_vb
);
418 gtk_widget_show(main_vb
);
420 /* Make sure everything is set up */
421 if (parent_filter_te
)
422 filter_te_str
= gtk_entry_get_text(GTK_ENTRY(parent_filter_te
));
424 /* Container for each row of widgets */
425 filter_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 0, FALSE
);
426 gtk_container_set_border_width(GTK_CONTAINER(filter_vb
), 0);
427 gtk_box_pack_start(GTK_BOX (main_vb
), filter_vb
, TRUE
, TRUE
, 0);
428 gtk_widget_show(filter_vb
);
430 /* Top row: Buttons and filter list */
431 top_hb
= ws_gtk_box_new(GTK_ORIENTATION_HORIZONTAL
, 0, FALSE
);
432 gtk_box_pack_start(GTK_BOX (filter_vb
), top_hb
, TRUE
, TRUE
, 0);
434 gtk_widget_show(top_hb
);
436 edit_fr
= gtk_frame_new("Edit");
437 gtk_box_pack_start(GTK_BOX(top_hb
), edit_fr
, FALSE
, FALSE
, 0);
438 gtk_widget_show(edit_fr
);
440 list_bb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 0, TRUE
);
441 gtk_container_set_border_width(GTK_CONTAINER(list_bb
), 5);
442 gtk_container_add(GTK_CONTAINER(edit_fr
), list_bb
);
443 gtk_widget_show(list_bb
);
445 new_bt
= gtk_button_new_from_stock(GTK_STOCK_NEW
);
446 g_signal_connect(new_bt
, "clicked", G_CALLBACK(filter_new_bt_clicked_cb
), filter_list_type_p
);
447 gtk_widget_show(new_bt
);
448 gtk_box_pack_start (GTK_BOX (list_bb
), new_bt
, FALSE
, FALSE
, 0);
449 gtk_widget_set_tooltip_text(new_bt
, "Create a new filter at the end of the list (with the current properties)");
451 del_bt
= gtk_button_new_from_stock(GTK_STOCK_DELETE
);
452 gtk_widget_set_sensitive(del_bt
, FALSE
);
453 g_signal_connect(del_bt
, "clicked", G_CALLBACK(filter_del_bt_clicked_cb
), filter_list_type_p
);
454 g_object_set_data(G_OBJECT(main_w
), E_FILT_DEL_BT_KEY
, del_bt
);
455 gtk_widget_show(del_bt
);
456 gtk_box_pack_start (GTK_BOX (list_bb
), del_bt
, FALSE
, FALSE
, 0);
457 gtk_widget_set_tooltip_text(del_bt
, "Delete the selected filter");
459 filter_fr
= gtk_frame_new(list_name
);
460 gtk_box_pack_start(GTK_BOX(top_hb
), filter_fr
, TRUE
, TRUE
, 0);
461 gtk_widget_show(filter_fr
);
463 filter_sc
= scrolled_window_new(NULL
, NULL
);
464 gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(filter_sc
),
467 gtk_container_set_border_width (GTK_CONTAINER (filter_sc
), 5);
468 gtk_container_add(GTK_CONTAINER(filter_fr
), filter_sc
);
469 gtk_widget_show(filter_sc
);
471 store
= gtk_list_store_new(2, G_TYPE_STRING
, G_TYPE_POINTER
);
472 filter_l
= tree_view_new(GTK_TREE_MODEL(store
));
473 gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(filter_l
), FALSE
);
474 renderer
= gtk_cell_renderer_text_new();
475 column
= gtk_tree_view_column_new_with_attributes("", renderer
, "text",
477 gtk_tree_view_column_set_sort_column_id(column
, 0);
478 gtk_tree_view_append_column(GTK_TREE_VIEW(filter_l
), column
);
479 sel
= gtk_tree_view_get_selection(GTK_TREE_VIEW(filter_l
));
480 gtk_tree_selection_set_mode(sel
, GTK_SELECTION_SINGLE
);
481 g_signal_connect(sel
, "changed", G_CALLBACK(filter_sel_list_cb
), NULL
);
482 g_signal_connect(filter_l
, "button_press_event", G_CALLBACK(filter_sel_list_button_cb
),
484 g_object_set_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
, filter_l
);
485 gtk_container_add(GTK_CONTAINER(filter_sc
), filter_l
);
486 gtk_widget_show(filter_l
);
488 g_object_set_data(G_OBJECT(filter_l
), E_FILT_DBLFUNC_KEY
, filter_dlg_dclick
);
489 g_object_set_data(G_OBJECT(filter_l
), E_FILT_DBLARG_KEY
, main_w
);
490 /* This is a Boolean, but we make it a non-null pointer for TRUE
491 and a null pointer for FALSE, as object data is a pointer. */
492 g_object_set_data(G_OBJECT(filter_l
), E_FILT_DBLACTIVATE_KEY
,
493 construct_args
->activate_on_ok
? (gpointer
)"" : NULL
);
496 l_select
= fill_list(main_w
, list_type
, filter_te_str
);
498 g_object_unref(G_OBJECT(store
));
501 props_fr
= gtk_frame_new("Properties");
502 gtk_box_pack_start(GTK_BOX(filter_vb
), props_fr
, FALSE
, FALSE
, 0);
503 gtk_widget_show(props_fr
);
505 props_vb
= ws_gtk_box_new(GTK_ORIENTATION_VERTICAL
, 3, FALSE
);
506 gtk_container_set_border_width(GTK_CONTAINER(props_vb
), 5);
507 gtk_container_add(GTK_CONTAINER(props_fr
), props_vb
);
508 gtk_widget_show(props_vb
);
510 /* row: Filter name entry */
511 middle_hb
= ws_gtk_box_new(GTK_ORIENTATION_HORIZONTAL
, 3, FALSE
);
512 gtk_box_pack_start(GTK_BOX (props_vb
), middle_hb
, TRUE
, TRUE
, 0);
513 gtk_widget_show(middle_hb
);
515 name_lb
= gtk_label_new("Filter name:");
516 gtk_box_pack_start(GTK_BOX(middle_hb
), name_lb
, FALSE
, FALSE
, 0);
517 gtk_widget_show(name_lb
);
519 name_te
= gtk_entry_new();
520 gtk_box_pack_start(GTK_BOX(middle_hb
), name_te
, TRUE
, TRUE
, 0);
521 g_object_set_data(G_OBJECT(main_w
), E_FILT_NAME_TE_KEY
, name_te
);
522 g_signal_connect(name_te
, "changed", G_CALLBACK(filter_name_te_changed_cb
), filter_list_type_p
);
523 gtk_widget_show(name_te
);
525 /* row: Filter text entry */
526 bottom_hb
= ws_gtk_box_new(GTK_ORIENTATION_HORIZONTAL
, 3, FALSE
);
527 gtk_box_pack_start(GTK_BOX (props_vb
), bottom_hb
, TRUE
, TRUE
, 0);
528 gtk_widget_show(bottom_hb
);
530 filter_lb
= gtk_label_new("Filter string:");
531 gtk_box_pack_start(GTK_BOX(bottom_hb
), filter_lb
, FALSE
, FALSE
, 0);
532 gtk_widget_show(filter_lb
);
534 filter_te
= gtk_entry_new();
535 gtk_box_pack_start(GTK_BOX(bottom_hb
), filter_te
, TRUE
, TRUE
, 0);
536 g_object_set_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
, filter_te
);
537 g_signal_connect(filter_te
, "changed", G_CALLBACK(filter_name_te_changed_cb
), filter_list_type_p
);
538 if (list_type
== DFILTER_EDITED_LIST
) {
539 colorize_filter_te_as_empty(filter_te
);
541 g_object_set_data(G_OBJECT(main_w
), E_FILT_AUTOCOMP_PTR_KEY
, NULL
);
542 g_signal_connect(filter_te
, "key-press-event", G_CALLBACK (filter_string_te_key_pressed_cb
), NULL
);
543 g_signal_connect(main_w
, "key-press-event", G_CALLBACK (filter_parent_dlg_key_pressed_cb
), NULL
);
545 gtk_widget_show(filter_te
);
547 g_object_set_data(G_OBJECT(main_w
), E_FILT_PARENT_FILTER_TE_KEY
, parent_filter_te
);
549 if (list_type
== DFILTER_EDITED_LIST
) {
550 gtk_widget_set_tooltip_text(filter_te
,
551 "Enter a display filter. "
552 "The background color of this field is changed by a continuous syntax check"
553 " (green is valid, red is invalid, yellow may have unexpected results).");
555 /* Create the "Add Expression..." button, to pop up a dialog
556 for constructing filter comparison expressions. */
557 add_expression_bt
= gtk_button_new_from_stock(WIRESHARK_STOCK_ADD_EXPRESSION
);
558 g_signal_connect(add_expression_bt
, "clicked", G_CALLBACK(filter_add_expr_bt_cb
), main_w
);
559 gtk_box_pack_start(GTK_BOX(bottom_hb
), add_expression_bt
, FALSE
, FALSE
, 0);
560 gtk_widget_show(add_expression_bt
);
561 gtk_widget_set_tooltip_text(add_expression_bt
, "Add an expression to the filter string");
565 /* button row (create all possible buttons and hide the unrequired later - it's a lot easier) */
566 bbox
= dlg_button_row_new(GTK_STOCK_OK
, GTK_STOCK_APPLY
, GTK_STOCK_SAVE
, GTK_STOCK_CANCEL
, GTK_STOCK_HELP
, NULL
);
567 gtk_box_pack_start(GTK_BOX(main_vb
), bbox
, FALSE
, FALSE
, 5);
568 gtk_widget_show(bbox
);
570 ok_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), GTK_STOCK_OK
);
571 g_signal_connect(ok_bt
, "clicked", G_CALLBACK(filter_dlg_ok_cb
), filter_list_type_p
);
572 gtk_widget_set_tooltip_text(ok_bt
, "Apply the filters and close this dialog");
574 /* Catch the "activate" signal on the filter name and filter
575 expression text entries, so that if the user types Return
576 there, we act as if the "OK" button had been selected, as
577 happens if Return is typed if some widget that *doesn't*
578 handle the Return key has the input focus. */
579 if (parent_filter_te
!= NULL
) {
580 dlg_set_activate(name_te
, ok_bt
);
581 dlg_set_activate(filter_te
, ok_bt
);
584 apply_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), GTK_STOCK_APPLY
);
585 g_signal_connect(apply_bt
, "clicked", G_CALLBACK(filter_dlg_apply_cb
), filter_list_type_p
);
586 gtk_widget_set_tooltip_text(apply_bt
, "Apply the filters and keep this dialog open");
588 save_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), GTK_STOCK_SAVE
);
589 g_signal_connect(save_bt
, "clicked", G_CALLBACK(filter_dlg_save_cb
), filter_list_type_p
);
590 gtk_widget_set_tooltip_text(save_bt
, "Save the filters permanently and keep this dialog open");
592 cancel_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), GTK_STOCK_CANCEL
);
593 gtk_widget_set_tooltip_text(cancel_bt
, "Cancel the changes");
594 g_signal_connect(cancel_bt
, "clicked", G_CALLBACK(filter_dlg_cancel_cb
), filter_list_type_p
);
595 window_set_cancel_button(main_w
, cancel_bt
, NULL
);
597 help_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(bbox
), GTK_STOCK_HELP
);
598 if (list_type
== CFILTER_EDITED_LIST
) {
599 g_signal_connect(help_bt
, "clicked", G_CALLBACK(topic_cb
), (gpointer
)HELP_CAPTURE_FILTERS_DIALOG
);
601 g_signal_connect(help_bt
, "clicked", G_CALLBACK(topic_cb
), (gpointer
)HELP_DISPLAY_FILTERS_DIALOG
);
603 gtk_widget_set_tooltip_text(help_bt
, "Show topic specific help");
606 gtk_widget_grab_default(ok_bt
);
609 remember_filter_dialog(main_w
, filter_dialogs
);
611 if (button
!= NULL
) {
612 /* This dialog box was created by a "Filter" button.
613 Set the E_FILT_BUTTON_PTR_KEY for the new dialog to point to
615 g_object_set_data(G_OBJECT(main_w
), E_FILT_BUTTON_PTR_KEY
, button
);
617 /* Set the E_FILT_DIALOG_PTR_KEY for the button to point to us */
618 g_object_set_data(G_OBJECT(button
), E_FILT_DIALOG_PTR_KEY
, main_w
);
621 /* DO SELECTION THINGS *AFTER* SHOWING THE DIALOG! */
622 /* otherwise the updatings can get confused */
624 gtk_tree_selection_select_iter(sel
, l_select
);
626 } else if (filter_te_str
&& filter_te_str
[0]) {
627 gtk_entry_set_text(GTK_ENTRY(name_te
), "New filter");
628 gtk_entry_set_text(GTK_ENTRY(filter_te
), filter_te_str
);
631 g_signal_connect(main_w
, "delete_event", G_CALLBACK(filter_dlg_delete_event_cb
), filter_list_type_p
);
632 g_signal_connect(main_w
, "destroy", G_CALLBACK(filter_dlg_destroy_cb
), filter_list_type_p
);
634 gtk_widget_show(main_w
);
636 if(construct_args
->modal_and_transient
) {
637 parent
= gtk_widget_get_parent_window(parent_filter_te
);
638 gdk_window_set_transient_for(gtk_widget_get_window(main_w
), parent
);
639 gtk_window_set_modal(GTK_WINDOW(main_w
), TRUE
);
642 /* hide the Ok button, if we don't have to apply it and our caller wants a Save button */
643 if (parent_filter_te
== NULL
&& prefs
.gui_use_pref_save
) {
644 gtk_widget_hide(ok_bt
);
647 /* hide the Apply button, if our caller don't wants one */
648 if (!construct_args
->wants_apply_button
) {
649 gtk_widget_hide(apply_bt
);
652 /* hide the Save button if the user uses implicit save */
653 if (!prefs
.gui_use_pref_save
) {
654 gtk_widget_hide(save_bt
);
657 window_present(main_w
);
663 filter_dlg_dclick(GtkWidget
*filter_l
, gpointer main_w_arg
, gpointer activate
)
665 GtkWidget
*main_w
= GTK_WIDGET(main_w_arg
);
666 GtkWidget
*parent_filter_te
=
667 (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_PARENT_FILTER_TE_KEY
);
670 GtkTreeSelection
*sel
;
674 sel
= gtk_tree_view_get_selection(GTK_TREE_VIEW(filter_l
));
676 if (parent_filter_te
!= NULL
) {
678 * We have a text entry widget associated with this dialog
679 * box; is one of the filters in the list selected?
681 if (gtk_tree_selection_get_selected(sel
, &model
, &iter
)) {
683 * Yes. Is there a filter definition for that filter?
685 gtk_tree_model_get(model
, &iter
, 1, &flp
, -1);
688 * Yes - put it in the text entry widget.
690 filt
= (filter_def
*) flp
->data
;
691 gtk_entry_set_text(GTK_ENTRY(parent_filter_te
),
695 * Are we supposed to cause the filter we
696 * put there to be applied?
698 if (activate
!= NULL
) {
702 g_signal_emit_by_name(G_OBJECT(parent_filter_te
), "activate", NULL
);
708 window_destroy(main_w
);
712 filter_dlg_ok_cb(GtkWidget
*ok_bt
, gpointer data
)
714 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
717 * Destroy the dialog box and apply the filter.
719 filter_apply(gtk_widget_get_toplevel(ok_bt
), TRUE
);
721 /* if we don't have a Save button, just save the settings now */
722 if (!prefs
.gui_use_pref_save
) {
723 filter_dlg_save(list_type
);
728 filter_dlg_apply_cb(GtkWidget
*apply_bt
, gpointer data
)
730 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
733 * Apply the filter, but don't destroy the dialog box.
735 filter_apply(gtk_widget_get_toplevel(apply_bt
), FALSE
);
737 /* if we don't have a Save button, just save the settings now */
738 if (!prefs
.gui_use_pref_save
) {
739 filter_dlg_save(list_type
);
744 filter_apply(GtkWidget
*main_w
, gboolean destroy
)
746 construct_args_t
*construct_args
=
747 (construct_args_t
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_CONSTRUCT_ARGS_KEY
);
748 GtkWidget
*parent_filter_te
=
749 (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_PARENT_FILTER_TE_KEY
);
750 GtkWidget
*filter_te
;
751 const gchar
*filter_string
;
753 if (parent_filter_te
!= NULL
) {
755 * We have a text entry widget associated with this dialog
756 * box; put the filter in our text entry widget into that
759 filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
);
761 (const gchar
*)gtk_entry_get_text(GTK_ENTRY(filter_te
));
762 gtk_entry_set_text(GTK_ENTRY(parent_filter_te
), filter_string
);
768 * Destroy the filter dialog box.
770 window_destroy(main_w
);
773 if (parent_filter_te
!= NULL
) {
775 * We have a text entry widget associated with this dialog
776 * box; activate that widget to cause the filter we put
777 * there to be applied if we're supposed to do so.
779 * We do this after dismissing the filter dialog box,
780 * as activating the widget the dialog box to which
781 * it belongs to be dismissed, and that may cause it
782 * to destroy our dialog box if the filter succeeds.
783 * This means that our subsequent attempt to destroy
786 * We don't know whether it'll destroy our dialog box,
787 * so we can't rely on it to do so. Instead, we
788 * destroy it ourselves, which will clear the
789 * E_FILT_DIALOG_PTR_KEY pointer for their dialog box,
790 * meaning they won't think it has one and won't try
793 if (construct_args
->activate_on_ok
) {
794 g_signal_emit_by_name(G_OBJECT(parent_filter_te
), "activate", NULL
);
801 filter_dlg_save(filter_list_type_t list_type
)
806 const char *filter_type
;
810 case CFILTER_EDITED_LIST
:
811 filter_type
= "capture";
812 list_type
= CFILTER_LIST
;
813 copy_filter_list(CFILTER_LIST
, CFILTER_EDITED_LIST
);
816 case DFILTER_EDITED_LIST
:
817 filter_type
= "display";
818 list_type
= DFILTER_LIST
;
819 copy_filter_list(DFILTER_LIST
, DFILTER_EDITED_LIST
);
823 g_assert_not_reached();
828 /* Create the directory that holds personal configuration files,
830 if (create_persconffile_dir(&pf_dir_path
) == -1) {
831 simple_dialog(ESD_TYPE_ERROR
, ESD_BTN_OK
,
832 "Can't create directory\n\"%s\"\nfor filter files: %s.",
833 pf_dir_path
, g_strerror(errno
));
838 save_filter_list(list_type
, &f_path
, &f_save_errno
);
839 if (f_path
!= NULL
) {
840 /* We had an error saving the filter. */
841 simple_dialog(ESD_TYPE_ERROR
, ESD_BTN_OK
,
842 "Could not save to your %s filter file\n\"%s\": %s.",
843 filter_type
, f_path
, g_strerror(f_save_errno
));
850 filter_dlg_save_cb(GtkWidget
*save_bt _U_
, gpointer data
)
852 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
854 filter_dlg_save(list_type
);
858 /* update a remaining dialog if another one was cancelled */
860 filter_dlg_update_list_cb(gpointer data
, gpointer user_data
)
862 GtkWidget
*main_w
= data
;
863 filter_list_type_t list_type
= *(filter_list_type_t
*)user_data
;
865 /* refill the list */
867 fill_list(main_w
, list_type
, NULL
);
871 /* cancel button pressed, revert changes and exit dialog */
873 filter_dlg_cancel_cb(GtkWidget
*cancel_bt
, gpointer data
)
875 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
876 GtkWidget
*main_w
= gtk_widget_get_toplevel(cancel_bt
);
877 static GList
*filter_list
;
880 window_destroy(GTK_WIDGET(main_w
));
882 /* if this was the last open filter dialog, revert the changes made */
883 filter_list
= get_filter_dialog_list(list_type
);
884 if(g_list_length(filter_list
) == 0) {
885 /* revert changes in the edited list */
887 case CFILTER_EDITED_LIST
:
888 copy_filter_list(CFILTER_EDITED_LIST
, CFILTER_LIST
);
890 case DFILTER_EDITED_LIST
:
891 copy_filter_list(DFILTER_EDITED_LIST
, DFILTER_LIST
);
894 g_assert_not_reached();
900 /* update other open filter dialogs */
901 g_list_foreach(get_filter_dialog_list(list_type
), filter_dlg_update_list_cb
, &list_type
);
905 /* Treat this as a cancel, by calling "filter_dlg_cancel_cb()" */
907 filter_dlg_delete_event_cb(GtkWidget
*main_w
, GdkEvent
*event _U_
,
910 filter_dlg_cancel_cb(main_w
, data
);
916 filter_dlg_destroy_cb(GtkWidget
*win
, gpointer data
)
918 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
921 /* Get the button that requested that we be popped up, if any.
922 (It should arrange to destroy us if it's destroyed, so
923 that we don't get a pointer to a non-existent window here.) */
924 button
= (GtkWidget
*)g_object_get_data(G_OBJECT(win
), E_FILT_BUTTON_PTR_KEY
);
926 if (button
!= NULL
) {
927 /* Tell it we no longer exist. */
928 g_object_set_data(G_OBJECT(button
), E_FILT_DIALOG_PTR_KEY
, NULL
);
930 /* This is an editing dialog popped up from, for example,
931 a menu item; note that we no longer have one. */
935 case CFILTER_EDITED_LIST
:
936 g_assert(win
== global_cfilter_w
);
937 global_cfilter_w
= NULL
;
941 g_assert_not_reached();
946 /* Remove this from the list of filter dialog windows. */
947 forget_filter_dialog(win
, list_type
);
951 filter_sel_list_button_cb(GtkWidget
*list
, GdkEventButton
*event
,
954 void (* func
)(GtkWidget
*, gpointer
, gpointer
);
956 gpointer func_activate
;
958 if (event
->type
== GDK_2BUTTON_PRESS
) {
959 func
= ( void (*)(GtkWidget
*, void *, void *))g_object_get_data(G_OBJECT(list
), E_FILT_DBLFUNC_KEY
);
960 func_arg
= g_object_get_data(G_OBJECT(list
), E_FILT_DBLARG_KEY
);
961 func_activate
= g_object_get_data(G_OBJECT(list
), E_FILT_DBLACTIVATE_KEY
);
964 (*func
)(list
, func_arg
, func_activate
);
971 filter_sel_list_cb(GtkTreeSelection
*sel
, gpointer data _U_
)
973 GtkWidget
*filter_l
= GTK_WIDGET(gtk_tree_selection_get_tree_view(sel
));
974 GtkWidget
*main_w
= gtk_widget_get_toplevel(filter_l
);
977 GtkWidget
*name_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_NAME_TE_KEY
);
978 GtkWidget
*filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
);
979 GtkWidget
*chg_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_CHG_BT_KEY
);
980 GtkWidget
*copy_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_COPY_BT_KEY
);
981 GtkWidget
*del_bt
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_DEL_BT_KEY
);
983 gchar
*name
= NULL
, *strval
= NULL
;
985 gint sensitivity
= FALSE
;
987 if (gtk_tree_selection_get_selected(sel
, &model
, &iter
)) {
988 gtk_tree_model_get(model
, &iter
, 1, &flp
, -1);
990 filt
= (filter_def
*) flp
->data
;
991 name
= g_strdup(filt
->name
);
992 strval
= g_strdup(filt
->strval
);
998 * Did you know that this function is called when the window is destroyed?
1000 * This means that we have to:
1002 * attach to the top-level window data items containing pointers to
1003 * the widgets we affect here;
1005 * give each of those widgets their own destroy callbacks;
1007 * clear that pointer when the widget is destroyed;
1009 * don't do anything to the widget if the pointer we get back is
1012 * so that if we're called after any of the widgets we'd affect are
1013 * destroyed, we know that we shouldn't do anything to those widgets.
1015 if (name_te
!= NULL
)
1016 gtk_entry_set_text(GTK_ENTRY(name_te
), name
? name
: "");
1017 if (filter_te
!= NULL
)
1018 gtk_entry_set_text(GTK_ENTRY(filter_te
), strval
? strval
: "");
1020 gtk_widget_set_sensitive(chg_bt
, sensitivity
);
1021 if (copy_bt
!= NULL
)
1022 gtk_widget_set_sensitive(copy_bt
, sensitivity
);
1024 gtk_widget_set_sensitive(del_bt
, sensitivity
);
1029 /* To do: add input checking to each of these callbacks */
1031 /* Structure containing arguments to be passed to "new_filter_cb()".
1033 "active_filter_l" is the list in the dialog box in which "New" or
1034 "Copy" was clicked; in that dialog box, but not in any other dialog
1035 box, we select the newly created list item.
1037 "nflp" is the GList member in the model (filter list) for the new
1040 GtkWidget
*active_filter_l
;
1042 } new_filter_cb_args_t
;
1045 new_filter_cb(gpointer data
, gpointer user_data
)
1047 GtkWidget
*main_w
= (GtkWidget
*)data
;
1048 GtkTreeView
*filter_l
;
1049 GtkListStore
*store
;
1051 new_filter_cb_args_t
*args
= (new_filter_cb_args_t
*)user_data
;
1052 filter_def
*nfilt
= (filter_def
*)args
->nflp
->data
;
1054 filter_l
= GTK_TREE_VIEW(g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
));
1055 store
= GTK_LIST_STORE(gtk_tree_view_get_model(filter_l
));
1056 gtk_list_store_append(store
, &iter
);
1057 gtk_list_store_set(store
, &iter
, 0, nfilt
->name
, 1, args
->nflp
, -1);
1058 if (GTK_WIDGET(filter_l
) == args
->active_filter_l
) {
1059 /* Select the item. */
1060 gtk_tree_selection_select_iter(gtk_tree_view_get_selection(filter_l
),
1066 filter_new_bt_clicked_cb(GtkWidget
*w
, gpointer data
)
1068 GtkWidget
*main_w
= gtk_widget_get_toplevel(w
);
1069 GtkWidget
*name_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_NAME_TE_KEY
);
1070 GtkWidget
*filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
);
1071 GtkWidget
*filter_l
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
1072 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
1074 const gchar
*name
, *strval
;
1075 new_filter_cb_args_t args
;
1077 name
= gtk_entry_get_text(GTK_ENTRY(name_te
));
1078 strval
= gtk_entry_get_text(GTK_ENTRY(filter_te
));
1080 /* if the user didn't entered a name, set default one */
1081 if (strlen(name
) == 0) {
1085 /* if the user didn't entered a string value, set default one */
1086 if (strlen(strval
) == 0) {
1090 /* Add a new entry to the filter list. */
1091 fl_entry
= add_to_filter_list(list_type
, name
, strval
);
1093 /* Update all the filter list widgets, not just the one in
1094 the dialog box in which we clicked on "Copy". */
1095 args
.active_filter_l
= filter_l
;
1096 args
.nflp
= fl_entry
;
1097 g_list_foreach(get_filter_dialog_list(list_type
), new_filter_cb
, &args
);
1102 chg_list_item_cb(GtkTreeModel
*model
, GtkTreePath
*path _U_
, GtkTreeIter
*iter
,
1105 GList
*flp
= (GList
*)data
;
1106 filter_def
*filt
= (filter_def
*)flp
->data
;
1109 gtk_tree_model_get(model
, iter
, 1, &nl_model
, -1);
1110 /* Is this the item corresponding to the filter list item in question? */
1111 if (flp
== nl_model
) {
1112 /* Yes - change the label to correspond to the new name for the
1114 gtk_list_store_set(GTK_LIST_STORE(model
), iter
, 0, filt
->name
, -1);
1121 chg_filter_cb(gpointer data
, gpointer user_data
)
1123 GtkWidget
*main_w
= (GtkWidget
*)data
;
1124 GtkWidget
*filter_l
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
1126 gtk_tree_model_foreach(gtk_tree_view_get_model(GTK_TREE_VIEW(filter_l
)),
1127 chg_list_item_cb
, user_data
);
1131 filter_name_te_changed_cb(GtkWidget
*w
, gpointer data
)
1133 GtkWidget
*main_w
= gtk_widget_get_toplevel(w
);
1134 GtkWidget
*name_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_NAME_TE_KEY
);
1135 GtkWidget
*filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
);
1136 GtkWidget
*filter_l
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
1139 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
1140 const gchar
*name
= "";
1141 const gchar
*strval
= "";
1143 GtkTreeSelection
*sel
;
1144 GtkTreeModel
*model
;
1147 sel
= gtk_tree_view_get_selection(GTK_TREE_VIEW(filter_l
));
1148 name
= gtk_entry_get_text(GTK_ENTRY(name_te
));
1149 strval
= gtk_entry_get_text(GTK_ENTRY(filter_te
));
1151 if (DFILTER_EDITED_LIST
== list_type
) {
1152 /* colorize filter string entry */
1153 filter_te_syntax_check_cb(filter_te
, NULL
);
1156 /* if something was selected */
1157 if (gtk_tree_selection_get_selected(sel
, &model
, &iter
)) {
1158 gtk_tree_model_get(model
, &iter
, 1, &fl_entry
, -1);
1159 if (fl_entry
!= NULL
) {
1160 filt
= (filter_def
*) fl_entry
->data
;
1162 if (strlen(name
) > 0 && strlen(strval
) > 0 && filt
) {
1164 g_free(filt
->strval
);
1165 filt
->name
= g_strdup(name
);
1166 filt
->strval
= g_strdup(strval
);
1168 /* Update all the filter list widgets, not just the one in
1169 the dialog box in which we clicked on "Copy". */
1170 g_list_foreach(get_filter_dialog_list(list_type
), chg_filter_cb
,
1178 delete_filter_cb(gpointer data
, gpointer user_data
)
1180 GtkWidget
*main_w
= (GtkWidget
*)data
;
1181 GtkWidget
*filter_l
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
1182 gchar
*pos
= (gchar
*)user_data
;
1183 GtkTreeModel
*model
= gtk_tree_view_get_model(GTK_TREE_VIEW(filter_l
));
1186 gtk_tree_model_get_iter_from_string(model
, &iter
, pos
);
1187 gtk_list_store_remove(GTK_LIST_STORE(model
), &iter
);
1191 filter_del_bt_clicked_cb(GtkWidget
*w
, gpointer data
)
1193 GtkWidget
*main_w
= gtk_widget_get_toplevel(w
);
1194 GtkWidget
*filter_l
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_L_KEY
);
1195 filter_list_type_t list_type
= *(filter_list_type_t
*)data
;
1198 GtkTreeSelection
*sel
;
1199 GtkTreeModel
*model
;
1203 sel
= gtk_tree_view_get_selection(GTK_TREE_VIEW(filter_l
));
1204 /* If something was selected */
1205 if (gtk_tree_selection_get_selected(sel
, &model
, &iter
)) {
1206 gtk_tree_model_get(model
, &iter
, 1, &fl_entry
, -1);
1207 path
= gtk_tree_model_get_path(model
, &iter
);
1208 pos
= gtk_tree_path_to_string(path
);
1209 gtk_tree_path_free(path
);
1210 if (fl_entry
!= NULL
) {
1211 /* Remove the entry from the filter list. */
1212 remove_from_filter_list(list_type
, fl_entry
);
1214 /* Update all the filter list widgets, not just the one in
1215 the dialog box in which we clicked on "Delete". */
1216 g_list_foreach(get_filter_dialog_list(list_type
), delete_filter_cb
, pos
);
1223 filter_add_expr_bt_cb(GtkWidget
*w _U_
, gpointer main_w_arg
)
1225 GtkWidget
*main_w
= GTK_WIDGET(main_w_arg
);
1226 GtkWidget
*filter_te
, *dfilter_w
;
1228 filter_te
= (GtkWidget
*)g_object_get_data(G_OBJECT(main_w
), E_FILT_FILTER_TE_KEY
);
1229 dfilter_w
= dfilter_expr_dlg_new(filter_te
);
1231 /* If we're opening a series of modal dialogs (such as when going
1232 * through file->open, make the latest dialog modal also so that it
1233 * takes over "control" from the other modal dialogs. Also set
1234 * the transient property of the new dialog so the user doesn't try
1235 * to interact with the previous window when they can't.
1236 * XXX: containing widget might be the Filter Toolbar */
1238 if ( GTK_IS_WINDOW(main_w
) && gtk_window_get_modal(GTK_WINDOW(main_w
))) {
1239 gtk_window_set_modal(GTK_WINDOW(dfilter_w
), TRUE
);
1240 gtk_window_set_transient_for(GTK_WINDOW(dfilter_w
),
1241 GTK_WINDOW(main_w
));
1246 color_filter_te(GtkWidget
*w
, guint16 red
, guint16 green
, guint16 blue
)
1248 #if GTK_CHECK_VERSION(3,0,0)
1249 static GdkRGBA black
= { 0, 0, 0, 1.0 };
1252 bg
.red
= red
/ 65535.0;
1253 bg
.green
= green
/ 65535.0;
1254 bg
.blue
= blue
/ 65535.0;
1257 gtk_widget_override_color(w
, GTK_STATE_FLAG_NORMAL
, &black
);
1258 gtk_widget_override_background_color(w
, GTK_STATE_FLAG_NORMAL
, &bg
);
1259 gtk_widget_override_cursor(w
, &black
, &black
);
1261 static GdkColor black
= { 0, 0, 0, 0 };
1269 gtk_widget_modify_text(w
, GTK_STATE_NORMAL
, &black
);
1270 gtk_widget_modify_base(w
, GTK_STATE_NORMAL
, &bg
);
1271 gtk_widget_modify_cursor(w
, &black
, &black
);
1276 colorize_filter_te_as_empty(GtkWidget
*w
)
1278 #if GTK_CHECK_VERSION(3,0,0)
1280 gtk_widget_override_color(w
, GTK_STATE_FLAG_NORMAL
, NULL
);
1281 gtk_widget_override_background_color(w
, GTK_STATE_FLAG_NORMAL
, NULL
);
1282 gtk_widget_override_cursor(w
, NULL
, NULL
);
1285 gtk_widget_modify_text(w
, GTK_STATE_NORMAL
, NULL
);
1286 gtk_widget_modify_base(w
, GTK_STATE_NORMAL
, NULL
);
1287 gtk_widget_modify_cursor(w
, NULL
, NULL
);
1292 colorize_filter_te_as_invalid(GtkWidget
*w
)
1294 color_filter_te(w
, prefs
.gui_text_invalid
.red
, prefs
.gui_text_invalid
.green
, prefs
.gui_text_invalid
.blue
);
1298 colorize_filter_te_as_deprecated(GtkWidget
*w
)
1300 color_filter_te(w
, prefs
.gui_text_deprecated
.red
, prefs
.gui_text_deprecated
.green
, prefs
.gui_text_deprecated
.blue
);
1304 colorize_filter_te_as_valid(GtkWidget
*w
)
1306 color_filter_te(w
, prefs
.gui_text_valid
.red
, prefs
.gui_text_valid
.green
, prefs
.gui_text_valid
.blue
);
1310 * XXX This calls dfilter_compile, which might call get_host_ipaddr or
1311 * get_host_ipaddr6. Either of of these will freeze the UI if the host
1312 * name resolution takes a long time to complete. We need to work
1313 * around this, either by disabling host name resolution or by doing
1314 * the resolution asynchronously.
1316 * We could use a separate thread but we have be careful to only call
1317 * GTK+/GDK routines from the main thread. From the GDK threads
1320 * "With the Win32 backend, GDK calls should not be attempted from
1321 * multiple threads at all."
1325 filter_te_syntax_check_cb(GtkWidget
*w
, gpointer user_data _U_
)
1327 const gchar
*strval
;
1329 GPtrArray
*depr
= NULL
;
1330 gboolean use_statusbar
;
1333 strval
= gtk_entry_get_text(GTK_ENTRY(w
));
1334 use_statusbar
= g_object_get_data(G_OBJECT(w
), E_FILT_FIELD_USE_STATUSBAR_KEY
) ? TRUE
: FALSE
;
1336 if (use_statusbar
) {
1337 statusbar_pop_filter_msg();
1340 /* colorize filter string entry */
1341 if (g_object_get_data(G_OBJECT(w
), E_FILT_FIELD_NAME_ONLY_KEY
) &&
1342 strval
&& (c
= proto_check_field_name(strval
)) != 0)
1344 colorize_filter_te_as_invalid(w
);
1345 if (use_statusbar
) {
1346 statusbar_push_filter_msg(" Illegal character in field name: '%c'", c
);
1348 } else if (strval
&& dfilter_compile(strval
, &dfp
)) {
1350 depr
= dfilter_deprecated_tokens(dfp
);
1352 if (strlen(strval
) == 0) {
1353 colorize_filter_te_as_empty(w
);
1355 /* You keep using that word. I do not think it means what you think it means. */
1356 colorize_filter_te_as_deprecated(w
);
1357 if (use_statusbar
) {
1359 * We're being lazy and only printing the first "problem" token.
1360 * Would it be better to print all of them?
1362 statusbar_push_temporary_msg(" \"%s\" may have unexpected results (see the User's Guide)",
1363 (const char *) g_ptr_array_index(depr
, 0));
1366 colorize_filter_te_as_valid(w
);
1370 colorize_filter_te_as_invalid(w
);
1371 if (use_statusbar
) {
1372 if (dfilter_error_msg
) {
1373 statusbar_push_filter_msg(" Invalid filter: %s", dfilter_error_msg
);
1375 statusbar_push_filter_msg(" Invalid filter");
1387 * indent-tabs-mode: nil
1390 * ex: set shiftwidth=4 tabstop=8 expandtab:
1391 * :indentSize=4:tabSize=8:noTabs=true: