1 /* main_filter_toolbar.c
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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26 * This file implements the "filter" toolbar for Wireshark.
37 #include "gtk/old-gtk-compat.h"
39 #include "filter_dlg.h"
40 #include "filter_autocomplete.h"
42 #include "epan/prefs.h"
45 #include "gtkglobals.h"
46 #include "stock_icons.h"
51 #include "main_toolbar.h"
52 #include "main_filter_toolbar.h"
53 #include "filter_expression_save_dlg.h"
55 #define MENU_BAR_PATH_FILE_OPEN "/Menubar/FileMenu/Open"
56 #define MENU_BAR_PATH_EDIT_COPY_AS_FLT "/Menubar/EditMenu/Copy/AsFilter"
57 #define MENU_BAR_PATH_ANALYZE_DISPLAY_FLT "/Menubar/AnalyzeMenu/DisplayFilters"
58 #define MENU_BAR_PATH_ANALYZE_FOLLOW_TCP_STREAM "/Menubar/AnalyzeMenu/FollowTCPStream"
59 #define MENU_BAR_PATH_ANALYZE_FOLLOW_UDP_STREAM "/Menubar/AnalyzeMenu/FollowUDPStream"
60 #define MENU_BAR_PATH_ANALYZE_FOLLOW_SSL_STREAM "/Menubar/AnalyzeMenu/FollowSSLStream"
61 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/Selected"
62 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_NOT_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/NotSelected"
63 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_AND_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/AndSelected"
64 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_OR_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/OrSelected"
65 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_AND_NOT_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/AndNotSelected"
66 #define MENU_BAR_PATH_ANALYZE_APL_AS_FLT_OR_NOT_SEL "/Menubar/AnalyzeMenu/ApplyAsFilter/OrNotSelected"
67 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/Selected"
68 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_NOT_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/NotSelected"
69 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_AND_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/AndSelected"
70 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_OR_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/OrSelected"
71 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_AND_NOT_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/AndNotSelected"
72 #define MENU_BAR_PATH_ANALYZE_PREP_A_FLT_OR_NOT_SEL "/Menubar/AnalyzeMenu/PrepareaFilter/OrNotSelected"
74 GtkWidget *main_display_filter_widget=NULL;
76 /* Run the current display filter on the current packet set, and
79 filter_activate_cb(GtkWidget *w _U_, gpointer data)
83 s = gtk_entry_get_text(GTK_ENTRY(data));
84 main_filter_packets(&cfile, s, FALSE);
87 /* Enable both Clear and Apply button when filter is changed */
89 filter_changed_cb(GtkWidget *w _U_, gpointer data)
91 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(data), E_DFILTER_APPLY_KEY), TRUE);
92 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(data), E_DFILTER_CLEAR_KEY), TRUE);
93 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(data), E_DFILTER_SAVE_KEY), TRUE);
96 /* redisplay with no display filter */
98 filter_reset_cb(GtkWidget *w, gpointer data _U_)
100 GtkWidget *filter_te = NULL;
102 if ((filter_te = g_object_get_data(G_OBJECT(w), E_DFILTER_TE_KEY))) {
103 gtk_entry_set_text(GTK_ENTRY(filter_te), "");
105 main_filter_packets(&cfile, NULL, FALSE);
109 filter_save_cb(GtkWidget *w _U_, GtkWindow *parent_w)
111 filter_expression_save_dlg(parent_w);
115 GtkWidget *filter_toolbar_new(void)
117 GtkWidget *filter_cm;
118 GtkWidget *filter_te;
119 GtkWidget *filter_tb;
120 GtkToolItem *filter_bt, *filter_add_expr_bt, *filter_reset;
121 GtkToolItem *filter_apply, *filter_save, *item;
124 /* Display filter construct dialog has an Apply button, and "OK" not
125 only sets our text widget, it activates it (i.e., it causes us to
126 filter the capture). */
127 static construct_args_t args = {
128 "Wireshark: Display Filter",
135 filter_tb = gtk_toolbar_new();
136 gtk_orientable_set_orientation(GTK_ORIENTABLE(filter_tb),
137 GTK_ORIENTATION_HORIZONTAL);
139 g_object_set_data(G_OBJECT(top_level), E_TB_FILTER_KEY, filter_tb);
140 gtk_widget_show(filter_tb);
142 /* Create the "Filter:" button */
143 filter_bt = gtk_tool_button_new_from_stock (WIRESHARK_STOCK_DISPLAY_FILTER_ENTRY);
144 g_signal_connect(filter_bt, "clicked", G_CALLBACK(display_filter_construct_cb), &args);
145 gtk_widget_show(GTK_WIDGET (filter_bt));
146 g_object_set_data(G_OBJECT(top_level), E_FILT_BT_PTR_KEY, filter_bt);
148 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
151 gtk_widget_set_tooltip_text( GTK_WIDGET(filter_bt), "Open the \"Display Filter\" dialog, to edit/apply filters");
153 /* Create the filter combobox */
154 filter_cm = gtk_combo_box_text_new_with_entry ();
155 filter_te = gtk_bin_get_child(GTK_BIN(filter_cm));
156 main_display_filter_widget=filter_te;
157 g_object_set_data(G_OBJECT(filter_bt), E_FILT_TE_PTR_KEY, filter_te);
158 g_object_set_data(G_OBJECT(filter_te), E_DFILTER_CM_KEY, filter_cm);
159 g_object_set_data(G_OBJECT(top_level), E_DFILTER_CM_KEY, filter_cm);
160 g_signal_connect(filter_te, "activate", G_CALLBACK(filter_activate_cb), filter_te);
161 g_signal_connect(filter_te, "changed", G_CALLBACK(filter_changed_cb), filter_cm);
162 g_signal_connect(filter_te, "changed", G_CALLBACK(filter_te_syntax_check_cb), NULL);
163 g_object_set_data(G_OBJECT(filter_tb), E_FILT_AUTOCOMP_PTR_KEY, NULL);
164 g_object_set_data(G_OBJECT(filter_te), E_FILT_FIELD_USE_STATUSBAR_KEY, "");
165 g_signal_connect(filter_te, "key-press-event", G_CALLBACK (filter_string_te_key_pressed_cb), NULL);
166 g_signal_connect(filter_tb, "key-press-event", G_CALLBACK (filter_parent_dlg_key_pressed_cb), NULL);
168 gtk_widget_set_size_request(filter_cm, 400, -1);
169 gtk_widget_show(filter_cm);
170 item = gtk_tool_item_new ();
171 gtk_container_add (GTK_CONTAINER (item), filter_cm);
172 gtk_widget_show (GTK_WIDGET (item));
174 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
178 /* setting a tooltip for a combobox will do nothing, so add it to the corresponding text entry */
179 gtk_widget_set_tooltip_text(filter_cm,
180 "Enter a display filter, or choose one of your recently used filters. "
181 "The background color of this field is changed by a continuous syntax check "
182 "(green is valid, red is invalid, yellow may have unexpected results).");
184 /* Create the "Add Expression..." button, to pop up a dialog
185 for constructing filter comparison expressions. */
186 filter_add_expr_bt = gtk_tool_button_new_from_stock(WIRESHARK_STOCK_ADD_EXPRESSION);
187 g_object_set_data(G_OBJECT(filter_tb), E_FILT_FILTER_TE_KEY, filter_te);
188 g_signal_connect(filter_add_expr_bt, "clicked", G_CALLBACK(filter_add_expr_bt_cb), filter_tb);
189 gtk_widget_show(GTK_WIDGET(filter_add_expr_bt));
191 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
195 gtk_widget_set_tooltip_text(GTK_WIDGET(filter_add_expr_bt), "Add an expression to this filter string");
197 /* Create the "Clear" button */
198 filter_reset = gtk_tool_button_new_from_stock(WIRESHARK_STOCK_CLEAR_EXPRESSION);
199 g_object_set_data(G_OBJECT(filter_reset), E_DFILTER_TE_KEY, filter_te);
200 g_object_set_data (G_OBJECT(filter_cm), E_DFILTER_CLEAR_KEY, filter_reset);
201 g_signal_connect(filter_reset, "clicked", G_CALLBACK(filter_reset_cb), NULL);
202 gtk_widget_set_sensitive (GTK_WIDGET(filter_reset), FALSE);
203 gtk_widget_show(GTK_WIDGET(filter_reset));
204 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
208 gtk_widget_set_tooltip_text(GTK_WIDGET(filter_reset), "Clear this filter string and update the display");
210 /* Create the "Apply" button */
211 filter_apply = gtk_tool_button_new_from_stock(WIRESHARK_STOCK_APPLY_EXPRESSION);
212 g_object_set_data(G_OBJECT(filter_apply), E_DFILTER_CM_KEY, filter_cm);
213 g_object_set_data (G_OBJECT(filter_cm), E_DFILTER_APPLY_KEY, filter_apply);
214 g_signal_connect(filter_apply, "clicked", G_CALLBACK(filter_activate_cb), filter_te);
215 gtk_widget_set_sensitive (GTK_WIDGET(filter_apply), FALSE);
216 gtk_widget_show(GTK_WIDGET(filter_apply));
218 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
222 gtk_widget_set_tooltip_text(GTK_WIDGET(filter_apply), "Apply this filter string to the display");
224 /* Create the "Save" button */
225 filter_save = gtk_tool_button_new_from_stock(GTK_STOCK_SAVE);
226 g_object_set_data(G_OBJECT(filter_save), E_DFILTER_CM_KEY, filter_cm);
227 g_object_set_data(G_OBJECT(filter_cm), E_DFILTER_SAVE_KEY, filter_save);
228 g_signal_connect(filter_save, "clicked", G_CALLBACK(filter_save_cb), filter_te);
229 gtk_widget_set_sensitive (GTK_WIDGET(filter_save), FALSE);
230 gtk_widget_show(GTK_WIDGET(filter_save));
232 gtk_toolbar_insert(GTK_TOOLBAR(filter_tb),
236 gtk_widget_set_tooltip_text(GTK_WIDGET(filter_save), "Save this filter string");
238 /* Sets the text entry widget pointer as the E_DILTER_TE_KEY data
239 * of any widget that ends up calling a callback which needs
240 * that text entry pointer */
241 set_menu_object_data(MENU_BAR_PATH_FILE_OPEN, E_DFILTER_TE_KEY, filter_te);
242 set_menu_object_data(MENU_BAR_PATH_EDIT_COPY_AS_FLT, E_DFILTER_TE_KEY,
244 set_menu_object_data(MENU_BAR_PATH_ANALYZE_DISPLAY_FLT, E_FILT_TE_PTR_KEY,
246 set_menu_object_data(MENU_BAR_PATH_ANALYZE_FOLLOW_TCP_STREAM, E_DFILTER_TE_KEY,
248 set_menu_object_data(MENU_BAR_PATH_ANALYZE_FOLLOW_UDP_STREAM, E_DFILTER_TE_KEY,
250 set_menu_object_data(MENU_BAR_PATH_ANALYZE_FOLLOW_SSL_STREAM, E_DFILTER_TE_KEY,
252 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_SEL, E_DFILTER_TE_KEY,
254 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_NOT_SEL, E_DFILTER_TE_KEY,
256 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_AND_SEL, E_DFILTER_TE_KEY,
258 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_OR_SEL, E_DFILTER_TE_KEY,
260 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_AND_NOT_SEL, E_DFILTER_TE_KEY,
262 set_menu_object_data(MENU_BAR_PATH_ANALYZE_APL_AS_FLT_OR_NOT_SEL, E_DFILTER_TE_KEY,
264 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_SEL, E_DFILTER_TE_KEY,
266 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_NOT_SEL, E_DFILTER_TE_KEY,
268 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_AND_SEL, E_DFILTER_TE_KEY,
270 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_OR_SEL, E_DFILTER_TE_KEY,
272 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_AND_NOT_SEL, E_DFILTER_TE_KEY,
274 set_menu_object_data(MENU_BAR_PATH_ANALYZE_PREP_A_FLT_OR_NOT_SEL, E_DFILTER_TE_KEY,
277 set_toolbar_object_data(E_DFILTER_TE_KEY, filter_te);
278 g_object_set_data(G_OBJECT(popup_menu_object), E_DFILTER_TE_KEY, filter_te);
280 filter_expression_save_dlg_init(filter_tb, filter_te);
282 /* make current preferences effective */
283 toolbar_redraw_all();
289 dfilter_entry_match(GtkWidget *filter_cm, char *s, int *index)
291 GtkTreeModel *model = gtk_combo_box_get_model (GTK_COMBO_BOX(filter_cm));
293 GValue value = { 0, {{0}}};
294 const char *filter_str;
298 if (!gtk_tree_model_get_iter_first (model, &iter)){
304 gtk_tree_model_get_value (model, &iter, 0, &value);
305 filter_str = g_value_get_string (&value);
307 if(strcmp(s, filter_str) == 0){
308 g_value_unset (&value);
313 g_value_unset (&value);
314 }while (gtk_tree_model_iter_next (model, &iter));
320 /* add a display filter to the combo box */
321 /* Note: a new filter string will not replace an old identical one */
323 dfilter_combo_add(GtkWidget *filter_cm, char *s) {
326 if(!dfilter_entry_match(filter_cm,s, &index))
327 gtk_combo_box_text_append_text (GTK_COMBO_BOX_TEXT(filter_cm), s);
334 /* write all non empty display filters (until maximum count)
335 * of the combo box GList to the user's recent file */
337 dfilter_recent_combo_write_all(FILE *rf) {
338 GtkWidget *filter_cm = g_object_get_data(G_OBJECT(top_level), E_DFILTER_CM_KEY);
339 GtkTreeModel *model = gtk_combo_box_get_model (GTK_COMBO_BOX(filter_cm));
341 GValue value = { 0, {{0}}};
342 const char *filter_str;
345 if (!gtk_tree_model_get_iter_first (model, &iter))
348 gtk_tree_model_get_value (model, &iter, 0, &value);
349 filter_str = g_value_get_string (&value);
351 fprintf (rf, RECENT_KEY_DISPLAY_FILTER ": %s\n", filter_str);
352 g_value_unset (&value);
354 }while (gtk_tree_model_iter_next (model, &iter)&& (max_count++ < prefs.gui_recent_df_entries_max));
358 /* add a display filter coming from the user's recent file to the dfilter combo box */
360 dfilter_combo_add_recent(gchar *s) {
361 GtkWidget *filter_cm = g_object_get_data(G_OBJECT(top_level), E_DFILTER_CM_KEY);
366 return dfilter_combo_add(filter_cm, dup);
369 /* call cf_filter_packets() and add this filter string to the recent filter list */
371 main_filter_packets(capture_file *cf, const gchar *dftext, gboolean force)
373 GtkWidget *filter_cm = g_object_get_data(G_OBJECT(top_level), E_DFILTER_CM_KEY);
374 gboolean free_filter = TRUE;
376 cf_status_t cf_status;
378 s = g_strdup(dftext);
380 cf_status = cf_filter_packets(cf, s, force);
382 if (cf_status == CF_OK) {
383 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(filter_cm), E_DFILTER_APPLY_KEY), FALSE);
384 if (!s || strlen (s) == 0) {
385 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(filter_cm), E_DFILTER_CLEAR_KEY), FALSE);
386 gtk_widget_set_sensitive (g_object_get_data (G_OBJECT(filter_cm), E_DFILTER_SAVE_KEY), FALSE);
391 return (cf_status == CF_OK);
393 /* GtkCombos don't let us get at their list contents easily, so we maintain
394 our own filter list, and feed it to gtk_combo_set_popdown_strings when
395 a new filter is added. */
396 if (cf_status == CF_OK && strlen(s) > 0) {
399 if(!dfilter_entry_match(filter_cm,s, &index)){
400 gtk_combo_box_text_prepend_text(GTK_COMBO_BOX_TEXT(filter_cm), s);
403 while ((guint)index >= prefs.gui_recent_df_entries_max){
404 gtk_combo_box_text_remove(GTK_COMBO_BOX_TEXT(filter_cm), index);
411 return (cf_status == CF_OK);