aboutsummaryrefslogtreecommitdiffstats
path: root/gtk2/proto_dlg.c
diff options
context:
space:
mode:
Diffstat (limited to 'gtk2/proto_dlg.c')
-rw-r--r--gtk2/proto_dlg.c412
1 files changed, 412 insertions, 0 deletions
diff --git a/gtk2/proto_dlg.c b/gtk2/proto_dlg.c
new file mode 100644
index 0000000000..957f407d8f
--- /dev/null
+++ b/gtk2/proto_dlg.c
@@ -0,0 +1,412 @@
+/* proto_dlg.c
+ *
+ * $Id: proto_dlg.c,v 1.1 2002/08/31 09:55:22 oabad Exp $
+ *
+ * Laurent Deniel <deniel@worldnet.fr>
+ *
+ * Ethereal - Network traffic analyzer
+ * By Gerald Combs <gerald@ethereal.com>
+ * Copyright 2000 Gerald Combs
+ *
+ * This program is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU General Public License
+ * as published by the Free Software Foundation; either version 2
+ * of the License, or (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
+ */
+
+#ifdef HAVE_CONFIG_H
+# include "config.h"
+#endif
+
+#include <gtk/gtk.h>
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+
+#include "prefs.h"
+#include "globals.h"
+#include "main.h"
+#include "util.h"
+#include "ui_util.h"
+#include "dlg_utils.h"
+#include "proto_dlg.h"
+
+static gboolean proto_delete_cb(GtkWidget *, gpointer);
+static void proto_ok_cb(GtkWidget *, gpointer);
+static void proto_apply_cb(GtkWidget *, gpointer);
+static void proto_cancel_cb(GtkWidget *, gpointer);
+static void proto_destroy_cb(GtkWidget *, gpointer);
+
+static void show_proto_selection(GtkWidget *main, GtkWidget *container);
+static gboolean set_proto_selection(GtkWidget *);
+static gboolean revert_proto_selection(void);
+
+static void toggle_all_cb(GtkWidget *button, gpointer parent_w);
+static void enable_all_cb(GtkWidget *button, gpointer parent_w);
+static void disable_all_cb(GtkWidget *button, gpointer parent_w);
+
+static GtkWidget *proto_w = NULL;
+
+/* list of protocols */
+static GSList *protocol_list = NULL;
+
+typedef struct protocol_data {
+ char *name;
+ char *abbrev;
+ int hfinfo_index;
+ gboolean was_enabled;
+} protocol_data_t;
+
+void proto_cb(GtkWidget *w _U_, gpointer data _U_)
+{
+
+ GtkWidget *main_vb, *bbox, *proto_nb, *apply_bt, *cancel_bt, *ok_bt,
+ *label, *scrolled_w, *selection_vb, *button;
+
+ if (proto_w != NULL) {
+ reactivate_window(proto_w);
+ return;
+ }
+
+ proto_w = dlg_window_new("Ethereal: Protocol");
+ g_signal_connect(G_OBJECT(proto_w), "delete_event",
+ G_CALLBACK(proto_delete_cb), NULL);
+ g_signal_connect(G_OBJECT(proto_w), "destroy",
+ G_CALLBACK(proto_destroy_cb), NULL);
+ gtk_widget_set_size_request(GTK_WIDGET(proto_w), DEF_WIDTH * 2/3,
+ DEF_HEIGHT * 2/3);
+
+ /* Container for each row of widgets */
+
+ main_vb = gtk_vbox_new(FALSE, 0);
+ gtk_container_border_width(GTK_CONTAINER(main_vb), 1);
+ gtk_container_add(GTK_CONTAINER(proto_w), main_vb);
+ gtk_widget_show(main_vb);
+
+ /* Protocol topics container */
+
+ proto_nb = gtk_notebook_new();
+ gtk_container_add(GTK_CONTAINER(main_vb), proto_nb);
+ /* XXX do not know why I need this to fill all space around buttons */
+ gtk_widget_set_size_request(GTK_WIDGET(proto_nb), DEF_WIDTH * 2/3 - 50,
+ DEF_HEIGHT * 2/3 - 50);
+
+ /* Protocol selection panel ("enable/disable" protocols) */
+
+ selection_vb = gtk_vbox_new(FALSE, 0);
+ gtk_container_border_width(GTK_CONTAINER(selection_vb), 1);
+ label = gtk_label_new("Button pressed: protocol decoding is enabled");
+ gtk_widget_show(label);
+ gtk_box_pack_start(GTK_BOX(selection_vb), label, FALSE, FALSE, 0);
+ scrolled_w = scrolled_window_new(NULL, NULL);
+ gtk_container_set_border_width(GTK_CONTAINER(scrolled_w), 1);
+ gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(scrolled_w),
+ GTK_POLICY_AUTOMATIC,
+ GTK_POLICY_ALWAYS);
+ gtk_box_pack_start(GTK_BOX(selection_vb), scrolled_w, TRUE, TRUE, 0);
+ show_proto_selection(proto_w, scrolled_w);
+ gtk_widget_show(scrolled_w);
+ gtk_widget_show(selection_vb);
+ label = gtk_label_new("Decoding");
+ gtk_notebook_append_page(GTK_NOTEBOOK(proto_nb), selection_vb, label);
+ label = gtk_label_new("Note that when a protocol is disabled, "
+ "all linked sub-protocols are as well");
+ gtk_widget_show(label);
+ gtk_box_pack_start(GTK_BOX(selection_vb), label, FALSE, FALSE, 0);
+
+
+ bbox = gtk_hbutton_box_new();
+ gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
+ gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5);
+ gtk_box_pack_start(GTK_BOX(selection_vb), bbox, FALSE, FALSE, 0);
+ gtk_widget_show(bbox);
+
+ /* Toggle All */
+ button = gtk_button_new_with_label("Toggle All");
+ g_signal_connect(G_OBJECT(button), "clicked",
+ G_CALLBACK(toggle_all_cb), GTK_OBJECT(proto_w));
+ gtk_box_pack_start(GTK_BOX(bbox), button, TRUE, TRUE, 0);
+ gtk_widget_show(button);
+
+ /* Enable All */
+ button = gtk_button_new_with_label("Enable All");
+ g_signal_connect(G_OBJECT(button), "clicked",
+ G_CALLBACK(enable_all_cb), GTK_OBJECT(proto_w));
+ gtk_box_pack_start(GTK_BOX(bbox), button, TRUE, TRUE, 0);
+ gtk_widget_show(button);
+
+ /* Disable All */
+ button = gtk_button_new_with_label("Disable All");
+ g_signal_connect(G_OBJECT(button), "clicked",
+ G_CALLBACK(disable_all_cb), GTK_OBJECT(proto_w));
+ gtk_box_pack_start(GTK_BOX(bbox), button, TRUE, TRUE, 0);
+ gtk_widget_show(button);
+
+
+ /* XXX add other protocol-related panels here ... */
+
+ gtk_widget_show(proto_nb);
+
+ /* Ok, Apply, Cancel Buttons */
+
+ bbox = gtk_hbutton_box_new();
+ gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
+ gtk_button_box_set_spacing(GTK_BUTTON_BOX(bbox), 5);
+ gtk_container_add(GTK_CONTAINER(main_vb), bbox);
+ gtk_widget_show(bbox);
+
+ ok_bt = gtk_button_new_from_stock(GTK_STOCK_OK);
+ g_signal_connect(G_OBJECT(ok_bt), "clicked",
+ G_CALLBACK(proto_ok_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(ok_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), ok_bt, TRUE, TRUE, 0);
+ gtk_widget_grab_default(ok_bt);
+ gtk_widget_show(ok_bt);
+
+ apply_bt = gtk_button_new_from_stock(GTK_STOCK_APPLY);
+ g_signal_connect(G_OBJECT(apply_bt), "clicked",
+ G_CALLBACK(proto_apply_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(apply_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), apply_bt, TRUE, TRUE, 0);
+ gtk_widget_show(apply_bt);
+
+ cancel_bt = gtk_button_new_from_stock(GTK_STOCK_CANCEL);
+ g_signal_connect(G_OBJECT(cancel_bt), "clicked",
+ G_CALLBACK(proto_cancel_cb), GTK_OBJECT(proto_w));
+ GTK_WIDGET_SET_FLAGS(cancel_bt, GTK_CAN_DEFAULT);
+ gtk_box_pack_start(GTK_BOX (bbox), cancel_bt, TRUE, TRUE, 0);
+ gtk_widget_show(cancel_bt);
+
+ dlg_set_cancel(proto_w, cancel_bt);
+
+ gtk_quit_add_destroy(gtk_main_level(), GTK_OBJECT(proto_w));
+ gtk_widget_show(proto_w);
+
+} /* proto_cb */
+
+
+/* Toggle All */
+static void
+toggle_all_cb(GtkWidget *button _U_, gpointer parent_w)
+{
+
+ GSList *entry;
+
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ GtkWidget *button;
+ protocol_data_t *p = entry->data;
+
+ button = (GtkWidget *) gtk_object_get_data(GTK_OBJECT(parent_w),
+ p->abbrev);
+ /* gtk_toggle_button_toggled() didn't work for me... */
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button),
+ !gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(button)));
+ }
+}
+
+/* Enable/Disable All Helper */
+static void
+set_active_all(gpointer parent_w, gboolean new_state)
+{
+
+ GSList *entry;
+
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ GtkWidget *button;
+ protocol_data_t *p = entry->data;
+
+ button = (GtkWidget *) gtk_object_get_data(GTK_OBJECT(parent_w),
+ p->abbrev);
+ gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(button), new_state);
+ }
+}
+
+/* Enable All */
+static void
+enable_all_cb(GtkWidget *button _U_, gpointer parent_w)
+{
+ set_active_all(parent_w, TRUE);
+}
+
+/* Disable All */
+static void
+disable_all_cb(GtkWidget *button _U_, gpointer parent_w)
+{
+ set_active_all(parent_w, FALSE);
+}
+
+static void proto_destroy_cb(GtkWidget *w _U_, gpointer data _U_)
+{
+ GSList *entry;
+
+ if (proto_w)
+ gtk_widget_destroy(proto_w);
+ proto_w = NULL;
+
+ /* remove protocol list */
+ if (protocol_list) {
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ g_free(entry->data);
+ }
+ g_slist_free(protocol_list);
+ protocol_list = NULL;
+ }
+}
+
+/* Treat this as a cancel, by calling "proto_cancel_cb()".
+ XXX - that'll destroy the Protocols dialog; will that upset
+ a higher-level handler that says "OK, we've been asked to delete
+ this, so destroy it"? */
+static gboolean proto_delete_cb(GtkWidget *proto_w, gpointer dummy _U_)
+{
+ proto_cancel_cb(NULL, proto_w);
+ return FALSE;
+}
+
+static void proto_ok_cb(GtkWidget *ok_bt _U_, gpointer parent_w)
+{
+ gboolean redissect;
+
+ redissect = set_proto_selection(GTK_WIDGET(parent_w));
+ gtk_widget_destroy(GTK_WIDGET(parent_w));
+ if (redissect)
+ redissect_packets(&cfile);
+}
+
+static void proto_apply_cb(GtkWidget *apply_bt _U_, gpointer parent_w)
+{
+ if (set_proto_selection(GTK_WIDGET(parent_w)))
+ redissect_packets(&cfile);
+}
+
+static void proto_cancel_cb(GtkWidget *cancel_bt _U_, gpointer parent_w)
+{
+ gboolean redissect;
+
+ redissect = revert_proto_selection();
+ gtk_widget_destroy(GTK_WIDGET(parent_w));
+ if (redissect)
+ redissect_packets(&cfile);
+}
+
+static gboolean set_proto_selection(GtkWidget *parent_w)
+{
+ GSList *entry;
+ gboolean need_redissect = FALSE;
+
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ GtkWidget *button;
+ protocol_data_t *p = entry->data;
+
+ button = (GtkWidget *) gtk_object_get_data(GTK_OBJECT(parent_w),
+ p->abbrev);
+ if (proto_is_protocol_enabled(p->hfinfo_index) != GTK_TOGGLE_BUTTON (button)->active) {
+ proto_set_decoding(p->hfinfo_index, GTK_TOGGLE_BUTTON (button)->active);
+ need_redissect = TRUE;
+ }
+ }
+
+ return need_redissect;
+
+} /* set_proto_selection */
+
+static gboolean revert_proto_selection(void)
+{
+ GSList *entry;
+ gboolean need_redissect = FALSE;
+
+ /*
+ * Undo all the changes we've made to protocol enable flags.
+ */
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ protocol_data_t *p = entry->data;
+
+ if (proto_is_protocol_enabled(p->hfinfo_index) != p->was_enabled) {
+ proto_set_decoding(p->hfinfo_index, p->was_enabled);
+ need_redissect = TRUE;
+ }
+ }
+
+ return need_redissect;
+
+} /* revert_proto_selection */
+
+gint protocol_data_compare(gconstpointer a, gconstpointer b)
+{
+ return strcmp(((protocol_data_t *)a)->abbrev,
+ ((protocol_data_t *)b)->abbrev);
+}
+
+static void show_proto_selection(GtkWidget *main, GtkWidget *container)
+{
+
+#define NB_COL 7
+
+ GSList *entry;
+ GtkTooltips *tooltips;
+ GtkWidget *table;
+ int i, t = 0, l = 0, nb_line, nb_proto = 0;
+ void *cookie;
+ protocol_data_t *p;
+
+ /* Iterate over all the protocols */
+
+ for (i = proto_get_first_protocol(&cookie); i != -1;
+ i = proto_get_next_protocol(&cookie)) {
+ if (proto_can_disable_protocol(i)) {
+ p = g_malloc(sizeof(protocol_data_t));
+ p->name = proto_get_protocol_name(i);
+ p->abbrev = proto_get_protocol_filter_name(i);
+ p->hfinfo_index = i;
+ p->was_enabled = proto_is_protocol_enabled(i);
+ protocol_list = g_slist_insert_sorted(protocol_list,
+ p, protocol_data_compare);
+ nb_proto ++;
+ }
+ }
+
+ /* create a table (n x NB_COL) of buttons */
+
+ nb_line = (nb_proto % NB_COL) ? nb_proto / NB_COL + 1 : nb_proto / NB_COL;
+ table = gtk_table_new (nb_line, NB_COL, FALSE);
+ gtk_table_set_row_spacings(GTK_TABLE (table), 1);
+ gtk_table_set_col_spacings(GTK_TABLE (table), 1);
+ gtk_scrolled_window_add_with_viewport(GTK_SCROLLED_WINDOW(container), table);
+ gtk_widget_show(table);
+
+ tooltips = gtk_tooltips_new();
+
+ nb_proto = 0;
+
+ for (entry = protocol_list; entry != NULL; entry = g_slist_next(entry)) {
+ GtkWidget *button;
+
+ p = entry->data;
+ /* button label is the protocol abbrev */
+ button = gtk_toggle_button_new_with_label(p->abbrev);
+ /* tip is the complete protocol name */
+ gtk_tooltips_set_tip(tooltips, button, p->name, NULL);
+ gtk_toggle_button_set_state(GTK_TOGGLE_BUTTON(button),
+ proto_is_protocol_enabled(p->hfinfo_index));
+ gtk_object_set_data(GTK_OBJECT(main), p->abbrev, button);
+ gtk_table_attach_defaults (GTK_TABLE (table), button, l, l+1, t, t+1);
+ gtk_widget_show (button);
+ if (++nb_proto % NB_COL) {
+ l++;
+ }
+ else {
+ l = 0;
+ t++;
+ }
+ }
+
+} /* show_proto_selection */