/* range_utils.c * Packet range routines (save, print, ...) for GTK things * * $Id$ * * Ulf Lamping * * Ethereal - Network traffic analyzer * By Gerald Combs * Copyright 1998 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 #include "globals.h" #include "packet-range.h" #include "gui_utils.h" #include "dlg_utils.h" #include "compat_macros.h" #include "simple_dialog.h" #include "range_utils.h" #define RANGE_VALUES_KEY "range_values" #define RANGE_CAPTURED_BT_KEY "range_captured_button" #define RANGE_DISPLAYED_BT_KEY "range_displayed_button" #define RANGE_SELECT_ALL_KEY "range_select_all_rb" #define RANGE_SELECT_ALL_C_KEY "range_select_all_c_lb" #define RANGE_SELECT_ALL_D_KEY "range_select_all_d_lb" #define RANGE_SELECT_CURR_KEY "range_select_curr_rb" #define RANGE_SELECT_CURR_C_KEY "range_select_curr_c_lb" #define RANGE_SELECT_CURR_D_KEY "range_select_curr_d_lb" #define RANGE_SELECT_MARKED_KEY "range_select_marked_only_rb" #define RANGE_SELECT_MARKED_C_KEY "range_select_marked_only_c_lb" #define RANGE_SELECT_MARKED_D_KEY "range_select_marked_only_d_lb" #define RANGE_SELECT_MARKED_RANGE_KEY "range_select_marked_range_rb" #define RANGE_SELECT_MARKED_RANGE_C_KEY "range_select_marked_range_c_lb" #define RANGE_SELECT_MARKED_RANGE_D_KEY "range_select_marked_range_d_lb" #define RANGE_SELECT_USER_KEY "range_select_user_range_rb" #define RANGE_SELECT_USER_C_KEY "range_select_user_range_c_lb" #define RANGE_SELECT_USER_D_KEY "range_select_user_range_d_lb" #define RANGE_SELECT_USER_ENTRY_KEY "range_select_user_range_entry" gboolean range_check_validity(packet_range_t *range) { switch (packet_range_check(range)) { case CVT_NO_ERROR: return TRUE; case CVT_SYNTAX_ERROR: simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "The specified range of packets isn't a valid range."); return FALSE; case CVT_NUMBER_TOO_BIG: simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "The specified range of packets has a packet number that's too large."); return FALSE; default: g_assert_not_reached(); return FALSE; } } /* update all "dynamic" things */ void range_update_dynamics(gpointer data) { packet_range_t *range; GtkWidget *range_displayed_bt; gboolean filtered_active; gint selected_num; gboolean can_select; gchar label_text[100]; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); range_displayed_bt = OBJECT_GET_DATA(data, RANGE_DISPLAYED_BT_KEY); filtered_active = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(range_displayed_bt)); /* Enable saving only the displayed packets only if there *are* displayed packets. */ if (range->displayed_cnt != 0) gtk_widget_set_sensitive(range_displayed_bt, TRUE); else { /* If saving the displayed packets is selected, select saving the captured packets. */ filtered_active = FALSE; gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_KEY)), FALSE); gtk_widget_set_sensitive(range_displayed_bt, FALSE); } gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_C_KEY), !filtered_active); g_snprintf(label_text, sizeof(label_text), "%u", cfile.count); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_C_KEY)), label_text); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_D_KEY), filtered_active); g_snprintf(label_text, sizeof(label_text), "%u", range->displayed_cnt); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_D_KEY)), label_text); /* Enable saving the currently-selected packet only if there *is* a currently-selected packet. */ selected_num = (cfile.current_frame) ? cfile.current_frame->num : 0; can_select = (selected_num != 0); if (can_select) { gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_KEY), TRUE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_C_KEY), !filtered_active); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_D_KEY), filtered_active); } else { /* If "save selected packet" is selected, select "save all packets". */ if (range->process == range_process_selected) { range->process = range_process_all; gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_KEY)), TRUE); } gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_C_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_D_KEY), FALSE); } /* XXX: how to update the radio button label but keep the mnemonic? */ /*g_snprintf(label_text, sizeof(label_text), "_Selected packet #%u only", selected_num); gtk_label_set_text(GTK_LABEL(GTK_BIN(select_curr_rb)->child), label_text);*/ g_snprintf(label_text, sizeof(label_text), "%u", selected_num ? 1 : 0); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_C_KEY)), label_text); g_snprintf(label_text, sizeof(label_text), "%u", selected_num ? 1 : 0); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_CURR_D_KEY)), label_text); /* Enable the buttons for saving marked packets only if there *are* marked packets. */ if (filtered_active) can_select = (range->displayed_marked_cnt != 0); else can_select = (cfile.marked_count > 0); if (can_select) { gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_KEY), TRUE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_C_KEY), !filtered_active); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_D_KEY), filtered_active); } else { /* If "save marked packet" is selected, select "save all packets". */ if (range->process == range_process_marked) { range->process = range_process_all; gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_KEY)), TRUE); } gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_C_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_D_KEY), FALSE); } g_snprintf(label_text, sizeof(label_text), "%u", cfile.marked_count); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_C_KEY)), label_text); g_snprintf(label_text, sizeof(label_text), "%u", range->displayed_marked_cnt); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_D_KEY)), label_text); /* Enable the buttons for saving the range of marked packets only if there *is* a range of marked packets. */ if (filtered_active) can_select = (range->displayed_mark_range_cnt != 0); else can_select = (range->mark_range_cnt != 0); if (can_select) { gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_KEY), TRUE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_C_KEY), !filtered_active); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_D_KEY), filtered_active); } else { /* If "save range between first and last marked packet" is selected, select "save all packets". */ if (range->process == range_process_marked_range) { range->process = range_process_all; gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(OBJECT_GET_DATA(data, RANGE_SELECT_ALL_KEY)), TRUE); } gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_C_KEY), FALSE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_D_KEY), FALSE); } gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_C_KEY)), label_text); g_snprintf(label_text, sizeof(label_text), "%u", range->displayed_mark_range_cnt); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_MARKED_RANGE_D_KEY)), label_text); g_snprintf(label_text, sizeof(label_text), "%u", range->mark_range_cnt); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_USER_KEY), TRUE); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_USER_C_KEY), !filtered_active); gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_USER_D_KEY), filtered_active); g_snprintf(label_text, sizeof(label_text), "%u", range->user_range_cnt); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_USER_C_KEY)), label_text); g_snprintf(label_text, sizeof(label_text), "%u", range->displayed_user_range_cnt); gtk_label_set_text(GTK_LABEL(OBJECT_GET_DATA(data, RANGE_SELECT_USER_D_KEY)), label_text); } static void toggle_captured_cb(GtkWidget *widget, gpointer data) { GtkWidget *bt; packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { /* They changed the state of the "captured" button. */ range->process_filtered = FALSE; bt = OBJECT_GET_DATA(data, RANGE_CAPTURED_BT_KEY); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(bt), TRUE); bt = OBJECT_GET_DATA(data, RANGE_DISPLAYED_BT_KEY); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(bt), FALSE); range_update_dynamics(data); } } static void toggle_filtered_cb(GtkWidget *widget, gpointer data) { GtkWidget *bt; packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process_filtered = TRUE; bt = OBJECT_GET_DATA(data, RANGE_CAPTURED_BT_KEY); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(bt), FALSE); bt = OBJECT_GET_DATA(data, RANGE_DISPLAYED_BT_KEY); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(bt), TRUE); range_update_dynamics(data); } } static void toggle_select_all(GtkWidget *widget, gpointer data) { packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process = range_process_all; range_update_dynamics(data); } } static void toggle_select_selected(GtkWidget *widget, gpointer data) { packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process = range_process_selected; range_update_dynamics(data); } } static void toggle_select_marked_only(GtkWidget *widget, gpointer data) { packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process = range_process_marked; range_update_dynamics(data); } } static void toggle_select_marked_range(GtkWidget *widget, gpointer data) { packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process = range_process_marked_range; range_update_dynamics(data); } } static void toggle_select_user_range(GtkWidget *widget, gpointer data) { packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); /* is the button now active? */ if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget))) { range->process = range_process_user_range; range_update_dynamics(data); } /* Make the entry widget sensitive or insensitive */ gtk_widget_set_sensitive(OBJECT_GET_DATA(data, RANGE_SELECT_USER_ENTRY_KEY), range->process == range_process_user_range); /* When selecting user specified range, then focus on the entry */ if (range->process == range_process_user_range) gtk_widget_grab_focus(OBJECT_GET_DATA(data, RANGE_SELECT_USER_ENTRY_KEY)); } static void range_entry(GtkWidget *widget _U_, gpointer data) { const gchar *entry_text; GtkWidget *entry; packet_range_t *range; range = OBJECT_GET_DATA(data, RANGE_VALUES_KEY); entry = OBJECT_GET_DATA(data, RANGE_SELECT_USER_ENTRY_KEY); gtk_toggle_button_set_active(OBJECT_GET_DATA(data, RANGE_SELECT_USER_KEY), TRUE); entry_text = gtk_entry_get_text (GTK_ENTRY (entry)); packet_range_convert_str(range, entry_text); range_update_dynamics(data); } static void range_entry_in_event(GtkWidget *widget, GdkEventFocus *event _U_, gpointer user_data) { /* This event is called, if the "enter" key is pressed while the key focus (right name?) */ /* is in the range entry field. */ /* Calling range_entry() isn't necessary as all changes are already done while the */ /* entry was edited. Calling it here will cause a NULL pointer exception, */ /* so don't do: process_filtered); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(displayed_bt), range->process_filtered); switch(range->process) { case(range_process_all): gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(select_all_rb), TRUE); break; case(range_process_selected): gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(select_curr_rb), TRUE); break; case(range_process_marked): gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(select_marked_only_rb), TRUE); break; case(range_process_marked_range): gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(select_marked_range_rb), TRUE); break; case(range_process_user_range): gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(select_user_range_rb), TRUE); break; default: g_assert_not_reached(); } return range_tb; }