diff options
author | Jörg Mayer <jmayer@loplof.de> | 2012-01-15 21:59:11 +0000 |
---|---|---|
committer | Jörg Mayer <jmayer@loplof.de> | 2012-01-15 21:59:11 +0000 |
commit | be706c63801fb98d42fb743b27b16cc36273651e (patch) | |
tree | 62ed0b552191eb0753d26a3edcbab73459a15f7f /ui/gtk/print_dlg.c | |
parent | 6d69ef093cd6868ab51f8b52477a510172033353 (diff) | |
download | wireshark-be706c63801fb98d42fb743b27b16cc36273651e.tar.gz |
Move gtk to ui/gtk.
This looses the last checkin to gtk, will add this manually back.
svn path=/trunk/; revision=40518
Diffstat (limited to 'ui/gtk/print_dlg.c')
-rw-r--r-- | ui/gtk/print_dlg.c | 1139 |
1 files changed, 1139 insertions, 0 deletions
diff --git a/ui/gtk/print_dlg.c b/ui/gtk/print_dlg.c new file mode 100644 index 0000000000..63f7c30a39 --- /dev/null +++ b/ui/gtk/print_dlg.c @@ -0,0 +1,1139 @@ +/* print_dlg.c + * Dialog boxes for printing and exporting to text files + * + * $Id$ + * + * Wireshark - Network traffic analyzer + * By Gerald Combs <gerald@wireshark.org> + * 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 <string.h> + +#include <gtk/gtk.h> + +#include <epan/prefs.h> +#include <epan/epan_dissect.h> +#include <epan/filesystem.h> + +#include "../print.h" +#include "../alert_box.h" +#include "../simple_dialog.h" +#include "../util.h" +#include <wsutil/file_util.h> + +#include "ui/gtk/gtkglobals.h" +#include "ui/gtk/keys.h" +#include "ui/gtk/capture_file_dlg.h" +#include "ui/gtk/gui_utils.h" +#include "ui/gtk/dlg_utils.h" +#include "ui/gtk/file_dlg.h" +#include "ui/gtk/main.h" +#include "ui/gtk/stock_icons.h" +#include "ui/gtk/range_utils.h" +#include "ui/gtk/help_dlg.h" + +#ifdef _WIN32 +#include <gdk/gdkwin32.h> +#include <windows.h> +#include "win32/file_dlg_win32.h" +#include "win32/print_win32.h" +#include "../tempfile.h" +#endif + +/* dialog output action */ +typedef enum { + output_action_print, /* print text to printer */ + output_action_export_text, /* export to plain text */ + output_action_export_ps, /* export to postscript */ + output_action_export_psml, /* export to packet summary markup language */ + output_action_export_pdml, /* export to packet data markup language */ + output_action_export_csv, /* export to csv file */ + output_action_export_carrays /* export to C array file */ +} output_action_e; + + +/* On Win32, a GUI application apparently can't use "popen()" (it + "returns an invalid file handle, if used in a Windows program, + that will cause the program to hang indefinitely"), so we can't + use a pipe to a print command to print to a printer. + + Eventually, we should try to use the native Win32 printing API + for this (and also use various UNIX printing APIs, when present?). +*/ + +static GtkWidget * +open_print_dialog(const char *title, output_action_e action, print_args_t *args); +static void print_cmd_toggle_dest(GtkWidget *widget, gpointer data); +static void print_cmd_toggle_detail(GtkWidget *widget, gpointer data); +static void print_ok_cb(GtkWidget *ok_bt, gpointer parent_w); +static void print_destroy_cb(GtkWidget *win, gpointer user_data); + + + +#define PRINT_ARGS_KEY "printer_args" + +#define PRINT_PS_RB_KEY "printer_ps_radio_button" +#define PRINT_PDML_RB_KEY "printer_pdml_radio_button" +#define PRINT_PSML_RB_KEY "printer_psml_radio_button" +#define PRINT_CSV_RB_KEY "printer_csv_radio_button" +#define PRINT_CARRAYS_RB_KEY "printer_carrays_radio_button" +#define PRINT_DEST_CB_KEY "printer_destination_check_button" + +#define PRINT_SUMMARY_CB_KEY "printer_summary_check_button" +#define PRINT_DETAILS_CB_KEY "printer_details_check_button" +#define PRINT_COLLAPSE_ALL_RB_KEY "printer_collapse_all_radio_button" +#define PRINT_AS_DISPLAYED_RB_KEY "printer_as_displayed_radio_button" +#define PRINT_EXPAND_ALL_RB_KEY "printer_expand_all_radio_button" +#define PRINT_HEX_CB_KEY "printer_hex_check_button" +#define PRINT_FORMFEED_CB_KEY "printer_formfeed_check_button" + +#define PRINT_BT_KEY "printer_button" + +#define PRINT_TE_PTR_KEY "printer_file_te_ptr" + + +/* + * Keep a static pointer to the current "Print" window, if any, so that if + * somebody tries to do "File:Print" while there's already a "Print" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *print_win = NULL; + +static print_args_t print_args; + +static gboolean print_prefs_init = FALSE; + + +static void +file_print_cmd(gboolean print_selected) +{ + print_args_t *args = &print_args; + + if (print_win != NULL) { + /* There's already a "Print" dialog box; reactivate it. */ + reactivate_window(print_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(print_prefs_init == FALSE) { + print_prefs_init = TRUE; + args->format = prefs.pr_format; + args->to_file = prefs.pr_dest; + args->file = g_strdup(prefs.pr_file); + args->cmd = g_strdup(prefs.pr_cmd); + args->print_summary = TRUE; + args->print_dissections = print_dissections_as_displayed; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + if(print_selected) { + args->range.process = range_process_selected; + } + + print_win = open_print_dialog("Wireshark: Print", output_action_print, args); + g_signal_connect(print_win, "destroy", G_CALLBACK(print_destroy_cb), &print_win); +} + +void +file_print_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + file_print_cmd(FALSE); +} + +void +file_print_selected_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + file_print_cmd(TRUE); +} + +/* + * Keep a static pointer to the current "Export text" window, if any, so that if + * somebody tries to do "File:Export to text" while there's already a "Export text" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_text_win = NULL; + +static print_args_t export_text_args; + +static gboolean export_text_prefs_init = FALSE; + + +#ifdef _WIN32 +void +export_text_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_text); + return; +} +#else +void +export_text_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_text_args; + + if (export_text_win != NULL) { + /* There's already a "Export text" dialog box; reactivate it. */ + reactivate_window(export_text_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_text_prefs_init == FALSE) { + export_text_prefs_init = TRUE; + args->format = PR_FMT_TEXT; + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = TRUE; + args->print_dissections = print_dissections_as_displayed; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_text_win = open_print_dialog("Wireshark: Export as \"Plain Text\" File", output_action_export_text, args); + g_signal_connect(export_text_win, "destroy", G_CALLBACK(print_destroy_cb), &export_text_win); +} +#endif + + +/* + * Keep a static pointer to the current "Export ps" window, if any, so that if + * somebody tries to do "File:Export to ps" while there's already a "Export ps" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_ps_win = NULL; + +static print_args_t export_ps_args; + +static gboolean export_ps_prefs_init = FALSE; + + +#ifdef _WIN32 +void +export_ps_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_ps); + return; +} +#else +void +export_ps_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_ps_args; + + if (export_ps_win != NULL) { + /* There's already a "Export ps" dialog box; reactivate it. */ + reactivate_window(export_ps_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_ps_prefs_init == FALSE) { + export_ps_prefs_init = TRUE; + args->format = PR_FMT_PS; + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = TRUE; + args->print_dissections = print_dissections_as_displayed; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_ps_win = open_print_dialog("Wireshark: Export as \"PostScript\" file", output_action_export_ps, args); + g_signal_connect(export_ps_win, "destroy", G_CALLBACK(print_destroy_cb), &export_ps_win); +} +#endif + + +/* + * Keep a static pointer to the current "Export psml" window, if any, so that if + * somebody tries to do "File:Export to psml" while there's already a "Export psml" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_psml_win = NULL; + +static print_args_t export_psml_args; + +static gboolean export_psml_prefs_init = FALSE; + + +#ifdef _WIN32 +void +export_psml_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_psml); + return; +} +#else +void +export_psml_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_psml_args; + + if (export_psml_win != NULL) { + /* There's already a "Export psml" dialog box; reactivate it. */ + reactivate_window(export_psml_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_psml_prefs_init == FALSE) { + export_psml_prefs_init = TRUE; + args->format = PR_FMT_TEXT; /* XXX */ + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = TRUE; + args->print_dissections = print_dissections_as_displayed; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_psml_win = open_print_dialog("Wireshark: Export as \"PSML\" file", output_action_export_psml, args); + g_signal_connect(export_psml_win, "destroy", G_CALLBACK(print_destroy_cb), &export_psml_win); +} +#endif + +/* + * Keep a static pointer to the current "Export pdml" window, if any, so that if + * somebody tries to do "File:Export to pdml" while there's already a "Export pdml" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_pdml_win = NULL; + +static print_args_t export_pdml_args; + +static gboolean export_pdml_prefs_init = FALSE; + + +#ifdef _WIN32 +void +export_pdml_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_pdml); + return; +} +#else +void +export_pdml_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_pdml_args; + + if (export_pdml_win != NULL) { + /* There's already a "Export pdml" dialog box; reactivate it. */ + reactivate_window(export_pdml_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_pdml_prefs_init == FALSE) { + export_pdml_prefs_init = TRUE; + args->format = PR_FMT_TEXT; /* XXX */ + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = TRUE; + args->print_dissections = print_dissections_as_displayed; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_pdml_win = open_print_dialog("Wireshark: Export as \"PDML\" file", output_action_export_pdml, args); + g_signal_connect(export_pdml_win, "destroy", G_CALLBACK(print_destroy_cb), &export_pdml_win); +} +#endif + +/* + * Keep a static pointer to the current "Export csv" window, if any, so that if + * somebody tries to do "File:Export to CSV" while there's already a "Export csv" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_csv_win = NULL; + +static print_args_t export_csv_args; + +static gboolean export_csv_prefs_init = FALSE; + +#ifdef _WIN32 +void +export_csv_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_csv); + return; +} +#else +void +export_csv_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_csv_args; + + if (export_csv_win != NULL) { + /* There's already a "Export csv" dialog box; reactivate it. */ + reactivate_window(export_csv_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_csv_prefs_init == FALSE) { + export_csv_prefs_init = TRUE; + args->format = PR_FMT_TEXT; /* XXX */ + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = FALSE; + args->print_dissections = print_dissections_none; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_csv_win = open_print_dialog("Wireshark: Export as \"Comma Separated Values\" File", output_action_export_csv, args); + g_signal_connect(export_csv_win, "destroy", G_CALLBACK(print_destroy_cb), &export_csv_win); +} +#endif + +/* + * Keep a static pointer to the current "Export carrays" window, if any, so that if + * somebody tries to do "File:Export to carrays" while there's already a "Export carrays" window + * up, we just pop up the existing one, rather than creating a new one. + */ +static GtkWidget *export_carrays_win = NULL; + +static print_args_t export_carrays_args; + +static gboolean export_carrays_prefs_init = FALSE; + +#ifdef _WIN32 +void +export_carrays_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + win32_export_file(GDK_WINDOW_HWND(gtk_widget_get_window(top_level)), export_type_carrays); + return; +} +#else +void +export_carrays_cmd_cb(GtkWidget *widget _U_, gpointer data _U_) +{ + print_args_t *args = &export_carrays_args; + + if (export_carrays_win != NULL) { + /* There's already a "Export carrays" dialog box; reactivate it. */ + reactivate_window(export_carrays_win); + return; + } + + /* get settings from preferences (and other initial values) only once */ + if(export_carrays_prefs_init == FALSE) { + export_carrays_prefs_init = TRUE; + args->format = PR_FMT_TEXT; + args->to_file = TRUE; + args->file = g_strdup(""); + args->cmd = g_strdup(""); + args->print_summary = FALSE; + args->print_dissections = print_dissections_none; + args->print_hex = FALSE; + args->print_formfeed = FALSE; + } + + /* init the printing range */ + packet_range_init(&args->range); + + export_carrays_win = open_print_dialog("Wireshark: Export as \"C Arrays\" File", + output_action_export_carrays, args); + g_signal_connect(export_carrays_win, "destroy", G_CALLBACK(print_destroy_cb), &export_carrays_win); +} +#endif + +static void +print_browse_file_cb(GtkWidget *file_bt, GtkWidget *file_te) +{ + file_selection_browse(file_bt, file_te, "Wireshark: Print to File", + FILE_SELECTION_WRITE_BROWSE); +} + + + +/* Open the print dialog */ +static GtkWidget * +open_print_dialog(const char *title, output_action_e action, print_args_t *args) +{ + GtkWidget *main_win; + GtkWidget *main_vb; + + GtkWidget *printer_fr, *printer_vb, *export_format_lb; + GtkWidget *text_rb, *ps_rb, *pdml_rb, *psml_rb, *csv_rb, *carrays_rb; + GtkWidget *printer_tb, *dest_cb; +#ifndef _WIN32 + GtkWidget *cmd_lb, *cmd_te; +#endif + GtkWidget *file_bt, *file_te; + + GtkWidget *range_fr, *range_tb; + + GtkWidget *packet_hb; + + GtkWidget *format_fr, *format_vb; + GtkWidget *summary_cb; + + GtkWidget *details_cb; + GtkWidget *details_hb, *details_vb; + GtkWidget *collapse_all_rb, *as_displayed_rb, *expand_all_rb; + GtkWidget *hex_cb; + GtkWidget *sep, *formfeed_cb; + + GtkWidget *bbox, *ok_bt, *cancel_bt, *help_bt; + + /* dialog window */ + main_win = dlg_window_new(title); + + /* Vertical enclosing container for each row of widgets */ + main_vb = gtk_vbox_new(FALSE, 5); + gtk_container_set_border_width(GTK_CONTAINER(main_vb), 5); + gtk_container_add(GTK_CONTAINER(main_win), main_vb); + gtk_widget_show(main_vb); + +/*****************************************************/ + + /*** printer frame ***/ + printer_fr = gtk_frame_new(action == output_action_print ? "Printer" : "Export to file:"); + gtk_box_pack_start(GTK_BOX(main_vb), printer_fr, FALSE, FALSE, 0); + gtk_widget_show(printer_fr); + printer_vb = gtk_vbox_new(FALSE, 5); + gtk_container_set_border_width(GTK_CONTAINER(printer_vb), 5); + gtk_container_add(GTK_CONTAINER(printer_fr), printer_vb); + gtk_widget_show(printer_vb); + + /* "Plain text" / "Postscript" / "PDML", ... radio buttons */ + text_rb = gtk_radio_button_new_with_mnemonic_from_widget(NULL, "Plain _text"); + if (args->format == PR_FMT_TEXT) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(text_rb), TRUE); + gtk_widget_set_tooltip_text(text_rb, "Print output in ascii \"plain text\" format. If you're unsure, use this format."); + gtk_box_pack_start(GTK_BOX(printer_vb), text_rb, FALSE, FALSE, 0); + if(action == output_action_print) + gtk_widget_show(text_rb); + + ps_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(text_rb), "_PostScript"); + if (args->format == PR_FMT_PS) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(ps_rb), TRUE); + gtk_widget_set_tooltip_text(ps_rb, "Print output in \"postscript\" format, for postscript capable printers or print servers."); + gtk_box_pack_start(GTK_BOX(printer_vb), ps_rb, FALSE, FALSE, 0); + if(action == output_action_print) + gtk_widget_show(ps_rb); + + pdml_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(text_rb), "PDM_L (XML: Packet Details Markup Language)"); + if (action == output_action_export_pdml) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(pdml_rb), TRUE); + gtk_widget_set_tooltip_text(pdml_rb, + "Print output in \"PDML\" (Packet Details Markup Language), " + "an XML based packet data interchange format. " + "Usually used in combination with the \"Output to file\" option to export packet data into an XML file."); + gtk_box_pack_start(GTK_BOX(printer_vb), pdml_rb, FALSE, FALSE, 0); + /* gtk_widget_show(pdml_rb); */ + + psml_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(text_rb), "PSML (XML: Packet Summary Markup Language)"); + if (action == output_action_export_psml) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(psml_rb), TRUE); + gtk_widget_set_tooltip_text(psml_rb, + "Print output in \"PSML\" (Packet Summary Markup Language), " + "an XML based packet summary interchange format. " + "Usually used in combination with the \"Output to file\" option to export packet data into an XML file."); + gtk_box_pack_start(GTK_BOX(printer_vb), psml_rb, FALSE, FALSE, 0); + /* gtk_widget_show(psml_rb); */ + + csv_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(text_rb), "_CSV"); + if (action == output_action_export_csv) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(csv_rb), TRUE); + gtk_widget_set_tooltip_text(csv_rb, + "Print output in \"Comma Separated Values\" (CSV) format, " + "a text format compatible with OpenOffice and Excel. " + "One row for each packet, with its timestamp and size."); + gtk_box_pack_start(GTK_BOX(printer_vb), csv_rb, FALSE, FALSE, 0); + /* gtk_widget_show(csv_rb); */ + + carrays_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(text_rb), "C Arrays"); + if (action == output_action_export_carrays) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(carrays_rb), TRUE); + gtk_widget_set_tooltip_text(carrays_rb, + "Print output in C Arrays format, " + "a text file suitable for use in C/C++ programs. " + "One char[] for each packet."); + gtk_box_pack_start(GTK_BOX(printer_vb), carrays_rb, FALSE, FALSE, 0); + /* gtk_widget_show(carrays_rb); */ + + /* printer table */ +#ifndef _WIN32 + printer_tb = gtk_table_new(2, 3, FALSE); +#else + printer_tb = gtk_table_new(2, 2, FALSE); +#endif + gtk_box_pack_start(GTK_BOX(printer_vb), printer_tb, FALSE, FALSE, 0); + gtk_table_set_row_spacings(GTK_TABLE(printer_tb), 5); + gtk_table_set_col_spacings(GTK_TABLE(printer_tb), 5); + gtk_widget_show(printer_tb); + + + /* Output to file button */ + dest_cb = gtk_check_button_new_with_mnemonic("Output to _file:"); + if (args->to_file) + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(dest_cb), TRUE); + gtk_widget_set_tooltip_text(dest_cb, "Output to file instead of printer"); + gtk_table_attach_defaults(GTK_TABLE(printer_tb), dest_cb, 0, 1, 0, 1); + if(action == output_action_print) + gtk_widget_show(dest_cb); + + /* File text entry */ + file_te = gtk_entry_new(); + g_object_set_data(G_OBJECT(dest_cb), PRINT_FILE_TE_KEY, file_te); + gtk_widget_set_tooltip_text(file_te, "Enter Output filename"); + gtk_entry_set_text(GTK_ENTRY(file_te), args->file); + gtk_table_attach_defaults(GTK_TABLE(printer_tb), file_te, 1, 2, 0, 1); + gtk_widget_set_sensitive(file_te, args->to_file); + gtk_widget_show(file_te); + if (args->to_file) + gtk_widget_grab_focus(file_te); + + /* "Browse" button */ + file_bt = gtk_button_new_from_stock(WIRESHARK_STOCK_BROWSE); + g_object_set_data(G_OBJECT(dest_cb), PRINT_FILE_BT_KEY, file_bt); + g_object_set_data(G_OBJECT(file_bt), PRINT_TE_PTR_KEY, file_te); + gtk_widget_set_tooltip_text(file_bt, "Browse output filename in filesystem"); + gtk_table_attach_defaults(GTK_TABLE(printer_tb), file_bt, 2, 3, 0, 1); + gtk_widget_set_sensitive(file_bt, args->to_file); + gtk_widget_show(file_bt); + + /* Command label and text entry */ +#ifndef _WIN32 + cmd_lb = gtk_label_new("Print command:"); + g_object_set_data(G_OBJECT(dest_cb), PRINT_CMD_LB_KEY, cmd_lb); + gtk_misc_set_alignment(GTK_MISC(cmd_lb), 1.0f, 0.5f); + gtk_table_attach_defaults(GTK_TABLE(printer_tb), cmd_lb, 0, 1, 1, 2); + gtk_widget_set_sensitive(cmd_lb, !args->to_file); + if(action == output_action_print) + gtk_widget_show(cmd_lb); + + cmd_te = gtk_entry_new(); + g_object_set_data(G_OBJECT(dest_cb), PRINT_CMD_TE_KEY, cmd_te); + gtk_widget_set_tooltip_text(cmd_te, "Enter print command"); + gtk_entry_set_text(GTK_ENTRY(cmd_te), args->cmd); + gtk_table_attach_defaults(GTK_TABLE(printer_tb), cmd_te, 1, 2, 1, 2); + gtk_widget_set_sensitive(cmd_te, !args->to_file); + if(action == output_action_print) + gtk_widget_show(cmd_te); +#endif + + g_signal_connect(dest_cb, "toggled", G_CALLBACK(print_cmd_toggle_dest), NULL); + g_signal_connect(file_bt, "clicked", G_CALLBACK(print_browse_file_cb), file_te); + + if(action == output_action_export_ps) { + export_format_lb = gtk_label_new("(PostScript files can be easily converted to PDF files using ghostscript's ps2pdf)"); + gtk_box_pack_start(GTK_BOX(printer_vb), export_format_lb, FALSE, FALSE, 0); + gtk_widget_show(export_format_lb); + } + +/*****************************************************/ + + /*** hor box for range and format frames ***/ + packet_hb = gtk_hbox_new(FALSE, 5); + gtk_container_add(GTK_CONTAINER(main_vb), packet_hb); + gtk_widget_show(packet_hb); + + /*** packet range frame ***/ + range_fr = gtk_frame_new("Packet Range"); + gtk_box_pack_start(GTK_BOX(packet_hb), range_fr, FALSE, FALSE, 0); + gtk_widget_show(range_fr); + + range_tb = range_new(&args->range); + gtk_container_add(GTK_CONTAINER(range_fr), range_tb); + gtk_widget_show(range_tb); + +/*****************************************************/ + + /*** packet format frame ***/ + format_fr = gtk_frame_new("Packet Format"); + gtk_box_pack_start(GTK_BOX(packet_hb), format_fr, TRUE, TRUE, 0); + if( action == output_action_print || + action == output_action_export_text || + action == output_action_export_ps) + gtk_widget_show(format_fr); + format_vb = gtk_vbox_new(FALSE, 5); + gtk_container_set_border_width(GTK_CONTAINER(format_vb), 5); + gtk_container_add(GTK_CONTAINER(format_fr), format_vb); + gtk_widget_show(format_vb); + + /* "Print summary line" check button */ + summary_cb = gtk_check_button_new_with_mnemonic("Packet summary line"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(summary_cb), args->print_summary); + g_signal_connect(summary_cb, "clicked", G_CALLBACK(print_cmd_toggle_detail), main_win); + gtk_widget_set_tooltip_text(summary_cb, "Output of a packet summary line, like in the packet list"); + gtk_container_add(GTK_CONTAINER(format_vb), summary_cb); + gtk_widget_show(summary_cb); + + + /* "Details" check button */ + details_cb = gtk_check_button_new_with_mnemonic("Packet details:"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(details_cb), args->print_dissections != print_dissections_none); + g_signal_connect(details_cb, "clicked", G_CALLBACK(print_cmd_toggle_detail), main_win); + gtk_widget_set_tooltip_text(details_cb, "Output format of the selected packet details (protocol tree)."); + gtk_container_add(GTK_CONTAINER(format_vb), details_cb); + gtk_widget_show(details_cb); + + /*** packet details ***/ + details_hb = gtk_hbox_new(FALSE, 6); + gtk_container_set_border_width(GTK_CONTAINER(details_hb), 0); + gtk_container_add(GTK_CONTAINER(format_vb), details_hb); + gtk_widget_show(details_hb); + + details_vb = gtk_vbox_new(FALSE, 6); + gtk_container_set_border_width(GTK_CONTAINER(details_vb), 0); + gtk_container_add(GTK_CONTAINER(details_hb), details_vb); + gtk_widget_show(details_vb); + + details_vb = gtk_vbox_new(FALSE, 6); + gtk_container_set_border_width(GTK_CONTAINER(details_vb), 0); + gtk_container_add(GTK_CONTAINER(details_hb), details_vb); + gtk_widget_show(details_vb); + + /* "All collapsed"/"As displayed"/"All Expanded" radio buttons */ + collapse_all_rb = gtk_radio_button_new_with_mnemonic_from_widget(NULL, "All co_llapsed"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(collapse_all_rb), args->print_dissections == print_dissections_collapsed); + gtk_widget_set_tooltip_text(collapse_all_rb, "Output of the packet details tree \"collapsed\""); + gtk_container_add(GTK_CONTAINER(details_vb), collapse_all_rb); + gtk_widget_show(collapse_all_rb); + + as_displayed_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(collapse_all_rb), "As displa_yed"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(as_displayed_rb), args->print_dissections == print_dissections_as_displayed); + gtk_widget_set_tooltip_text(as_displayed_rb, "Output of the packet details tree \"as displayed\""); + gtk_container_add(GTK_CONTAINER(details_vb), as_displayed_rb); + gtk_widget_show(as_displayed_rb); + + expand_all_rb = gtk_radio_button_new_with_mnemonic_from_widget(GTK_RADIO_BUTTON(collapse_all_rb), "All e_xpanded"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(expand_all_rb), args->print_dissections == print_dissections_expanded); + gtk_widget_set_tooltip_text(expand_all_rb, "Output of the packet details tree \"expanded\""); + gtk_container_add(GTK_CONTAINER(details_vb), expand_all_rb); + gtk_widget_show(expand_all_rb); + + /* "Print hex" check button. */ + hex_cb = gtk_check_button_new_with_mnemonic("Packet bytes"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(hex_cb), args->print_hex); + g_signal_connect(hex_cb, "clicked", G_CALLBACK(print_cmd_toggle_detail), main_win); + gtk_widget_set_tooltip_text(hex_cb, "Add a hexdump of the packet data"); + gtk_container_add(GTK_CONTAINER(format_vb), hex_cb); + gtk_widget_show(hex_cb); + + /* seperator */ + sep = gtk_hseparator_new(); + gtk_container_add(GTK_CONTAINER(format_vb), sep); + gtk_widget_show(sep); + + /* "Each packet on a new page" check button. */ + formfeed_cb = gtk_check_button_new_with_mnemonic("Each packet on a new page"); + gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(formfeed_cb), args->print_formfeed); + gtk_widget_set_tooltip_text (formfeed_cb, "When checked, a new page will be used for each packet. " + "This is done by adding a formfeed (or similar) between the packet outputs."); + gtk_container_add(GTK_CONTAINER(format_vb), formfeed_cb); + gtk_widget_show(formfeed_cb); + + + g_object_set_data(G_OBJECT(main_win), PRINT_ARGS_KEY, args); + g_object_set_data(G_OBJECT(main_win), PRINT_SUMMARY_CB_KEY, summary_cb); + g_object_set_data(G_OBJECT(main_win), PRINT_DETAILS_CB_KEY, details_cb); + g_object_set_data(G_OBJECT(main_win), PRINT_COLLAPSE_ALL_RB_KEY, collapse_all_rb); + g_object_set_data(G_OBJECT(main_win), PRINT_AS_DISPLAYED_RB_KEY, as_displayed_rb); + g_object_set_data(G_OBJECT(main_win), PRINT_EXPAND_ALL_RB_KEY, expand_all_rb); + g_object_set_data(G_OBJECT(main_win), PRINT_HEX_CB_KEY, hex_cb); + +/*****************************************************/ + + + /* Button row */ + bbox = dlg_button_row_new(action == output_action_print ? GTK_STOCK_PRINT : GTK_STOCK_OK, GTK_STOCK_CANCEL, GTK_STOCK_HELP, NULL); + gtk_box_pack_start(GTK_BOX(main_vb), bbox, FALSE, FALSE, 0); + gtk_widget_show(bbox); + + ok_bt = g_object_get_data(G_OBJECT(bbox), action == output_action_print ? GTK_STOCK_PRINT : GTK_STOCK_OK); + + g_object_set_data(G_OBJECT(main_win), PRINT_BT_KEY, ok_bt); + + g_object_set_data(G_OBJECT(ok_bt), PRINT_PS_RB_KEY, ps_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_PDML_RB_KEY, pdml_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_PSML_RB_KEY, psml_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_CSV_RB_KEY, csv_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_CARRAYS_RB_KEY, carrays_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_DEST_CB_KEY, dest_cb); +#ifndef _WIN32 + g_object_set_data(G_OBJECT(ok_bt), PRINT_CMD_TE_KEY, cmd_te); +#endif + + g_object_set_data(G_OBJECT(ok_bt), PRINT_ARGS_KEY, args); + g_object_set_data(G_OBJECT(ok_bt), PRINT_FILE_TE_KEY, file_te); + g_object_set_data(G_OBJECT(ok_bt), PRINT_SUMMARY_CB_KEY, summary_cb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_DETAILS_CB_KEY, details_cb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_COLLAPSE_ALL_RB_KEY, collapse_all_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_AS_DISPLAYED_RB_KEY, as_displayed_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_EXPAND_ALL_RB_KEY, expand_all_rb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_HEX_CB_KEY, hex_cb); + g_object_set_data(G_OBJECT(ok_bt), PRINT_FORMFEED_CB_KEY, formfeed_cb); + g_signal_connect(ok_bt, "clicked", G_CALLBACK(print_ok_cb), main_win); + gtk_widget_set_tooltip_text (ok_bt, "Start output"); + + cancel_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_CANCEL); + window_set_cancel_button(main_win, cancel_bt, window_cancel_button_cb); + gtk_widget_set_tooltip_text (cancel_bt, "Cancel and exit dialog"); + + if(action == output_action_print) { + help_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_HELP); + g_signal_connect(help_bt, "clicked", G_CALLBACK(topic_cb), (gpointer)HELP_PRINT_DIALOG); + } else { +#ifdef _WIN32 + help_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_HELP); + g_signal_connect(help_bt, "clicked", G_CALLBACK(topic_cb), (gpointer)HELP_EXPORT_FILE_WIN32_DIALOG); +#else + help_bt = g_object_get_data(G_OBJECT(bbox), GTK_STOCK_HELP); + g_signal_connect(help_bt, "clicked", G_CALLBACK(topic_cb), (gpointer)HELP_EXPORT_FILE_DIALOG); +#endif + } + + gtk_widget_grab_default(ok_bt); + + /* Catch the "activate" signal on the "Command" and "File" text entries, + so that if the user types Return there, we act as if the "OK" button + had been selected, as happens if Return is typed if some widget + that *doesn't* handle the Return key has the input focus. */ +#ifndef _WIN32 + dlg_set_activate(cmd_te, ok_bt); +#endif + if(action != output_action_print) + dlg_set_activate(file_te, ok_bt); + + g_signal_connect(main_win, "delete_event", G_CALLBACK(window_delete_event_cb), NULL); + + gtk_widget_show(main_win); + window_present(main_win); + + return main_win; +} + + +/* user changed "print to" destination */ +static void +print_cmd_toggle_dest(GtkWidget *widget, gpointer data _U_) +{ +#ifndef _WIN32 + GtkWidget *cmd_lb, *cmd_te; +#endif + GtkWidget *file_bt, *file_te; + int to_file; + +#ifndef _WIN32 + cmd_lb = GTK_WIDGET(g_object_get_data(G_OBJECT(widget), PRINT_CMD_LB_KEY)); + cmd_te = GTK_WIDGET(g_object_get_data(G_OBJECT(widget), PRINT_CMD_TE_KEY)); +#endif + file_bt = GTK_WIDGET(g_object_get_data(G_OBJECT(widget), PRINT_FILE_BT_KEY)); + file_te = GTK_WIDGET(g_object_get_data(G_OBJECT(widget), PRINT_FILE_TE_KEY)); + + to_file = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (widget)); +#ifndef _WIN32 + gtk_widget_set_sensitive(cmd_lb, !to_file); + gtk_widget_set_sensitive(cmd_te, !to_file); +#endif + gtk_widget_set_sensitive(file_bt, to_file); + gtk_widget_set_sensitive(file_te, to_file); +} + + +/* user changed "packet details" */ +static void +print_cmd_toggle_detail(GtkWidget *widget _U_, gpointer data) +{ + GtkWidget *print_bt, *summary_cb, *details_cb, *collapse_all_rb, *expand_all_rb, *as_displayed_rb, *hex_cb; + gboolean print_detail; + + + print_bt = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_BT_KEY)); + summary_cb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_SUMMARY_CB_KEY)); + details_cb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_DETAILS_CB_KEY)); + collapse_all_rb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_COLLAPSE_ALL_RB_KEY)); + as_displayed_rb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_AS_DISPLAYED_RB_KEY)); + expand_all_rb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_EXPAND_ALL_RB_KEY)); + hex_cb = GTK_WIDGET(g_object_get_data(G_OBJECT(data), PRINT_HEX_CB_KEY)); + + /* is user disabled details, disable the corresponding buttons */ + print_detail = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (details_cb)); + gtk_widget_set_sensitive(collapse_all_rb, print_detail); + gtk_widget_set_sensitive(as_displayed_rb, print_detail); + gtk_widget_set_sensitive(expand_all_rb, print_detail); + + /* if user selected nothing to print at all, disable the "ok" button */ + gtk_widget_set_sensitive(print_bt, + gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (summary_cb)) || + gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (details_cb)) || + gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (hex_cb))); +} + + +static void +print_ok_cb(GtkWidget *ok_bt, gpointer parent_w) +{ + GtkWidget *button; + print_args_t *args; + const gchar *g_dest; + gchar *f_name; + gchar *dirname; + gboolean export_as_pdml = FALSE, export_as_psml = FALSE; + gboolean export_as_csv = FALSE; + gboolean export_as_carrays = FALSE; +#ifdef _WIN32 + gboolean win_printer = FALSE; + int tmp_fd; + char *tmp_namebuf; + char *tmp_oldfile = NULL; +#endif + cf_print_status_t status; + + args = (print_args_t *)g_object_get_data(G_OBJECT(ok_bt), PRINT_ARGS_KEY); + + /* Check whether the range is valid. */ + if (!range_check_validity(&args->range)) { + /* The range isn't valid; don't dismiss the print/export dialog box, + just leave it around so that the user can, after they + dismiss the alert box popped up for the error, try again. */ + return; + } + + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_DEST_CB_KEY); + args->to_file = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button)); + + if (args->to_file) { + g_dest = gtk_entry_get_text(GTK_ENTRY(g_object_get_data(G_OBJECT(ok_bt), + PRINT_FILE_TE_KEY))); + if (!g_dest[0]) { + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, + "Output to file, but no file specified."); + return; + } + g_free(args->file); + args->file = g_strdup(g_dest); + /* Save the directory name for future file dialogs. */ + f_name = g_strdup(g_dest); + dirname = get_dirname(f_name); /* Overwrites f_name */ + set_last_open_dir(dirname); + g_free(f_name); + } else { +#ifdef _WIN32 + win_printer = TRUE; + /* We currently don't have a function in util.h to create just a tempfile */ + /* name, so simply create a tempfile using the "official" function, */ + /* then delete this file again. After this, the name MUST be available. */ + /* */ + /* Don't use tmpnam() or such, as this will fail under some ACL */ + /* circumstances: http://bugs.wireshark.org/bugzilla/show_bug.cgi?id=358 */ + /* Also: tmpnam is "insecure" and should not be used. */ + tmp_fd = create_tempfile(&tmp_namebuf, "wshprint"); + if(tmp_fd == -1) { + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, + "Couldn't create a temporary file for printing:\n%s", tmp_namebuf); + return; + } + /* remember to restore these values later! */ + tmp_oldfile = args->file; + args->file = g_strdup(tmp_namebuf); + ws_unlink(args->file); + args->to_file = TRUE; +#else + g_free(args->cmd); + args->cmd = g_strdup(gtk_entry_get_text(GTK_ENTRY(g_object_get_data(G_OBJECT(ok_bt), + PRINT_CMD_TE_KEY)))); +#endif + } + + args->format = PR_FMT_TEXT; + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_PS_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) + args->format = PR_FMT_PS; + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_PDML_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) + export_as_pdml = TRUE; + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_PSML_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) + export_as_psml = TRUE; + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_CSV_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) + export_as_csv = TRUE; + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_CARRAYS_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) + export_as_carrays = TRUE; + + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_SUMMARY_CB_KEY); + args->print_summary = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button)); + + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_COLLAPSE_ALL_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) { + args->print_dissections = print_dissections_collapsed; + } + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_AS_DISPLAYED_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) { + args->print_dissections = print_dissections_as_displayed; + } + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_EXPAND_ALL_RB_KEY); + if (gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) { + args->print_dissections = print_dissections_expanded; + } + + /* the details setting has priority over the radio buttons */ + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_DETAILS_CB_KEY); + if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button))) { + args->print_dissections = print_dissections_none; + } + + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_HEX_CB_KEY); + args->print_hex = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button)); + + button = (GtkWidget *)g_object_get_data(G_OBJECT(ok_bt), PRINT_FORMFEED_CB_KEY); + args->print_formfeed = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON (button)); + + + window_destroy(GTK_WIDGET(parent_w)); + + /* Now print/export the packets */ + if (export_as_pdml) + status = cf_write_pdml_packets(&cfile, args); + else if (export_as_psml) + status = cf_write_psml_packets(&cfile, args); + else if (export_as_csv) + status = cf_write_csv_packets(&cfile, args); + else if (export_as_carrays) + status = cf_write_carrays_packets(&cfile, args); + else { + switch (args->format) { + + case PR_FMT_TEXT: + if (args->to_file) { + args->stream = print_stream_text_new(TRUE, args->file); + if (args->stream == NULL) { + open_failure_alert_box(args->file, errno, TRUE); + return; + } + } else { + args->stream = print_stream_text_new(FALSE, args->cmd); + if (args->stream == NULL) { + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, + "Couldn't run print command %s.", args->cmd); + } + } + break; + + case PR_FMT_PS: + if (args->to_file) { + args->stream = print_stream_ps_new(TRUE, args->file); + if (args->stream == NULL) { + open_failure_alert_box(args->file, errno, TRUE); + return; + } + } else { + args->stream = print_stream_ps_new(FALSE, args->cmd); + if (args->stream == NULL) { + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, + "Couldn't run print command %s.", args->cmd); + } + } + break; + + default: + g_assert_not_reached(); + return; + } + status = cf_print_packets(&cfile, args); + } + switch (status) { + + case CF_PRINT_OK: + break; + + case CF_PRINT_OPEN_ERROR: + if (args->to_file) + open_failure_alert_box(args->file, errno, TRUE); + else + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "Couldn't run print command %s.", + args->cmd); + break; + + case CF_PRINT_WRITE_ERROR: + if (args->to_file) + write_failure_alert_box(args->file, errno); + else + simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, + "Error writing to print command: %s", g_strerror(errno)); + break; + } + +#ifdef _WIN32 + if (win_printer) { + print_mswin(args->file); + + /* trash temp file */ + ws_remove(args->file); + g_free(args->file); + + /* restore old settings */ + args->to_file = FALSE; + args->file = tmp_oldfile; + } +#endif +} + +static void +print_destroy_cb(GtkWidget *win, gpointer user_data) +{ + GtkWidget *fs; + + /* Is there a file selection dialog associated with this + Print File dialog? */ + fs = g_object_get_data(G_OBJECT(win), E_FILE_SEL_DIALOG_PTR_KEY); + + if (fs != NULL) { + /* Yes. Destroy it. */ + window_destroy(fs); + } + + /* Note that we no longer have a "Print" dialog box. */ + *((gpointer *) user_data) = NULL; +} |