wireshark/ui/qt/main_window_slots.cpp

4204 lines
136 KiB
C++

/* main_window_slots.cpp
*
* Wireshark - Network traffic analyzer
* By Gerald Combs <gerald@wireshark.org>
* Copyright 1998 Gerald Combs
*
* SPDX-License-Identifier: GPL-2.0-or-later
*/
#include <config.h>
// Qt 5.5.0 + Visual C++ 2013
#ifdef _MSC_VER
#pragma warning(push)
#pragma warning(disable:4996)
#endif
#include "main_window.h"
/*
* The generated Ui_MainWindow::setupUi() can grow larger than our configured limit,
* so turn off -Wframe-larger-than= for ui_main_window.h.
*/
DIAG_OFF(frame-larger-than=)
#include <ui_main_window.h>
DIAG_ON(frame-larger-than=)
#ifdef _WIN32
#include <windows.h>
#endif
#include "ui/dissect_opts.h"
#ifdef HAVE_LIBPCAP
#include "ui/capture.h"
#endif
#include "ui/commandline.h"
#include "ui/urls.h"
#include "epan/color_filters.h"
#include "epan/export_object.h"
#include "wsutil/file_util.h"
#include "wsutil/filesystem.h"
#include <wsutil/wslog.h>
#include <wsutil/ws_assert.h>
#include "epan/addr_resolv.h"
#include "epan/column.h"
#include "epan/dfilter/dfilter-macro.h"
#include "epan/conversation_filter.h"
#include "epan/epan_dissect.h"
#include "epan/filter_expressions.h"
#include "epan/prefs.h"
#include "epan/plugin_if.h"
#include "epan/uat.h"
#include "epan/uat-int.h"
#include "epan/value_string.h"
#ifdef HAVE_LUA
#include <epan/wslua/init_wslua.h>
#endif
#include "ui/alert_box.h"
#ifdef HAVE_LIBPCAP
#include "ui/capture_ui_utils.h"
#endif
#include "ui/capture_globals.h"
#include "ui/help_url.h"
#include "ui/main_statusbar.h"
#include "ui/preference_utils.h"
#include "ui/recent.h"
#include "ui/recent_utils.h"
#include "ui/ssl_key_export.h"
#include "ui/ws_ui_util.h"
#include "ui/all_files_wildcard.h"
#include "ui/qt/simple_dialog.h"
#include <ui/qt/utils/variant_pointer.h>
#include <ui/qt/widgets/drag_drop_toolbar.h>
#include "ui/qt/widgets/wireshark_file_dialog.h"
#ifdef HAVE_SOFTWARE_UPDATE
#include "ui/software_update.h"
#endif
#include "about_dialog.h"
#include "bluetooth_att_server_attributes_dialog.h"
#include "bluetooth_devices_dialog.h"
#include "bluetooth_hci_summary_dialog.h"
#include "capture_file_dialog.h"
#include "capture_file_properties_dialog.h"
#ifdef HAVE_LIBPCAP
#include "capture_options_dialog.h"
#endif
#include <ui/qt/utils/color_utils.h>
#include "coloring_rules_dialog.h"
#include "conversation_dialog.h"
#include "conversation_colorize_action.h"
#include "conversation_hash_tables_dialog.h"
#include "enabled_protocols_dialog.h"
#include "decode_as_dialog.h"
#include <ui/qt/widgets/display_filter_edit.h>
#include "display_filter_expression_dialog.h"
#include "dissector_tables_dialog.h"
#include "endpoint_dialog.h"
#include "expert_info_dialog.h"
#include "export_object_action.h"
#include "export_object_dialog.h"
#include "export_pdu_dialog.h"
#include "extcap_options_dialog.h"
#include "file_set_dialog.h"
#include "filter_action.h"
#include "filter_dialog.h"
#include "firewall_rules_dialog.h"
#include "funnel_statistics.h"
#include "gsm_map_summary_dialog.h"
#include "iax2_analysis_dialog.h"
#include "interface_toolbar.h"
#include "io_graph_dialog.h"
#include <ui/qt/widgets/additional_toolbar.h>
#include "lbm_stream_dialog.h"
#include "lbm_lbtrm_transport_dialog.h"
#include "lbm_lbtru_transport_dialog.h"
#include "lte_mac_statistics_dialog.h"
#include "lte_rlc_statistics_dialog.h"
#include "lte_rlc_graph_dialog.h"
#include "mtp3_summary_dialog.h"
#include "multicast_statistics_dialog.h"
#include "packet_comment_dialog.h"
#include "packet_diagram.h"
#include "packet_dialog.h"
#include "packet_list.h"
#include "credentials_dialog.h"
#include "preferences_dialog.h"
#include "print_dialog.h"
#include "profile_dialog.h"
#include "protocol_hierarchy_dialog.h"
#include <ui/qt/utils/qt_ui_utils.h>
#include "resolved_addresses_dialog.h"
#include "rpc_service_response_time_dialog.h"
#include "rtp_stream_dialog.h"
#include "rtp_analysis_dialog.h"
#include "sctp_all_assocs_dialog.h"
#include "sctp_assoc_analyse_dialog.h"
#include "sctp_graph_dialog.h"
#include "sequence_dialog.h"
#include "show_packet_bytes_dialog.h"
#include "stats_tree_dialog.h"
#include <ui/qt/utils/stock_icon.h>
#include "supported_protocols_dialog.h"
#include "tap_parameter_dialog.h"
#include "tcp_stream_dialog.h"
#include "time_shift_dialog.h"
#include "uat_dialog.h"
#include "voip_calls_dialog.h"
#include "wireshark_application.h"
#include "wlan_statistics_dialog.h"
#include <ui/qt/widgets/wireless_timeline.h>
#include <functional>
#include <QClipboard>
#include <QFileInfo>
#include <QMessageBox>
#include <QMetaObject>
#include <QToolBar>
#include <QDesktopServices>
#include <QUrl>
#include <QMutex>
// XXX You must uncomment QT_WINEXTRAS_LIB lines in CMakeList.txt and
// cmakeconfig.h.in.
// #if defined(QT_WINEXTRAS_LIB)
// #include <QWinJumpList>
// #include <QWinJumpListCategory>
// #include <QWinJumpListItem>
// #endif
//
// Public slots
//
bool MainWindow::openCaptureFile(QString cf_path, QString read_filter, unsigned int type, gboolean is_tempfile)
{
QString file_name = "";
dfilter_t *rfcode = NULL;
gchar *err_msg;
int err;
gboolean name_param;
gboolean ret = true;
// was a file name given as function parameter?
name_param = !cf_path.isEmpty();
for (;;) {
if (cf_path.isEmpty()) {
CaptureFileDialog open_dlg(this, capture_file_.capFile(), read_filter);
if (open_dlg.open(file_name, type)) {
cf_path = file_name;
} else {
ret = false;
goto finish;
}
} else {
this->welcome_page_->getInterfaceFrame()->showRunOnFile();
}
// TODO detect call from "cf_read" -> "update_progress_dlg"
// ("capture_file_.capFile()->read_lock"), possibly queue opening the
// file and return early to avoid the warning in testCaptureFileClose.
QString before_what(tr(" before opening another file"));
if (!testCaptureFileClose(before_what)) {
ret = false;
goto finish;
}
if (dfilter_compile(qUtf8Printable(read_filter), &rfcode, &err_msg)) {
cf_set_rfcode(CaptureFile::globalCapFile(), rfcode);
} else {
/* Not valid. Tell the user, and go back and run the file
selection box again once they dismiss the alert. */
//bad_dfilter_alert_box(top_level, read_filter->str);
QMessageBox::warning(this, tr("Invalid Display Filter"),
QString("The filter expression ") +
read_filter +
QString(" isn't a valid display filter. (") +
err_msg + QString(")."),
QMessageBox::Ok);
if (!name_param) {
// go back to the selection dialogue only if the file
// was selected from this dialogue
cf_path.clear();
continue;
}
}
/* Make the file name available via MainWindow */
setMwFileName(cf_path);
/* Try to open the capture file. This closes the current file if it succeeds. */
CaptureFile::globalCapFile()->window = this;
if (cf_open(CaptureFile::globalCapFile(), qUtf8Printable(cf_path), type, is_tempfile, &err) != CF_OK) {
/* We couldn't open it; don't dismiss the open dialog box,
just leave it around so that the user can, after they
dismiss the alert box popped up for the open error,
try again. */
CaptureFile::globalCapFile()->window = NULL;
dfilter_free(rfcode);
cf_path.clear();
continue;
}
switch (cf_read(CaptureFile::globalCapFile(), FALSE)) {
case CF_READ_OK:
case CF_READ_ERROR:
/* Just because we got an error, that doesn't mean we were unable
to read any of the file; we handle what we could get from the
file. */
break;
case CF_READ_ABORTED:
/* The user bailed out of re-reading the capture file; the
capture file has been closed - just free the capture file name
string and return (without changing the last containing
directory). */
capture_file_.setCapFile(NULL);
ret = false;
goto finish;
}
break;
}
wsApp->setLastOpenDirFromFilename(cf_path);
main_ui_->statusBar->showExpert();
finish:
#ifdef HAVE_LIBPCAP
if (global_commandline_info.quit_after_cap)
exit(0);
#endif
return ret;
}
void MainWindow::filterPackets(QString new_filter, bool force)
{
cf_status_t cf_status;
cf_status = cf_filter_packets(CaptureFile::globalCapFile(), new_filter.toUtf8().data(), force);
if (cf_status == CF_OK) {
emit displayFilterSuccess(true);
if (new_filter.length() > 0) {
int index = df_combo_box_->findText(new_filter);
if (index == -1) {
df_combo_box_->insertItem(0, new_filter);
df_combo_box_->setCurrentIndex(0);
} else {
df_combo_box_->setCurrentIndex(index);
}
} else {
df_combo_box_->lineEdit()->clear();
}
} else {
emit displayFilterSuccess(false);
}
if (packet_list_) {
packet_list_->resetColumns();
}
}
void MainWindow::layoutToolbars()
{
Qt::ToolButtonStyle tbstyle = Qt::ToolButtonIconOnly;
switch (prefs.gui_toolbar_main_style) {
case TB_STYLE_TEXT:
tbstyle = Qt::ToolButtonTextOnly;
break;
case TB_STYLE_BOTH:
tbstyle = Qt::ToolButtonTextUnderIcon;
break;
}
main_ui_->mainToolBar->setToolButtonStyle(tbstyle);
main_ui_->mainToolBar->setVisible(recent.main_toolbar_show);
main_ui_->displayFilterToolBar->setVisible(recent.filter_toolbar_show);
#if defined(HAVE_LIBNL) && defined(HAVE_NL80211)
main_ui_->wirelessToolBar->setVisible(recent.wireless_toolbar_show);
#endif
main_ui_->statusBar->setVisible(recent.statusbar_show);
foreach(QAction *action, main_ui_->menuInterfaceToolbars->actions()) {
QToolBar *toolbar = action->data().value<QToolBar *>();
if (g_list_find_custom(recent.interface_toolbars, action->text().toUtf8(), (GCompareFunc)strcmp)) {
toolbar->setVisible(true);
} else {
toolbar->setVisible(false);
}
}
QList<QToolBar *> toolbars = findChildren<QToolBar *>();
foreach(QToolBar *bar, toolbars) {
AdditionalToolBar *iftoolbar = dynamic_cast<AdditionalToolBar *>(bar);
if (iftoolbar) {
bool visible = false;
if (g_list_find_custom(recent.gui_additional_toolbars, qUtf8Printable(iftoolbar->menuName()), (GCompareFunc)strcmp))
visible = true;
iftoolbar->setVisible(visible);
}
}
}
void MainWindow::updatePreferenceActions()
{
main_ui_->actionViewPacketList->setEnabled(prefs_has_layout_pane_content(layout_pane_content_plist));
main_ui_->actionViewPacketDetails->setEnabled(prefs_has_layout_pane_content(layout_pane_content_pdetails));
main_ui_->actionViewPacketBytes->setEnabled(prefs_has_layout_pane_content(layout_pane_content_pbytes));
main_ui_->actionViewPacketDiagram->setEnabled(prefs_has_layout_pane_content(layout_pane_content_pdiagram));
main_ui_->actionViewNameResolutionPhysical->setChecked(gbl_resolv_flags.mac_name);
main_ui_->actionViewNameResolutionNetwork->setChecked(gbl_resolv_flags.network_name);
main_ui_->actionViewNameResolutionTransport->setChecked(gbl_resolv_flags.transport_name);
// Should this be a "recent" setting?
main_ui_->actionGoAutoScroll->setChecked(prefs.capture_auto_scroll);
}
void MainWindow::updateRecentActions()
{
main_ui_->actionViewMainToolbar->setChecked(recent.main_toolbar_show);
main_ui_->actionViewFilterToolbar->setChecked(recent.filter_toolbar_show);
main_ui_->actionViewWirelessToolbar->setChecked(recent.wireless_toolbar_show);
main_ui_->actionViewStatusBar->setChecked(recent.statusbar_show);
main_ui_->actionViewPacketList->setChecked(recent.packet_list_show && prefs_has_layout_pane_content(layout_pane_content_plist));
main_ui_->actionViewPacketDetails->setChecked(recent.tree_view_show && prefs_has_layout_pane_content(layout_pane_content_pdetails));
main_ui_->actionViewPacketBytes->setChecked(recent.byte_view_show && prefs_has_layout_pane_content(layout_pane_content_pbytes));
main_ui_->actionViewPacketDiagram->setChecked(recent.packet_diagram_show && prefs_has_layout_pane_content(layout_pane_content_pdiagram));
foreach(QAction *action, main_ui_->menuInterfaceToolbars->actions()) {
if (g_list_find_custom(recent.interface_toolbars, action->text().toUtf8(), (GCompareFunc)strcmp)) {
action->setChecked(true);
} else {
action->setChecked(false);
}
}
foreach(QAction * action, main_ui_->menuAdditionalToolbars->actions()) {
ext_toolbar_t * toolbar = VariantPointer<ext_toolbar_t>::asPtr(action->data());
bool checked = false;
if (toolbar && g_list_find_custom(recent.gui_additional_toolbars, toolbar->name, (GCompareFunc)strcmp))
checked = true;
action->setChecked(checked);
}
foreach(QAction* tda, td_actions.keys()) {
if (recent.gui_time_format == td_actions[tda]) {
tda->setChecked(true);
}
}
foreach(QAction* tpa, tp_actions.keys()) {
if (recent.gui_time_precision == tp_actions[tpa]) {
tpa->setChecked(true);
break;
}
}
main_ui_->actionViewTimeDisplaySecondsWithHoursAndMinutes->setChecked(recent.gui_seconds_format == TS_SECONDS_HOUR_MIN_SEC);
main_ui_->actionViewColorizePacketList->setChecked(recent.packet_list_colorize);
}
// Don't connect to this directly. Connect to or emit fiterAction(...) instead.
void MainWindow::queuedFilterAction(QString action_filter, FilterAction::Action action, FilterAction::ActionType type)
{
QString cur_filter, new_filter;
if (!df_combo_box_) return;
cur_filter = df_combo_box_->lineEdit()->text();
switch (type) {
case FilterAction::ActionTypePlain:
new_filter = action_filter;
break;
case FilterAction::ActionTypeAnd:
if (cur_filter.length()) {
new_filter = "(" + cur_filter + ") && (" + action_filter + ")";
}
else {
new_filter = action_filter;
}
break;
case FilterAction::ActionTypeOr:
if (cur_filter.length()) {
new_filter = "(" + cur_filter + ") || (" + action_filter + ")";
} else {
new_filter = action_filter;
}
break;
case FilterAction::ActionTypeNot:
new_filter = "!(" + action_filter + ")";
break;
case FilterAction::ActionTypeAndNot:
if (cur_filter.length()) {
new_filter = "(" + cur_filter + ") && !(" + action_filter + ")";
} else {
new_filter = "!(" + action_filter + ")";
}
break;
case FilterAction::ActionTypeOrNot:
if (cur_filter.length()) {
new_filter = "(" + cur_filter + ") || !(" + action_filter + ")";
} else {
new_filter = "!(" + action_filter + ")";
}
break;
default:
ws_assert_not_reached();
break;
}
switch (action) {
case FilterAction::ActionApply:
df_combo_box_->lineEdit()->setText(new_filter);
df_combo_box_->applyDisplayFilter();
break;
case FilterAction::ActionColorize:
colorizeWithFilter(new_filter.toUtf8());
break;
case FilterAction::ActionCopy:
wsApp->clipboard()->setText(new_filter);
break;
case FilterAction::ActionFind:
main_ui_->searchFrame->findFrameWithFilter(new_filter);
break;
case FilterAction::ActionPrepare:
df_combo_box_->lineEdit()->setText(new_filter);
df_combo_box_->lineEdit()->setFocus();
break;
case FilterAction::ActionWebLookup:
{
QString url = QString("https://www.google.com/search?q=") + new_filter;
QDesktopServices::openUrl(QUrl(url));
break;
}
default:
ws_assert_not_reached();
break;
}
}
// Capture callbacks
#ifdef HAVE_LIBPCAP
void MainWindow::captureCapturePrepared(capture_session *session) {
setTitlebarForCaptureInProgress();
setWindowIcon(wsApp->captureIcon());
/* Disable menu items that make no sense if you're currently running
a capture. */
bool handle_toolbars = (session->session_will_restart ? false : true);
setForCaptureInProgress(true, handle_toolbars, session->capture_opts->ifaces);
// set_capture_if_dialog_for_capture_in_progress(TRUE);
// /* Don't set up main window for a capture file. */
// main_set_for_capture_file(FALSE);
showCapture();
}
void MainWindow::captureCaptureUpdateStarted(capture_session *session) {
/* We've done this in "prepared" above, but it will be cleared while
switching to the next multiple file. */
setTitlebarForCaptureInProgress();
bool handle_toolbars = (session->session_will_restart ? false : true);
setForCaptureInProgress(true, handle_toolbars, session->capture_opts->ifaces);
setForCapturedPackets(true);
}
void MainWindow::captureCaptureUpdateFinished(capture_session *session) {
/* The capture isn't stopping any more - it's stopped. */
capture_stopping_ = false;
/* Update the main window as appropriate */
updateForUnsavedChanges();
/* Enable menu items that make sense if you're not currently running
a capture. */
bool handle_toolbars = (session->session_will_restart ? false : true);
setForCaptureInProgress(false, handle_toolbars);
setMenusForCaptureFile();
setWindowIcon(wsApp->normalIcon());
if (global_commandline_info.quit_after_cap) {
// Command line asked us to quit after capturing.
// Don't pop up a dialog to ask for unsaved files etc.
exit(0);
}
}
void MainWindow::captureCaptureFixedFinished(capture_session *) {
/* The capture isn't stopping any more - it's stopped. */
capture_stopping_ = false;
/* Enable menu items that make sense if you're not currently running
a capture. */
setForCaptureInProgress(false);
/* There isn't a real capture_file structure yet, so just force disabling
menu options. They will "refresh" when the capture file is reloaded to
display packets */
setMenusForCaptureFile(true);
setWindowIcon(wsApp->normalIcon());
if (global_commandline_info.quit_after_cap) {
// Command line asked us to quit after capturing.
// Don't pop up a dialog to ask for unsaved files etc.
exit(0);
}
}
void MainWindow::captureCaptureFailed(capture_session *) {
/* Capture isn't stopping any more. */
capture_stopping_ = false;
setForCaptureInProgress(false);
showWelcome();
// Reset expert information indicator
main_ui_->statusBar->captureFileClosing();
wsApp->popStatus(WiresharkApplication::FileStatus);
setWindowIcon(wsApp->normalIcon());
if (global_commandline_info.quit_after_cap) {
// Command line asked us to quit after capturing.
// Don't pop up a dialog to ask for unsaved files etc.
exit(0);
}
}
#endif // HAVE_LIBPCAP
// Callbacks from cfile.c and file.c via CaptureFile::captureFileCallback
void MainWindow::captureEventHandler(CaptureEvent ev)
{
switch (ev.captureContext()) {
case CaptureEvent::File:
switch (ev.eventType()) {
case CaptureEvent::Opened:
captureFileOpened();
break;
case CaptureEvent::Closing:
captureFileClosing();
break;
case CaptureEvent::Closed:
captureFileClosed();
break;
case CaptureEvent::Started:
captureFileReadStarted(tr("Loading"));
break;
case CaptureEvent::Finished:
captureFileReadFinished();
break;
default:
break;
}
break;
case CaptureEvent::Reload:
switch (ev.eventType()) {
case CaptureEvent::Started:
captureFileReadStarted(tr("Reloading"));
break;
case CaptureEvent::Finished:
captureFileReadFinished();
break;
default:
break;
}
break;
case CaptureEvent::Rescan:
switch (ev.eventType()) {
case CaptureEvent::Started:
setMenusForCaptureFile(true);
captureFileReadStarted(tr("Rescanning"));
break;
case CaptureEvent::Finished:
captureFileReadFinished();
break;
default:
break;
}
break;
case CaptureEvent::Retap:
switch (ev.eventType()) {
case CaptureEvent::Started:
freeze();
break;
case CaptureEvent::Finished:
thaw();
break;
case CaptureEvent::Flushed:
draw_tap_listeners(FALSE);
break;
default:
break;
}
break;
case CaptureEvent::Merge:
switch (ev.eventType()) {
case CaptureEvent::Started:
wsApp->popStatus(WiresharkApplication::FileStatus);
wsApp->pushStatus(WiresharkApplication::FileStatus, tr("Merging files."), QString());
break;
case CaptureEvent::Finished:
wsApp->popStatus(WiresharkApplication::FileStatus);
break;
default:
break;
}
break;
case CaptureEvent::Save:
switch (ev.eventType()) {
case CaptureEvent::Started:
{
QFileInfo file_info(ev.filePath());
wsApp->popStatus(WiresharkApplication::FileStatus);
wsApp->pushStatus(WiresharkApplication::FileStatus, tr("Saving %1…").arg(file_info.fileName()));
break;
}
default:
break;
}
break;
#ifdef HAVE_LIBPCAP
case CaptureEvent::Capture:
switch (ev.eventType()) {
case CaptureEvent::Prepared:
captureCapturePrepared(ev.capSession());
break;
case CaptureEvent::Stopping:
capture_stopping_ = true;
setMenusForCaptureStopping();
break;
case CaptureEvent::Failed:
captureCaptureFailed(ev.capSession());
default:
break;
}
break;
case CaptureEvent::Update:
switch (ev.eventType()) {
case CaptureEvent::Started:
captureCaptureUpdateStarted(ev.capSession());
break;
case CaptureEvent::Finished:
captureCaptureUpdateFinished(ev.capSession());
break;
default:
break;
}
break;
case CaptureEvent::Fixed:
switch (ev.eventType()) {
case CaptureEvent::Finished:
captureCaptureFixedFinished(ev.capSession());
break;
default:
break;
}
break;
#endif
}
}
void MainWindow::captureFileOpened() {
if (capture_file_.window() != this) return;
file_set_dialog_->fileOpened(capture_file_.capFile());
setMenusForFileSet(true);
emit setCaptureFile(capture_file_.capFile());
}
void MainWindow::captureFileReadStarted(const QString &action) {
// tap_param_dlg_update();
/* Set up main window for a capture file. */
// main_set_for_capture_file(TRUE);
wsApp->popStatus(WiresharkApplication::FileStatus);
QString msg = QString(tr("%1: %2")).arg(action).arg(capture_file_.fileName());
QString msgtip = QString();
wsApp->pushStatus(WiresharkApplication::FileStatus, msg, msgtip);
showCapture();
main_ui_->actionAnalyzeReloadLuaPlugins->setEnabled(false);
main_ui_->wirelessTimelineWidget->captureFileReadStarted(capture_file_.capFile());
}
void MainWindow::captureFileReadFinished() {
if (!capture_file_.capFile()->is_tempfile && capture_file_.capFile()->filename) {
/* Add this filename to the list of recent files in the "Recent Files" submenu */
add_menu_recent_capture_file(capture_file_.capFile()->filename);
/* Remember folder for next Open dialog and save it in recent */
wsApp->setLastOpenDirFromFilename(capture_file_.capFile()->filename);
}
/* Update the appropriate parts of the main window. */
updateForUnsavedChanges();
/* enable wireless timeline if capture allows it */
main_ui_->wirelessTimelineWidget->captureFileReadFinished();
/* Enable menu items that make sense if you have some captured packets. */
setForCapturedPackets(true);
main_ui_->statusBar->setFileName(capture_file_);
main_ui_->actionAnalyzeReloadLuaPlugins->setEnabled(true);
packet_list_->captureFileReadFinished();
emit setDissectedCaptureFile(capture_file_.capFile());
}
void MainWindow::captureFileClosing() {
setMenusForCaptureFile(true);
setForCapturedPackets(false);
setForCaptureInProgress(false);
// Reset expert information indicator
main_ui_->statusBar->captureFileClosing();
main_ui_->searchFrame->animatedHide();
main_ui_->goToFrame->animatedHide();
// gtk_widget_show(expert_info_none);
emit setCaptureFile(NULL);
emit setDissectedCaptureFile(NULL);
}
void MainWindow::captureFileClosed() {
packets_bar_update();
file_set_dialog_->fileClosed();
setMenusForFileSet(false);
setWindowModified(false);
// Reset expert information indicator
main_ui_->statusBar->captureFileClosing();
wsApp->popStatus(WiresharkApplication::FileStatus);
setWSWindowTitle();
setWindowIcon(wsApp->normalIcon());
setMenusForSelectedPacket();
setMenusForSelectedTreeRow();
#ifdef HAVE_LIBPCAP
if (!global_capture_opts.multi_files_on)
showWelcome();
#endif
}
//
// Private slots
//
// ui/gtk/capture_dlg.c:start_capture_confirmed
void MainWindow::startCapture() {
#ifdef HAVE_LIBPCAP
interface_options *interface_opts;
guint i;
/* did the user ever select a capture interface before? */
if (global_capture_opts.num_selected == 0) {
QString msg = QString(tr("No interface selected."));
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, msg);
main_ui_->actionCaptureStart->setChecked(false);
return;
}
// Ideally we should have disabled the start capture
// toolbar buttons and menu items. This may not be the
// case, e.g. with QtMacExtras.
if (!capture_filter_valid_) {
QString msg = QString(tr("Invalid capture filter."));
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, msg);
main_ui_->actionCaptureStart->setChecked(false);
return;
}
showCapture();
/* XXX - we might need to init other pref data as well... */
/* XXX - can this ever happen? */
if (cap_session_.state != CAPTURE_STOPPED)
return;
/* close the currently loaded capture file */
cf_close((capture_file *)cap_session_.cf);
/* Copy the selected interfaces to the set of interfaces to use for
this capture. */
collect_ifaces(&global_capture_opts);
CaptureFile::globalCapFile()->window = this;
info_data_.ui.ui = this;
if (capture_start(&global_capture_opts, NULL, &cap_session_, &info_data_,
main_window_update)) {
capture_options *capture_opts = cap_session_.capture_opts;
GString *interface_names;
/* Add "interface name<live capture in progress>" on main status bar */
interface_names = get_iface_list_string(capture_opts, 0);
if (strlen(interface_names->str) > 0) {
g_string_append(interface_names, ":");
}
g_string_append(interface_names, " ");
wsApp->popStatus(WiresharkApplication::FileStatus);
QString msg = QString("%1<live capture in progress>").arg(interface_names->str);
QString msgtip = QString("to file: ");
if (capture_opts->save_file)
msgtip += capture_opts->save_file;
wsApp->pushStatus(WiresharkApplication::FileStatus, msg, msgtip);
g_string_free(interface_names, TRUE);
/* The capture succeeded, which means the capture filter syntax is
valid; add this capture filter to the recent capture filter list. */
QByteArray filter_ba;
for (i = 0; i < global_capture_opts.ifaces->len; i++) {
interface_opts = &g_array_index(global_capture_opts.ifaces, interface_options, i);
if (interface_opts->cfilter) {
recent_add_cfilter(interface_opts->name, interface_opts->cfilter);
if (filter_ba.isEmpty()) {
filter_ba = interface_opts->cfilter;
} else {
/* Not the first selected interface; is its capture filter
the same as the one the other interfaces we've looked
at have? */
if (strcmp(interface_opts->cfilter, filter_ba.constData()) != 0) {
/* No, so not all selected interfaces have the same capture
filter. */
filter_ba.clear();
}
}
}
}
if (!filter_ba.isEmpty()) {
recent_add_cfilter(NULL, filter_ba.constData());
}
} else {
CaptureFile::globalCapFile()->window = NULL;
}
#endif // HAVE_LIBPCAP
}
// Copied from ui/gtk/gui_utils.c
void MainWindow::pipeTimeout() {
#ifdef _WIN32
HANDLE handle;
DWORD avail = 0;
gboolean result, result1;
DWORD childstatus;
gint iterations = 0;
/* try to read data from the pipe only 5 times, to avoid blocking */
while (iterations < 5) {
/* Oddly enough although Named pipes don't work on win9x,
PeekNamedPipe does !!! */
handle = (HANDLE)_get_osfhandle(pipe_source_);
result = PeekNamedPipe(handle, NULL, 0, NULL, &avail, NULL);
/* Get the child process exit status */
result1 = GetExitCodeProcess((HANDLE)*(pipe_child_process_),
&childstatus);
/* If the Peek returned an error, or there are bytes to be read
or the childwatcher thread has terminated then call the normal
callback */
if (!result || avail > 0 || childstatus != STILL_ACTIVE) {
/* And call the real handler */
if (!pipe_input_cb_(pipe_source_, pipe_user_data_)) {
ws_log(LOG_DOMAIN_MAIN, LOG_LEVEL_DEBUG, "pipe_timer_cb: input pipe closed, iterations: %u", iterations);
/* pipe closed, return false so that the old timer is not run again */
delete pipe_timer_;
return;
}
} else {
/* No data, stop now */
break;
}
iterations++;
}
#endif // _WIN32
}
void MainWindow::pipeActivated(int source) {
Q_UNUSED(source)
#ifndef _WIN32
ws_assert(source == pipe_source_);
pipe_notifier_->setEnabled(false);
if (pipe_input_cb_(pipe_source_, pipe_user_data_)) {
pipe_notifier_->setEnabled(true);
}
else {
delete pipe_notifier_;
}
#endif // _WIN32
}
void MainWindow::pipeNotifierDestroyed()
{
/* Pop the "<live capture in progress>" message off the status bar. */
main_ui_->statusBar->setFileName(capture_file_);
#ifdef _WIN32
pipe_timer_ = NULL;
#else
pipe_notifier_ = NULL;
#endif // _WIN32
}
void MainWindow::stopCapture() {
//#ifdef HAVE_AIRPCAP
// if (airpcap_if_active)
// airpcap_set_toolbar_stop_capture(airpcap_if_active);
//#endif
#ifdef HAVE_LIBPCAP
capture_stop(&cap_session_);
#endif // HAVE_LIBPCAP
}
// Keep focus rects from showing through the welcome screen. Primarily for
// macOS.
void MainWindow::mainStackChanged(int)
{
for (int i = 0; i < main_ui_->mainStack->count(); i++) {
main_ui_->mainStack->widget(i)->setEnabled(i == main_ui_->mainStack->currentIndex());
}
}
// XXX - Copied from ui/gtk/menus.c
/**
* Add the capture filename (with an absolute path) to the "Recent Files" menu.
*/
// XXX - We should probably create a RecentFile class.
void MainWindow::updateRecentCaptures() {
QAction *ra;
QMenu *recentMenu = main_ui_->menuOpenRecentCaptureFile;
QString action_cf_name;
if (!recentMenu) {
return;
}
recentMenu->clear();
#if 0
#if defined(QT_WINEXTRAS_LIB)
QWinJumpList recent_jl(this);
QWinJumpListCategory *recent_jlc = recent_jl.recent();
if (recent_jlc) {
recent_jlc->clear();
recent_jlc->setVisible(true);
}
#endif
#endif
#if defined(Q_OS_MAC)
if (!dock_menu_) {
dock_menu_ = new QMenu();
dock_menu_->setAsDockMenu();
}
dock_menu_->clear();
#endif
/* Iterate through the actions in menuOpenRecentCaptureFile,
* removing special items, a maybe duplicate entry and every item above count_max */
int shortcut = Qt::Key_0;
foreach(recent_item_status *ri, wsApp->recentItems()) {
// Add the new item
ra = new QAction(recentMenu);
ra->setData(ri->filename);
// XXX - Needs get_recent_item_status or equivalent
ra->setEnabled(ri->accessible);
recentMenu->insertAction(NULL, ra);
action_cf_name = ra->data().toString();
if (shortcut <= Qt::Key_9) {
ra->setShortcut(Qt::META | shortcut);
shortcut++;
}
ra->setText(action_cf_name);
connect(ra, SIGNAL(triggered()), this, SLOT(recentActionTriggered()));
/* This is slow, at least on my VM here. The added links also open Wireshark
* in a new window. It might make more sense to add a recent item when we
* open a capture file. */
#if 0
#if defined(QT_WINEXTRAS_LIB)
if (recent_jlc) {
QFileInfo fi(ri->filename);
QWinJumpListItem *jli = recent_jlc->addLink(
fi.fileName(),
QApplication::applicationFilePath(),
QStringList() << "-r" << ri->filename
);
// XXX set icon
jli->setWorkingDirectory(QDir::toNativeSeparators(QApplication::applicationDirPath()));
}
#endif
#endif
#if defined(Q_OS_MAC)
QAction *rda = new QAction(dock_menu_);
QFileInfo fi(ri->filename);
rda->setText(fi.fileName());
dock_menu_->insertAction(NULL, rda);
connect(rda, SIGNAL(triggered()), ra, SLOT(trigger()));
#endif
}
if (recentMenu->actions().count() > 0) {
// Separator + "Clear"
// XXX - Do we really need this?
ra = new QAction(recentMenu);
ra->setSeparator(true);
recentMenu->insertAction(NULL, ra);
ra = new QAction(recentMenu);
ra->setText(tr("Clear Menu"));
recentMenu->insertAction(NULL, ra);
connect(ra, SIGNAL(triggered()), wsApp, SLOT(clearRecentCaptures()));
} else {
if (main_ui_->actionDummyNoFilesFound) {
recentMenu->addAction(main_ui_->actionDummyNoFilesFound);
}
}
}
void MainWindow::recentActionTriggered() {
QAction *ra = qobject_cast<QAction*>(sender());
if (ra) {
QString cfPath = ra->data().toString();
openCaptureFile(cfPath);
}
}
QString MainWindow::commentToMenuText(QString text, int max_len)
{
text = text.trimmed().replace(QRegExp("(\\r?\\n|\\r\\n?)+"), " ");
if (text.size() > 0) {
if (text.size() > max_len) {
text.truncate(max_len);
text += "";
}
}
else {
text = tr("(empty comment)", "placeholder for empty comment");
}
return text;
}
void MainWindow::setEditCommentsMenu()
{
main_ui_->menuPacketComment->clear();
main_ui_->menuPacketComment->addAction(tr("Add New Comment…"), this, SLOT(actionAddPacketComment()), QKeySequence(Qt::CTRL + Qt::ALT + Qt::Key_C));
if (selectedRows().count() == 1) {
const int thisRow = selectedRows().first();
frame_data * current_frame = frameDataForRow(thisRow);
wtap_block_t pkt_block = cf_get_packet_block(capture_file_.capFile(), current_frame);
guint nComments = wtap_block_count_option(pkt_block, OPT_COMMENT);
if (nComments > 0) {
QAction *aPtr;
main_ui_->menuPacketComment->addSeparator();
for (guint i = 0; i < nComments; i++) {
QString comment = packet_list_->getPacketComment(i);
comment = this->commentToMenuText(comment);
aPtr = main_ui_->menuPacketComment->addAction(tr("Edit \"%1\"", "edit packet comment").arg(comment),
this, SLOT(actionEditPacketComment()));
aPtr->setData(i);
}
main_ui_->menuPacketComment->addSeparator();
for (guint i = 0; i < nComments; i++) {
QString comment = packet_list_->getPacketComment(i);
comment = this->commentToMenuText(comment);
aPtr = main_ui_->menuPacketComment->addAction(tr("Delete \"%1\"", "delete packet comment").arg(comment),
this, SLOT(actionDeletePacketComment()));
aPtr->setData(i);
}
main_ui_->menuPacketComment->addSeparator();
main_ui_->menuPacketComment->addAction(tr("Delete packet comments"), this, SLOT(actionDeleteCommentsFromPackets()));
}
wtap_block_unref(pkt_block);
}
if (selectedRows().count() > 1) {
main_ui_->menuPacketComment->addSeparator();
main_ui_->menuPacketComment->addAction(tr("Delete comments from %n packet(s)", nullptr, selectedRows().count()), this, SLOT(actionDeleteCommentsFromPackets()));
}
}
void MainWindow::setMenusForSelectedPacket()
{
gboolean is_ip = FALSE, is_tcp = FALSE, is_udp = FALSE, is_dccp = FALSE, is_sctp = FALSE, is_tls = FALSE, is_rtp = FALSE, is_lte_rlc = FALSE,
is_http = FALSE, is_http2 = FALSE, is_quic = FALSE, is_sip = FALSE, is_exported_pdu = FALSE;
/* Making the menu context-sensitive allows for easier selection of the
desired item and has the added benefit, with large captures, of
avoiding needless looping through huge lists for marked, ignored,
or time-referenced packets. */
/* We have one or more items in the packet list */
bool have_frames = false;
/* A frame is selected */
bool frame_selected = false;
bool multi_selection = false;
/* A visible packet comes after this one in the selection history */
bool next_selection_history = false;
/* A visible packet comes before this one in the selection history */
bool previous_selection_history = false;
/* We have marked frames. (XXX - why check frame_selected?) */
bool have_marked = false;
/* We have a marked frame other than the current frame (i.e.,
we have at least one marked frame, and either there's more
than one marked frame or the current frame isn't marked). */
bool another_is_marked = false;
/* One or more frames are hidden by a display filter */
bool have_filtered = false;
/* One or more frames have been ignored */
bool have_ignored = false;
bool have_time_ref = false;
/* We have a time reference frame other than the current frame (i.e.,
we have at least one time reference frame, and either there's more
than one time reference frame or the current frame isn't a
time reference frame). (XXX - why check frame_selected?) */
bool another_is_time_ref = false;
QList<QAction *> cc_actions = QList<QAction *>()
<< main_ui_->actionViewColorizeConversation1 << main_ui_->actionViewColorizeConversation2
<< main_ui_->actionViewColorizeConversation3 << main_ui_->actionViewColorizeConversation4
<< main_ui_->actionViewColorizeConversation5 << main_ui_->actionViewColorizeConversation6
<< main_ui_->actionViewColorizeConversation7 << main_ui_->actionViewColorizeConversation8
<< main_ui_->actionViewColorizeConversation9 << main_ui_->actionViewColorizeConversation10;
if (capture_file_.capFile()) {
QList<int> rows = selectedRows();
frame_data * current_frame = 0;
if (rows.count() > 0)
current_frame = frameDataForRow(rows.at(0));
frame_selected = rows.count() == 1;
if (packet_list_->multiSelectActive())
{
frame_selected = false;
multi_selection = true;
}
next_selection_history = packet_list_->haveNextHistory();
previous_selection_history = packet_list_->havePreviousHistory();
have_frames = capture_file_.capFile()->count > 0;
have_marked = capture_file_.capFile()->marked_count > 0;
another_is_marked = have_marked && rows.count() <= 1 &&
!(capture_file_.capFile()->marked_count == 1 && frame_selected && current_frame->marked);
have_filtered = capture_file_.capFile()->displayed_count > 0 && capture_file_.capFile()->displayed_count != capture_file_.capFile()->count;
have_ignored = capture_file_.capFile()->ignored_count > 0;
have_time_ref = capture_file_.capFile()->ref_time_count > 0;
another_is_time_ref = have_time_ref && rows.count() <= 1 &&
!(capture_file_.capFile()->ref_time_count == 1 && frame_selected && current_frame->ref_time);
if (capture_file_.capFile()->edt && ! multi_selection)
{
proto_get_frame_protocols(capture_file_.capFile()->edt->pi.layers,
&is_ip, &is_tcp, &is_udp, &is_sctp,
&is_tls, &is_rtp, &is_lte_rlc);
is_dccp = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "dccp");
is_http = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "http");
is_http2 = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "http2");
/* TODO: to follow a QUIC stream we need a *decrypted* QUIC connection, i.e. checking for "quic" in the protocol stack is not enough */
is_quic = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "quic");
is_sip = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "sip");
is_exported_pdu = proto_is_frame_protocol(capture_file_.capFile()->edt->pi.layers, "exported_pdu");
/* For Exported PDU there is a tag inserting IP addresses into the SRC and DST columns */
if (is_exported_pdu &&
(capture_file_.capFile()->edt->pi.net_src.type == AT_IPv4 || capture_file_.capFile()->edt->pi.net_src.type == AT_IPv6) &&
(capture_file_.capFile()->edt->pi.net_dst.type == AT_IPv4 || capture_file_.capFile()->edt->pi.net_dst.type == AT_IPv6)) {
is_ip = TRUE;
}
}
}
main_ui_->actionEditMarkPacket->setText(tr("&Mark/Unmark Packet(s)", "", selectedRows().count()));
main_ui_->actionEditIgnorePacket->setText(tr("&Ignore/Unignore Packet(s)", "", selectedRows().count()));
main_ui_->actionCopyListAsText->setEnabled(selectedRows().count() > 0);
main_ui_->actionCopyListAsCSV->setEnabled(selectedRows().count() > 0);
main_ui_->actionCopyListAsYAML->setEnabled(selectedRows().count() > 0);
main_ui_->actionEditMarkPacket->setEnabled(frame_selected || multi_selection);
main_ui_->actionEditMarkAllDisplayed->setEnabled(have_frames);
/* Unlike un-ignore, do not allow unmark of all frames when no frames are displayed */
main_ui_->actionEditUnmarkAllDisplayed->setEnabled(have_marked);
main_ui_->actionEditNextMark->setEnabled(another_is_marked);
main_ui_->actionEditPreviousMark->setEnabled(another_is_marked);
GArray * linkTypes = Q_NULLPTR;
if (capture_file_.capFile() && capture_file_.capFile()->linktypes)
linkTypes = capture_file_.capFile()->linktypes;
bool enableEditComments = linkTypes && wtap_dump_can_write(capture_file_.capFile()->linktypes, WTAP_COMMENT_PER_PACKET);
main_ui_->menuPacketComment->setEnabled(enableEditComments && selectedRows().count() > 0);
main_ui_->actionDeleteAllPacketComments->setEnabled(enableEditComments);
main_ui_->actionEditIgnorePacket->setEnabled(frame_selected || multi_selection);
main_ui_->actionEditIgnoreAllDisplayed->setEnabled(have_filtered);
/* Allow un-ignore of all frames even with no frames currently displayed */
main_ui_->actionEditUnignoreAllDisplayed->setEnabled(have_ignored);
main_ui_->actionEditSetTimeReference->setEnabled(frame_selected);
main_ui_->actionEditUnsetAllTimeReferences->setEnabled(have_time_ref);
main_ui_->actionEditNextTimeReference->setEnabled(another_is_time_ref);
main_ui_->actionEditPreviousTimeReference->setEnabled(another_is_time_ref);
main_ui_->actionEditTimeShift->setEnabled(have_frames);
main_ui_->actionGoGoToLinkedPacket->setEnabled(false);
main_ui_->actionGoNextHistoryPacket->setEnabled(next_selection_history);
main_ui_->actionGoPreviousHistoryPacket->setEnabled(previous_selection_history);
main_ui_->actionAnalyzeFollowTCPStream->setEnabled(is_tcp);
main_ui_->actionAnalyzeFollowUDPStream->setEnabled(is_udp);
main_ui_->actionAnalyzeFollowDCCPStream->setEnabled(is_dccp);
main_ui_->actionAnalyzeFollowTLSStream->setEnabled(is_tls && !is_quic);
main_ui_->actionAnalyzeFollowHTTPStream->setEnabled(is_http);
main_ui_->actionAnalyzeFollowHTTP2Stream->setEnabled(is_http2);
main_ui_->actionAnalyzeFollowQUICStream->setEnabled(is_quic);
main_ui_->actionAnalyzeFollowSIPCall->setEnabled(is_sip);
foreach(QAction *cc_action, cc_actions) {
cc_action->setEnabled(frame_selected);
}
main_ui_->actionViewColorizeNewColoringRule->setEnabled(frame_selected);
main_ui_->actionViewColorizeResetColorization->setEnabled(tmp_color_filters_used());
main_ui_->actionViewShowPacketInNewWindow->setEnabled(frame_selected);
main_ui_->actionViewEditResolvedName->setEnabled(frame_selected && is_ip);
emit packetInfoChanged(capture_file_.packetInfo());
// set_menu_sensitivity(ui_manager_main_menubar, "/Menubar/ViewMenu/NameResolution/ResolveName",
// frame_selected && (gbl_resolv_flags.mac_name || gbl_resolv_flags.network_name ||
// gbl_resolv_flags.transport_name));
main_ui_->actionToolsFirewallAclRules->setEnabled(frame_selected);
main_ui_->actionStatisticsTcpStreamRoundTripTime->setEnabled(is_tcp);
main_ui_->actionStatisticsTcpStreamStevens->setEnabled(is_tcp);
main_ui_->actionStatisticsTcpStreamTcptrace->setEnabled(is_tcp);
main_ui_->actionStatisticsTcpStreamThroughput->setEnabled(is_tcp);
main_ui_->actionStatisticsTcpStreamWindowScaling->setEnabled(is_tcp);
main_ui_->actionSCTPAnalyseThisAssociation->setEnabled(is_sctp);
main_ui_->actionSCTPShowAllAssociations->setEnabled(is_sctp);
main_ui_->actionSCTPFilterThisAssociation->setEnabled(is_sctp);
main_ui_->actionTelephonyRtpStreamAnalysis->setEnabled(is_rtp);
main_ui_->actionTelephonyRtpPlayer->setEnabled(is_rtp);
main_ui_->actionTelephonyLteRlcGraph->setEnabled(is_lte_rlc);
}
void MainWindow::setMenusForSelectedTreeRow(FieldInformation *finfo) {
bool can_match_selected = false;
bool is_framenum = false;
bool have_field_info = false;
bool have_subtree = false;
bool can_open_url = false;
bool have_packet_bytes = false;
QByteArray field_filter;
int field_id = -1;
field_info * fi = 0;
if (finfo)
fi = finfo->fieldInfo();
if (capture_file_.capFile()) {
capture_file_.capFile()->finfo_selected = fi;
if (fi && fi->tree_type != -1) {
have_subtree = true;
}
if (fi && fi->ds_tvb) {
have_packet_bytes = true;
}
}
if (capture_file_.capFile() != NULL && fi != NULL) {
header_field_info *hfinfo = fi->hfinfo;
int linked_frame = -1;
have_field_info = true;
can_match_selected = proto_can_match_selected(capture_file_.capFile()->finfo_selected, capture_file_.capFile()->edt);
if (hfinfo && hfinfo->type == FT_FRAMENUM) {
is_framenum = true;
linked_frame = fvalue_get_uinteger(&fi->value);
}
char *tmp_field = proto_construct_match_selected_string(fi, capture_file_.capFile()->edt);
field_filter = tmp_field;
wmem_free(NULL, tmp_field);
field_id = fi->hfinfo->id;
/* if the selected field isn't a protocol, get its parent */
if (!proto_registrar_is_protocol(field_id)) {
field_id = proto_registrar_get_parent(fi->hfinfo->id);
}
if (field_id >= 0) {
can_open_url = true;
main_ui_->actionContextWikiProtocolPage->setData(field_id);
main_ui_->actionContextFilterFieldReference->setData(field_id);
} else {
main_ui_->actionContextWikiProtocolPage->setData(QVariant());
main_ui_->actionContextFilterFieldReference->setData(QVariant());
}
if (linked_frame > 0) {
main_ui_->actionGoGoToLinkedPacket->setData(linked_frame);
} else {
main_ui_->actionGoGoToLinkedPacket->setData(QVariant());
}
}
// Always enable / disable the following items.
main_ui_->actionFileExportPacketBytes->setEnabled(have_field_info);
main_ui_->actionCopyAllVisibleItems->setEnabled(capture_file_.capFile() != NULL && ! packet_list_->multiSelectActive());
main_ui_->actionCopyAllVisibleSelectedTreeItems->setEnabled(can_match_selected);
main_ui_->actionEditCopyDescription->setEnabled(can_match_selected);
main_ui_->actionEditCopyFieldName->setEnabled(can_match_selected);
main_ui_->actionEditCopyValue->setEnabled(can_match_selected);
main_ui_->actionEditCopyAsFilter->setEnabled(can_match_selected);
main_ui_->actionAnalyzeShowPacketBytes->setEnabled(have_packet_bytes);
main_ui_->actionFileExportPacketBytes->setEnabled(have_packet_bytes);
main_ui_->actionViewExpandSubtrees->setEnabled(have_subtree);
main_ui_->actionViewCollapseSubtrees->setEnabled(have_subtree);
main_ui_->actionGoGoToLinkedPacket->setEnabled(is_framenum);
main_ui_->actionAnalyzeCreateAColumn->setEnabled(can_match_selected);
main_ui_->actionContextShowLinkedPacketInNewWindow->setEnabled(is_framenum);
main_ui_->actionContextWikiProtocolPage->setEnabled(can_open_url);
main_ui_->actionContextFilterFieldReference->setEnabled(can_open_url);
// Only enable / disable the following items if we have focus so that we
// don't clobber anything we may have set in setMenusForSelectedPacket.
if (!proto_tree_ || !proto_tree_->hasFocus()) return;
emit packetInfoChanged(capture_file_.packetInfo());
emit fieldFilterChanged(field_filter);
// set_menu_sensitivity(ui_manager_tree_view_menu, "/TreeViewPopup/ResolveName",
// frame_selected && (gbl_resolv_flags.mac_name || gbl_resolv_flags.network_name ||
// gbl_resolv_flags.transport_name));
}
void MainWindow::interfaceSelectionChanged()
{
#ifdef HAVE_LIBPCAP
// XXX This doesn't disable the toolbar button when using
// QtMacExtras.
if (global_capture_opts.num_selected > 0 && capture_filter_valid_) {
main_ui_->actionCaptureStart->setEnabled(true);
} else {
main_ui_->actionCaptureStart->setEnabled(false);
}
#endif // HAVE_LIBPCAP
}
void MainWindow::captureFilterSyntaxChanged(bool valid)
{
capture_filter_valid_ = valid;
interfaceSelectionChanged();
}
void MainWindow::startInterfaceCapture(bool valid, const QString capture_filter)
{
capture_filter_valid_ = valid;
welcome_page_->setCaptureFilter(capture_filter);
QString before_what(tr(" before starting a new capture"));
if (testCaptureFileClose(before_what)) {
// The interface tree will update the selected interfaces via its timer
// so no need to do anything here.
startCapture();
}
}
void MainWindow::applyGlobalCommandLineOptions()
{
if (global_dissect_options.time_format != TS_NOT_SET) {
foreach(QAction* tda, td_actions.keys()) {
if (global_dissect_options.time_format == td_actions[tda]) {
tda->setChecked(true);
recent.gui_time_format = global_dissect_options.time_format;
timestamp_set_type(global_dissect_options.time_format);
break;
}
}
}
if (global_commandline_info.full_screen) {
this->showFullScreen();
}
}
void MainWindow::redissectPackets()
{
if (capture_file_.capFile()) {
cf_redissect_packets(capture_file_.capFile());
main_ui_->statusBar->expertUpdate();
}
proto_free_deregistered_fields();
}
void MainWindow::checkDisplayFilter()
{
if (!df_combo_box_->checkDisplayFilter()) {
g_free(CaptureFile::globalCapFile()->dfilter);
CaptureFile::globalCapFile()->dfilter = NULL;
}
}
void MainWindow::fieldsChanged()
{
gchar *err_msg = NULL;
if (!color_filters_reload(&err_msg, color_filter_add_cb)) {
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_msg);
g_free(err_msg);
}
tap_listeners_dfilter_recompile();
emit checkDisplayFilter();
if (have_custom_cols(&CaptureFile::globalCapFile()->cinfo)) {
// Recreate packet list columns according to new/changed/deleted fields
packet_list_->fieldsChanged(CaptureFile::globalCapFile());
}
emit reloadFields();
}
void MainWindow::reloadLuaPlugins()
{
#ifdef HAVE_LUA
if (wsApp->isReloadingLua())
return;
gboolean uses_lua_filehandler = FALSE;
if (capture_file_.capFile()) {
// Check if the current capture file is opened with a Lua FileHandler
capture_file *cf = capture_file_.capFile();
uses_lua_filehandler = wtap_uses_lua_filehandler(cf->provider.wth);
if (uses_lua_filehandler && cf->unsaved_changes) {
// Prompt to save the file before reloading, in case the FileHandler has changed
QString before_what(tr(" before reloading Lua plugins"));
if (!testCaptureFileClose(before_what, Reload)) {
return;
}
}
}
wsApp->setReloadingLua(true);
wslua_reload_plugins(NULL, NULL);
funnel_statistics_reload_menus();
reloadDynamicMenus();
closePacketDialogs();
// Preferences may have been deleted so close all widgets using prefs
main_ui_->preferenceEditorFrame->animatedHide();
wsApp->readConfigurationFiles(true);
commandline_options_reapply();
prefs_apply_all();
fieldsChanged();
if (uses_lua_filehandler) {
// Reload the file in case the FileHandler has changed
if (cf_reload(capture_file_.capFile()) != CF_OK) {
cf_close(capture_file_.capFile());
}
proto_free_deregistered_fields();
} else {
redissectPackets();
}
wsApp->setReloadingLua(false);
SimpleDialog::displayQueuedMessages();
#endif
}
void MainWindow::showAccordionFrame(AccordionFrame *show_frame, bool toggle)
{
QList<AccordionFrame *>frame_list = QList<AccordionFrame *>()
<< main_ui_->goToFrame << main_ui_->searchFrame
<< main_ui_->addressEditorFrame << main_ui_->columnEditorFrame
<< main_ui_->preferenceEditorFrame << main_ui_->filterExpressionFrame;
frame_list.removeAll(show_frame);
foreach(AccordionFrame *af, frame_list) af->animatedHide();
if (toggle) {
if (show_frame->isVisible()) {
show_frame->animatedHide();
return;
}
}
show_frame->animatedShow();
}
void MainWindow::showColumnEditor(int column)
{
previous_focus_ = wsApp->focusWidget();
connect(previous_focus_, SIGNAL(destroyed()), this, SLOT(resetPreviousFocus()));
main_ui_->columnEditorFrame->editColumn(column);
showAccordionFrame(main_ui_->columnEditorFrame);
}
void MainWindow::showPreferenceEditor()
{
showAccordionFrame(main_ui_->preferenceEditorFrame);
}
void MainWindow::initViewColorizeMenu()
{
QList<QAction *> cc_actions = QList<QAction *>()
<< main_ui_->actionViewColorizeConversation1 << main_ui_->actionViewColorizeConversation2
<< main_ui_->actionViewColorizeConversation3 << main_ui_->actionViewColorizeConversation4
<< main_ui_->actionViewColorizeConversation5 << main_ui_->actionViewColorizeConversation6
<< main_ui_->actionViewColorizeConversation7 << main_ui_->actionViewColorizeConversation8
<< main_ui_->actionViewColorizeConversation9 << main_ui_->actionViewColorizeConversation10;
guint8 color_num = 1;
foreach(QAction *cc_action, cc_actions) {
cc_action->setData(color_num);
connect(cc_action, SIGNAL(triggered()), this, SLOT(colorizeConversation()));
const color_filter_t *colorf = color_filters_tmp_color(color_num);
if (colorf) {
QColor bg = ColorUtils::fromColorT(colorf->bg_color);
QColor fg = ColorUtils::fromColorT(colorf->fg_color);
cc_action->setIcon(StockIcon::colorIcon(bg.rgb(), fg.rgb(), QString::number(color_num)));
}
color_num++;
}
#ifdef Q_OS_MAC
// Spotlight uses Cmd+Space
main_ui_->actionViewColorizeResetColorization->setShortcut(QKeySequence("Meta+Space"));
#endif
}
void MainWindow::addStatsPluginsToMenu() {
GList *cfg_list = stats_tree_get_cfg_list();
GList *iter = g_list_first(cfg_list);
QAction *stats_tree_action;
QMenu *parent_menu;
bool first_item = true;
while (iter) {
stats_tree_cfg *cfg = gxx_list_data(stats_tree_cfg *, iter);
if (cfg->plugin) {
if (first_item) {
main_ui_->menuStatistics->addSeparator();
first_item = false;
}
parent_menu = main_ui_->menuStatistics;
// gtk/main_menubar.c compresses double slashes, hence SkipEmptyParts
#if QT_VERSION >= QT_VERSION_CHECK(5, 15, 0)
QStringList cfg_name_parts = QString(cfg->name).split("/", Qt::SkipEmptyParts);
#else
QStringList cfg_name_parts = QString(cfg->name).split("/", QString::SkipEmptyParts);
#endif
if (cfg_name_parts.isEmpty()) continue;
QString stat_name = cfg_name_parts.takeLast();
if (!cfg_name_parts.isEmpty()) {
QString menu_name = cfg_name_parts.join("/");
parent_menu = findOrAddMenu(parent_menu, menu_name);
}
stats_tree_action = new QAction(stat_name, this);
stats_tree_action->setData(cfg->abbr);
parent_menu->addAction(stats_tree_action);
connect(stats_tree_action, SIGNAL(triggered()), this, SLOT(actionStatisticsPlugin_triggered()));
}
iter = gxx_list_next(iter);
}
g_list_free(cfg_list);
}
void MainWindow::setFeaturesEnabled(bool enabled)
{
main_ui_->menuBar->setEnabled(enabled);
main_ui_->mainToolBar->setEnabled(enabled);
main_ui_->displayFilterToolBar->setEnabled(enabled);
if (enabled)
{
main_ui_->statusBar->clearMessage();
#ifdef HAVE_LIBPCAP
main_ui_->actionGoAutoScroll->setChecked(auto_scroll_live);
#endif
}
else
{
main_ui_->statusBar->showMessage(tr("Please wait while Wireshark is initializing…"));
}
}
// Display Filter Toolbar
void MainWindow::on_actionDisplayFilterExpression_triggered()
{
DisplayFilterExpressionDialog *dfe_dialog = new DisplayFilterExpressionDialog(this);
connect(dfe_dialog, SIGNAL(insertDisplayFilter(QString)),
df_combo_box_->lineEdit(), SLOT(insertFilter(const QString &)));
dfe_dialog->show();
}
void MainWindow::on_actionNewDisplayFilterExpression_triggered()
{
main_ui_->filterExpressionFrame->addExpression(df_combo_box_->lineEdit()->text());
}
void MainWindow::onFilterSelected(QString filterText, bool prepare)
{
if (filterText.length() <= 0)
return;
df_combo_box_->setDisplayFilter(filterText);
// Holding down the Shift key will only prepare filter.
if (!prepare)
df_combo_box_->applyDisplayFilter();
}
void MainWindow::onFilterPreferences()
{
emit showPreferencesDialog(PrefsModel::typeToString(PrefsModel::FilterButtons));
}
void MainWindow::onFilterEdit(int uatIndex)
{
main_ui_->filterExpressionFrame->editExpression(uatIndex);
}
void MainWindow::openStatCommandDialog(const QString &menu_path, const char *arg, void *userdata)
{
QString slot = QString("statCommand%1").arg(menu_path);
QMetaObject::invokeMethod(this, slot.toLatin1().constData(), Q_ARG(const char *, arg), Q_ARG(void *, userdata));
}
void MainWindow::openTapParameterDialog(const QString cfg_str, const QString arg, void *userdata)
{
TapParameterDialog *tp_dialog = TapParameterDialog::showTapParameterStatistics(*this, capture_file_, cfg_str, arg, userdata);
if (!tp_dialog) return;
connect(tp_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
connect(tp_dialog, SIGNAL(updateFilter(QString)),
df_combo_box_->lineEdit(), SLOT(setText(QString)));
tp_dialog->show();
}
void MainWindow::openTapParameterDialog()
{
QAction *tpa = qobject_cast<QAction *>(QObject::sender());
if (!tpa) return;
const QString cfg_str = tpa->data().toString();
openTapParameterDialog(cfg_str, NULL, NULL);
}
#if defined(HAVE_SOFTWARE_UPDATE) && defined(Q_OS_WIN)
void MainWindow::softwareUpdateRequested() {
// We could call testCaptureFileClose here, but that would give us yet
// another dialog. Just try again later.
if (capture_file_.capFile() && capture_file_.capFile()->state != FILE_CLOSED) {
wsApp->rejectSoftwareUpdate();
}
}
#endif
// File Menu
void MainWindow::on_actionFileOpen_triggered()
{
openCaptureFile();
}
void MainWindow::on_actionFileMerge_triggered()
{
mergeCaptureFile();
}
void MainWindow::on_actionFileImportFromHexDump_triggered()
{
importCaptureFile();
}
void MainWindow::on_actionFileClose_triggered() {
QString before_what(tr(" before closing the file"));
if (testCaptureFileClose(before_what))
showWelcome();
}
void MainWindow::on_actionFileSave_triggered()
{
saveCaptureFile(capture_file_.capFile(), false);
}
void MainWindow::on_actionFileSaveAs_triggered()
{
saveAsCaptureFile(capture_file_.capFile());
}
void MainWindow::on_actionFileSetListFiles_triggered()
{
file_set_dialog_->show();
}
void MainWindow::on_actionFileSetNextFile_triggered()
{
fileset_entry *entry = fileset_get_next();
if (entry) {
QString new_cf_path = entry->fullname;
openCaptureFile(new_cf_path);
}
}
void MainWindow::on_actionFileSetPreviousFile_triggered()
{
fileset_entry *entry = fileset_get_previous();
if (entry) {
QString new_cf_path = entry->fullname;
openCaptureFile(new_cf_path);
}
}
void MainWindow::on_actionFileExportPackets_triggered()
{
exportSelectedPackets();
}
void MainWindow::on_actionFileExportAsPlainText_triggered()
{
exportDissections(export_type_text);
}
void MainWindow::on_actionFileExportAsCSV_triggered()
{
exportDissections(export_type_csv);
}
void MainWindow::on_actionFileExportAsCArrays_triggered()
{
exportDissections(export_type_carrays);
}
void MainWindow::on_actionFileExportAsPSML_triggered()
{
exportDissections(export_type_psml);
}
void MainWindow::on_actionFileExportAsPDML_triggered()
{
exportDissections(export_type_pdml);
}
void MainWindow::on_actionFileExportAsJSON_triggered()
{
exportDissections(export_type_json);
}
void MainWindow::on_actionFileExportPacketBytes_triggered()
{
QString file_name;
if (!capture_file_.capFile() || !capture_file_.capFile()->finfo_selected) return;
file_name = WiresharkFileDialog::getSaveFileName(this,
wsApp->windowTitleString(tr("Export Selected Packet Bytes")),
wsApp->lastOpenDir().canonicalPath(),
tr("Raw data (*.bin *.dat *.raw);;All Files (" ALL_FILES_WILDCARD ")")
);
if (file_name.length() > 0) {
const guint8 *data_p;
data_p = tvb_get_ptr(capture_file_.capFile()->finfo_selected->ds_tvb, 0, -1) +
capture_file_.capFile()->finfo_selected->start;
write_file_binary_mode(qUtf8Printable(file_name), data_p, capture_file_.capFile()->finfo_selected->length);
/* Save the directory name for future file dialogs. */
wsApp->setLastOpenDirFromFilename(file_name);
}
}
void MainWindow::on_actionAnalyzeShowPacketBytes_triggered()
{
ShowPacketBytesDialog *spbd = new ShowPacketBytesDialog(*this, capture_file_);
spbd->addCodecs(text_codec_map_);
spbd->show();
}
void MainWindow::on_actionFileExportPDU_triggered()
{
ExportPDUDialog *exportpdu_dialog = new ExportPDUDialog(this);
if (exportpdu_dialog->isMinimized() == true)
{
exportpdu_dialog->showNormal();
}
else
{
exportpdu_dialog->show();
}
exportpdu_dialog->raise();
exportpdu_dialog->activateWindow();
}
void MainWindow::on_actionFileExportTLSSessionKeys_triggered()
{
QString file_name;
QString save_title;
int keylist_len;
keylist_len = ssl_session_key_count();
/* don't show up the dialog, if no data has to be saved */
if (keylist_len < 1) {
/* shouldn't happen as the menu item should have been greyed out */
QMessageBox::warning(
this,
tr("No Keys"),
tr("There are no TLS Session Keys to save."),
QMessageBox::Ok
);
return;
}
save_title.append(wsApp->windowTitleString(tr("Export TLS Session Keys (%Ln key(s))", "", keylist_len)));
file_name = WiresharkFileDialog::getSaveFileName(this,
save_title,
wsApp->lastOpenDir().canonicalPath(),
tr("TLS Session Keys (*.keys *.txt);;All Files (" ALL_FILES_WILDCARD ")")
);
if (file_name.length() > 0) {
gsize keylist_length;
gchar *keylist = ssl_export_sessions(&keylist_length);
write_file_binary_mode(qUtf8Printable(file_name), keylist, keylist_length);
/* Save the directory name for future file dialogs. */
wsApp->setLastOpenDirFromFilename(file_name);
g_free(keylist);
}
}
void MainWindow::on_actionStatisticsHpfeeds_triggered()
{
openStatisticsTreeDialog("hpfeeds");
}
void MainWindow::on_actionFilePrint_triggered()
{
capture_file *cf = capture_file_.capFile();
g_return_if_fail(cf);
QList<int> rows = packet_list_->selectedRows(true);
QStringList entries;
foreach (int row, rows)
entries << QString::number(row);
QString selRange = entries.join(",");
PrintDialog * pdlg_ = new PrintDialog(this, cf, selRange);
pdlg_->setWindowModality(Qt::ApplicationModal);
pdlg_->show();
}
// Edit Menu
// XXX This should probably be somewhere else.
void MainWindow::actionEditCopyTriggered(MainWindow::CopySelected selection_type)
{
char label_str[ITEM_LABEL_LENGTH];
QString clip;
if (!capture_file_.capFile()) return;
field_info *finfo_selected = capture_file_.capFile()->finfo_selected;
switch (selection_type) {
case CopySelectedDescription:
if (proto_tree_->selectionModel()->hasSelection()) {
QModelIndex idx = proto_tree_->selectionModel()->selectedIndexes().first();
clip = idx.data(Qt::DisplayRole).toString();
}
break;
case CopySelectedFieldName:
if (finfo_selected && finfo_selected->hfinfo->abbrev != 0) {
clip.append(finfo_selected->hfinfo->abbrev);
}
break;
case CopySelectedValue:
if (finfo_selected && capture_file_.capFile()->edt != 0) {
gchar* field_str = get_node_field_value(finfo_selected, capture_file_.capFile()->edt);
clip.append(field_str);
g_free(field_str);
}
break;
case CopyAllVisibleItems:
clip = proto_tree_->toString();
break;
case CopyAllVisibleSelectedTreeItems:
if (proto_tree_->selectionModel()->hasSelection()) {
clip = proto_tree_->toString(proto_tree_->selectionModel()->selectedIndexes().first());
}
break;
case CopyListAsText:
case CopyListAsCSV:
case CopyListAsYAML:
if (packet_list_->selectedRows().count() > 0)
{
QList<int> rows = packet_list_->selectedRows();
QStringList content;
PacketList::SummaryCopyType copyType = PacketList::CopyAsText;
if (selection_type == CopyListAsCSV)
copyType = PacketList::CopyAsCSV;
else if (selection_type == CopyListAsYAML)
copyType = PacketList::CopyAsYAML;
if ((copyType == PacketList::CopyAsText) ||
(copyType == PacketList::CopyAsCSV)) {
QString headerEntry = packet_list_->createHeaderSummaryText(copyType);
content << headerEntry;
}
foreach (int row, rows)
{
QModelIndex idx = packet_list_->model()->index(row, 0);
if (! idx.isValid())
continue;
QString entry = packet_list_->createSummaryText(idx, copyType);
content << entry;
}
if (content.count() > 0) {
clip = content.join("\n");
//
// Each YAML item ends with a newline, so the string
// ends with a newline already if it's CopyListAsYAML.
// If we add a newline, there'd be an extra blank
// line.
//
// Otherwise, we've used newlines as separators, not
// terminators, so there's no final newline. Add it.
//
if (selection_type != CopyListAsYAML)
clip += "\n";
}
}
break;
}
if (clip.length() == 0) {
/* If no representation then... Try to read the value */
proto_item_fill_label(capture_file_.capFile()->finfo_selected, label_str);
clip.append(label_str);
}
if (clip.length()) {
wsApp->clipboard()->setText(clip);
} else {
QString err = tr("Couldn't copy text. Try another item.");
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, err);
}
}
void MainWindow::on_actionCopyAllVisibleItems_triggered()
{
actionEditCopyTriggered(CopyAllVisibleItems);
}
void MainWindow::on_actionCopyListAsText_triggered()
{
actionEditCopyTriggered(CopyListAsText);
}
void MainWindow::on_actionCopyListAsCSV_triggered()
{
actionEditCopyTriggered(CopyListAsCSV);
}
void MainWindow::on_actionCopyListAsYAML_triggered()
{
actionEditCopyTriggered(CopyListAsYAML);
}
void MainWindow::on_actionCopyAllVisibleSelectedTreeItems_triggered()
{
actionEditCopyTriggered(CopyAllVisibleSelectedTreeItems);
}
void MainWindow::on_actionEditCopyDescription_triggered()
{
actionEditCopyTriggered(CopySelectedDescription);
}
void MainWindow::on_actionEditCopyFieldName_triggered()
{
actionEditCopyTriggered(CopySelectedFieldName);
}
void MainWindow::on_actionEditCopyValue_triggered()
{
actionEditCopyTriggered(CopySelectedValue);
}
void MainWindow::on_actionEditCopyAsFilter_triggered()
{
matchFieldFilter(FilterAction::ActionCopy, FilterAction::ActionTypePlain);
}
void MainWindow::on_actionEditFindPacket_triggered()
{
if (! packet_list_->model() || packet_list_->model()->rowCount() < 1) {
return;
}
previous_focus_ = wsApp->focusWidget();
connect(previous_focus_, SIGNAL(destroyed()), this, SLOT(resetPreviousFocus()));
if (!main_ui_->searchFrame->isVisible()) {
showAccordionFrame(main_ui_->searchFrame, true);
} else {
main_ui_->searchFrame->animatedHide();
}
main_ui_->searchFrame->setFocus();
}
void MainWindow::on_actionEditFindNext_triggered()
{
main_ui_->searchFrame->findNext();
}
void MainWindow::on_actionEditFindPrevious_triggered()
{
main_ui_->searchFrame->findPrevious();
}
void MainWindow::on_actionEditMarkPacket_triggered()
{
freeze();
packet_list_->markFrame();
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditMarkAllDisplayed_triggered()
{
freeze();
packet_list_->markAllDisplayedFrames(true);
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditUnmarkAllDisplayed_triggered()
{
freeze();
packet_list_->markAllDisplayedFrames(false);
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditNextMark_triggered()
{
if (capture_file_.capFile())
cf_find_packet_marked(capture_file_.capFile(), SD_FORWARD);
}
void MainWindow::on_actionEditPreviousMark_triggered()
{
if (capture_file_.capFile())
cf_find_packet_marked(capture_file_.capFile(), SD_BACKWARD);
}
void MainWindow::on_actionEditIgnorePacket_triggered()
{
freeze();
packet_list_->ignoreFrame();
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditIgnoreAllDisplayed_triggered()
{
freeze();
packet_list_->ignoreAllDisplayedFrames(true);
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditUnignoreAllDisplayed_triggered()
{
freeze();
packet_list_->ignoreAllDisplayedFrames(false);
thaw();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditSetTimeReference_triggered()
{
packet_list_->setTimeReference();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditUnsetAllTimeReferences_triggered()
{
packet_list_->unsetAllTimeReferences();
setMenusForSelectedPacket();
}
void MainWindow::on_actionEditNextTimeReference_triggered()
{
if (!capture_file_.capFile()) return;
cf_find_packet_time_reference(capture_file_.capFile(), SD_FORWARD);
}
void MainWindow::on_actionEditPreviousTimeReference_triggered()
{
if (!capture_file_.capFile()) return;
cf_find_packet_time_reference(capture_file_.capFile(), SD_BACKWARD);
}
void MainWindow::on_actionEditTimeShift_triggered()
{
TimeShiftDialog *ts_dialog = new TimeShiftDialog(this, capture_file_.capFile());
connect(ts_dialog, SIGNAL(finished(int)), this, SLOT(editTimeShiftFinished(int)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
ts_dialog, SLOT(setCaptureFile(capture_file*)));
connect(ts_dialog, SIGNAL(timeShifted()), packet_list_, SLOT(applyTimeShift()));
ts_dialog->setWindowModality(Qt::ApplicationModal);
ts_dialog->setAttribute(Qt::WA_DeleteOnClose);
ts_dialog->show();
}
void MainWindow::editTimeShiftFinished(int)
{
if (capture_file_.capFile()->unsaved_changes) {
updateForUnsavedChanges();
}
}
void MainWindow::actionAddPacketComment()
{
QList<int> rows = selectedRows();
if (rows.count() == 0)
return;
frame_data * fdata = frameDataForRow(rows.at(0));
if (! fdata)
return;
PacketCommentDialog* pc_dialog;
pc_dialog = new PacketCommentDialog(false, this, NULL);
connect(pc_dialog, &QDialog::finished, std::bind(&MainWindow::addPacketCommentFinished, this, pc_dialog, std::placeholders::_1));
pc_dialog->setWindowModality(Qt::ApplicationModal);
pc_dialog->setAttribute(Qt::WA_DeleteOnClose);
pc_dialog->show();
}
void MainWindow::addPacketCommentFinished(PacketCommentDialog* pc_dialog _U_, int result _U_)
{
if (result == QDialog::Accepted) {
packet_list_->addPacketComment(pc_dialog->text());
updateForUnsavedChanges();
}
}
void MainWindow::actionEditPacketComment()
{
QList<int> rows = selectedRows();
if (rows.count() != 1)
return;
QAction *ra = qobject_cast<QAction*>(sender());
guint nComment = ra->data().toUInt();
PacketCommentDialog* pc_dialog;
pc_dialog = new PacketCommentDialog(true, this, packet_list_->getPacketComment(nComment));
connect(pc_dialog, &QDialog::finished, std::bind(&MainWindow::editPacketCommentFinished, this, pc_dialog, std::placeholders::_1, nComment));
pc_dialog->setWindowModality(Qt::ApplicationModal);
pc_dialog->setAttribute(Qt::WA_DeleteOnClose);
pc_dialog->show();
}
void MainWindow::editPacketCommentFinished(PacketCommentDialog* pc_dialog _U_, int result _U_, guint nComment)
{
if (result == QDialog::Accepted) {
packet_list_->setPacketComment(nComment, pc_dialog->text());
updateForUnsavedChanges();
}
}
void MainWindow::actionDeletePacketComment()
{
QAction *ra = qobject_cast<QAction*>(sender());
guint nComment = ra->data().toUInt();
packet_list_->setPacketComment(nComment, QString(""));
updateForUnsavedChanges();
}
void MainWindow::actionDeleteCommentsFromPackets()
{
packet_list_->deleteCommentsFromPackets();
updateForUnsavedChanges();
}
void MainWindow::on_actionDeleteAllPacketComments_triggered()
{
QMessageBox *msg_dialog = new QMessageBox();
connect(msg_dialog, SIGNAL(finished(int)), this, SLOT(deleteAllPacketCommentsFinished(int)));
msg_dialog->setIcon(QMessageBox::Question);
msg_dialog->setText(tr("Are you sure you want to remove all packet comments?"));
msg_dialog->setStandardButtons(QMessageBox::Ok | QMessageBox::Cancel);
msg_dialog->setDefaultButton(QMessageBox::Ok);
msg_dialog->setWindowModality(Qt::ApplicationModal);
msg_dialog->setAttribute(Qt::WA_DeleteOnClose);
msg_dialog->show();
}
void MainWindow::deleteAllPacketCommentsFinished(int result)
{
if (result == QMessageBox::Ok) {
/* XXX Do we need a wait/hourglass for large files? */
packet_list_->deleteAllPacketComments();
updateForUnsavedChanges();
}
}
void MainWindow::on_actionEditConfigurationProfiles_triggered()
{
ProfileDialog *cp_dialog = new ProfileDialog();
cp_dialog->setWindowModality(Qt::ApplicationModal);
cp_dialog->setAttribute(Qt::WA_DeleteOnClose);
cp_dialog->show();
}
void MainWindow::showPreferencesDialog(QString module_name)
{
PreferencesDialog *pref_dialog = new PreferencesDialog(this);
connect(pref_dialog, SIGNAL(destroyed(QObject*)), wsApp, SLOT(flushAppSignals()));
saveWindowGeometry(); // Save in case the layout panes are rearranged
pref_dialog->setPane(module_name);
pref_dialog->setWindowModality(Qt::ApplicationModal);
pref_dialog->setAttribute(Qt::WA_DeleteOnClose);
pref_dialog->show();
}
void MainWindow::on_actionEditPreferences_triggered()
{
showPreferencesDialog(PrefsModel::typeToString(PrefsModel::Appearance));
}
// View Menu
void MainWindow::showHideMainWidgets(QAction *action)
{
if (!action) {
return;
}
bool show = action->isChecked();
QWidget *widget = action->data().value<QWidget*>();
// We may have come from the toolbar context menu, so check/uncheck each
// action as well.
if (widget == main_ui_->mainToolBar) {
recent.main_toolbar_show = show;
main_ui_->actionViewMainToolbar->setChecked(show);
} else if (widget == main_ui_->displayFilterToolBar) {
recent.filter_toolbar_show = show;
main_ui_->actionViewFilterToolbar->setChecked(show);
#if defined(HAVE_LIBNL) && defined(HAVE_NL80211)
} else if (widget == main_ui_->wirelessToolBar) {
recent.wireless_toolbar_show = show;
main_ui_->actionViewWirelessToolbar->setChecked(show);
#endif
} else if (widget == main_ui_->statusBar) {
recent.statusbar_show = show;
main_ui_->actionViewStatusBar->setChecked(show);
} else if (widget == packet_list_) {
recent.packet_list_show = show;
main_ui_->actionViewPacketList->setChecked(show);
} else if (widget == proto_tree_) {
recent.tree_view_show = show;
main_ui_->actionViewPacketDetails->setChecked(show);
} else if (widget == byte_view_tab_) {
recent.byte_view_show = show;
main_ui_->actionViewPacketBytes->setChecked(show);
} else if (widget == packet_diagram_) {
recent.packet_diagram_show = show;
main_ui_->actionViewPacketDiagram->setChecked(show);
} else {
foreach(QAction *action, main_ui_->menuInterfaceToolbars->actions()) {
QToolBar *toolbar = action->data().value<QToolBar *>();
if (widget == toolbar) {
GList *entry = g_list_find_custom(recent.interface_toolbars, action->text().toUtf8(), (GCompareFunc)strcmp);
if (show && !entry) {
recent.interface_toolbars = g_list_append(recent.interface_toolbars, g_strdup(action->text().toUtf8()));
} else if (!show && entry) {
recent.interface_toolbars = g_list_remove(recent.interface_toolbars, entry->data);
}
action->setChecked(show);
}
}
ext_toolbar_t * toolbar = VariantPointer<ext_toolbar_t>::asPtr(action->data());
if (toolbar) {
GList *entry = g_list_find_custom(recent.gui_additional_toolbars, toolbar->name, (GCompareFunc)strcmp);
if (show && !entry) {
recent.gui_additional_toolbars = g_list_append(recent.gui_additional_toolbars, g_strdup(toolbar->name));
} else if (!show && entry) {
recent.gui_additional_toolbars = g_list_remove(recent.gui_additional_toolbars, entry->data);
}
action->setChecked(show);
QList<QToolBar *> toolbars = findChildren<QToolBar *>();
foreach(QToolBar *bar, toolbars) {
AdditionalToolBar *iftoolbar = dynamic_cast<AdditionalToolBar *>(bar);
if (iftoolbar && iftoolbar->menuName().compare(toolbar->name) == 0) {
iftoolbar->setVisible(show);
}
}
}
}
if (widget) {
widget->setVisible(show);
}
}
void MainWindow::setTimestampFormat(QAction *action)
{
if (!action) {
return;
}
ts_type tsf = action->data().value<ts_type>();
if (recent.gui_time_format != tsf) {
timestamp_set_type(tsf);
recent.gui_time_format = tsf;
if (packet_list_) {
packet_list_->resetColumns();
}
if (capture_file_.capFile()) {
/* This call adjusts column width */
cf_timestamp_auto_precision(capture_file_.capFile());
}
}
}
void MainWindow::setTimestampPrecision(QAction *action)
{
if (!action) {
return;
}
ts_precision tsp = action->data().value<ts_precision>();
if (recent.gui_time_precision != tsp) {
/* the actual precision will be set in packet_list_queue_draw() below */
timestamp_set_precision(tsp);
recent.gui_time_precision = tsp;
if (packet_list_) {
packet_list_->resetColumns();
}
if (capture_file_.capFile()) {
/* This call adjusts column width */
cf_timestamp_auto_precision(capture_file_.capFile());
}
}
}
void MainWindow::on_actionViewTimeDisplaySecondsWithHoursAndMinutes_triggered(bool checked)
{
if (checked) {
recent.gui_seconds_format = TS_SECONDS_HOUR_MIN_SEC;
} else {
recent.gui_seconds_format = TS_SECONDS_DEFAULT;
}
timestamp_set_seconds_type(recent.gui_seconds_format);
if (packet_list_) {
packet_list_->resetColumns();
}
if (capture_file_.capFile()) {
/* This call adjusts column width */
cf_timestamp_auto_precision(capture_file_.capFile());
}
}
void MainWindow::on_actionViewEditResolvedName_triggered()
{
//int column = packet_list_->selectedColumn();
int column = -1;
if (packet_list_->currentIndex().isValid()) {
column = packet_list_->currentIndex().column();
}
main_ui_->addressEditorFrame->editAddresses(capture_file_, column);
showAccordionFrame(main_ui_->addressEditorFrame);
}
void MainWindow::setNameResolution()
{
gbl_resolv_flags.mac_name = main_ui_->actionViewNameResolutionPhysical->isChecked() ? TRUE : FALSE;
gbl_resolv_flags.network_name = main_ui_->actionViewNameResolutionNetwork->isChecked() ? TRUE : FALSE;
gbl_resolv_flags.transport_name = main_ui_->actionViewNameResolutionTransport->isChecked() ? TRUE : FALSE;
if (packet_list_) {
packet_list_->resetColumns();
}
wsApp->emitAppSignal(WiresharkApplication::NameResolutionChanged);
}
void MainWindow::on_actionViewNameResolutionPhysical_triggered()
{
setNameResolution();
}
void MainWindow::on_actionViewNameResolutionNetwork_triggered()
{
setNameResolution();
}
void MainWindow::on_actionViewNameResolutionTransport_triggered()
{
setNameResolution();
}
void MainWindow::zoomText()
{
wsApp->zoomTextFont(recent.gui_zoom_level);
}
void MainWindow::on_actionViewZoomIn_triggered()
{
recent.gui_zoom_level++;
zoomText();
}
void MainWindow::on_actionViewZoomOut_triggered()
{
recent.gui_zoom_level--;
zoomText();
}
void MainWindow::on_actionViewNormalSize_triggered()
{
recent.gui_zoom_level = 0;
zoomText();
}
void MainWindow::on_actionViewColorizePacketList_triggered(bool checked) {
recent.packet_list_colorize = checked;
packet_list_recolor_packets();
packet_list_->resetColorized();
}
void MainWindow::on_actionViewColoringRules_triggered()
{
ColoringRulesDialog *coloring_rules_dialog = new ColoringRulesDialog(this);
connect(coloring_rules_dialog, SIGNAL(accepted()),
packet_list_, SLOT(recolorPackets()));
connect(coloring_rules_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
coloring_rules_dialog->setWindowModality(Qt::ApplicationModal);
coloring_rules_dialog->setAttribute(Qt::WA_DeleteOnClose);
coloring_rules_dialog->show();
}
// actionViewColorizeConversation1 - 10
void MainWindow::colorizeConversation(bool create_rule)
{
QAction *colorize_action = qobject_cast<QAction *>(sender());
if (!colorize_action) return;
if (capture_file_.capFile() && selectedRows().count() > 0) {
packet_info *pi = capture_file_.packetInfo();
guint8 cc_num = colorize_action->data().toUInt();
gchar *filter = conversation_filter_from_packet(pi);
if (filter == NULL) {
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, tr("Unable to build conversation filter."));
return;
}
if (create_rule) {
ColoringRulesDialog coloring_rules_dialog(this, filter);
connect(&coloring_rules_dialog, SIGNAL(accepted()),
packet_list_, SLOT(recolorPackets()));
connect(&coloring_rules_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
coloring_rules_dialog.exec();
} else {
gchar *err_msg = NULL;
if (!color_filters_set_tmp(cc_num, filter, FALSE, &err_msg)) {
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_msg);
g_free(err_msg);
}
packet_list_->recolorPackets();
}
}
setMenusForSelectedPacket();
}
void MainWindow::colorizeActionTriggered()
{
QByteArray filter;
int color_number = -1;
ConversationAction *conv_action = qobject_cast<ConversationAction *>(sender());
if (conv_action) {
filter = conv_action->filter();
color_number = conv_action->colorNumber();
} else {
ColorizeAction *colorize_action = qobject_cast<ColorizeAction *>(sender());
if (colorize_action) {
filter = colorize_action->filter();
color_number = colorize_action->colorNumber();
}
}
colorizeWithFilter(filter, color_number);
}
void MainWindow::colorizeWithFilter(QByteArray filter, int color_number)
{
if (filter.isEmpty()) return;
if (color_number > 0) {
// Assume "Color X"
gchar *err_msg = NULL;
if (!color_filters_set_tmp(color_number, filter.constData(), FALSE, &err_msg)) {
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_msg);
g_free(err_msg);
}
packet_list_->recolorPackets();
} else {
// New coloring rule
ColoringRulesDialog coloring_rules_dialog(window(), filter);
connect(&coloring_rules_dialog, SIGNAL(accepted()),
packet_list_, SLOT(recolorPackets()));
connect(&coloring_rules_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
coloring_rules_dialog.exec();
}
main_ui_->actionViewColorizeResetColorization->setEnabled(tmp_color_filters_used());
}
void MainWindow::on_actionViewColorizeResetColorization_triggered()
{
gchar *err_msg = NULL;
if (!color_filters_reset_tmp(&err_msg)) {
simple_dialog(ESD_TYPE_ERROR, ESD_BTN_OK, "%s", err_msg);
g_free(err_msg);
}
packet_list_->recolorPackets();
setMenusForSelectedPacket();
}
void MainWindow::on_actionViewColorizeNewColoringRule_triggered()
{
colorizeConversation(true);
}
void MainWindow::on_actionViewResetLayout_triggered()
{
recent.gui_geometry_main_upper_pane = 0;
recent.gui_geometry_main_lower_pane = 0;
applyRecentPaneGeometry();
}
void MainWindow::on_actionViewResizeColumns_triggered()
{
if (! packet_list_->model())
return;
for (int col = 0; col < packet_list_->model()->columnCount(); col++) {
packet_list_->resizeColumnToContents(col);
recent_set_column_width(col, packet_list_->columnWidth(col));
}
}
void MainWindow::openPacketDialog(bool from_reference)
{
frame_data * fdata = Q_NULLPTR;
/* Find the frame for which we're popping up a dialog */
if (from_reference) {
guint32 framenum = fvalue_get_uinteger(&(capture_file_.capFile()->finfo_selected->value));
if (framenum == 0)
return;
fdata = frame_data_sequence_find(capture_file_.capFile()->provider.frames, framenum);
} else if (selectedRows().count() == 1) {
fdata = frameDataForRow(selectedRows().at(0));
} else if (selectedRows().count() > 1)
return;
/* If we have a frame, pop up the dialog */
if (fdata) {
PacketDialog *packet_dialog = new PacketDialog(*this, capture_file_, fdata);
connect(packet_dialog, SIGNAL(showProtocolPreferences(QString)),
this, SLOT(showPreferencesDialog(QString)));
connect(packet_dialog, SIGNAL(editProtocolPreference(preference*, pref_module*)),
main_ui_->preferenceEditorFrame, SLOT(editPreference(preference*, pref_module*)));
connect(this, SIGNAL(closePacketDialogs()),
packet_dialog, SLOT(close()));
zoomText(); // Emits wsApp->zoomMonospaceFont(QFont)
packet_dialog->show();
}
}
void MainWindow::on_actionViewInternalsConversationHashTables_triggered()
{
ConversationHashTablesDialog *conversation_hash_tables_dlg = new ConversationHashTablesDialog(this);
conversation_hash_tables_dlg->show();
}
void MainWindow::on_actionViewInternalsDissectorTables_triggered()
{
DissectorTablesDialog *dissector_tables_dlg = new DissectorTablesDialog(this);
dissector_tables_dlg->show();
}
void MainWindow::on_actionViewInternalsSupportedProtocols_triggered()
{
SupportedProtocolsDialog *supported_protocols_dlg = new SupportedProtocolsDialog(this);
supported_protocols_dlg->show();
}
void MainWindow::on_actionViewShowPacketInNewWindow_triggered()
{
openPacketDialog();
}
// This is only used in ProtoTree. Defining it here makes more sense.
void MainWindow::on_actionContextShowLinkedPacketInNewWindow_triggered()
{
openPacketDialog(true);
}
void MainWindow::on_actionViewReload_triggered()
{
capture_file *cf = CaptureFile::globalCapFile();
if (cf->unsaved_changes) {
QString before_what(tr(" before reloading the file"));
if (!testCaptureFileClose(before_what, Reload))
return;
}
cf_reload(cf);
}
void MainWindow::on_actionViewReload_as_File_Format_or_Capture_triggered()
{
capture_file *cf = CaptureFile::globalCapFile();
if (cf->unsaved_changes) {
QString before_what(tr(" before reloading the file"));
if (!testCaptureFileClose(before_what, Reload))
return;
}
if (cf->open_type == WTAP_TYPE_AUTO)
cf->open_type = open_info_name_to_type("MIME Files Format");
else /* TODO: This should be latest format chosen by user */
cf->open_type = WTAP_TYPE_AUTO;
cf_reload(cf);
}
// Expand / collapse slots in proto_tree
// Go Menu
// Analyze Menu
void MainWindow::filterMenuAboutToShow()
{
QMenu * menu = qobject_cast<QMenu *>(sender());
QString field_filter;
if (capture_file_.capFile() && capture_file_.capFile()->finfo_selected) {
char *tmp_field = proto_construct_match_selected_string(capture_file_.capFile()->finfo_selected,
capture_file_.capFile()->edt);
field_filter = QString(tmp_field);
wmem_free(NULL, tmp_field);
}
bool enable = ! field_filter.isEmpty();
bool prepare = menu->objectName().compare("menuPrepareAFilter") == 0;
menu->clear();
QActionGroup * group = FilterAction::createFilterGroup(field_filter, prepare, enable, menu);
menu->addActions(group->actions());
}
void MainWindow::matchFieldFilter(FilterAction::Action action, FilterAction::ActionType filter_type)
{
QString field_filter;
if (packet_list_->contextMenuActive() || packet_list_->hasFocus()) {
field_filter = packet_list_->getFilterFromRowAndColumn(packet_list_->currentIndex());
} else if (capture_file_.capFile() && capture_file_.capFile()->finfo_selected) {
char *tmp_field = proto_construct_match_selected_string(capture_file_.capFile()->finfo_selected,
capture_file_.capFile()->edt);
field_filter = QString(tmp_field);
wmem_free(NULL, tmp_field);
}
if (field_filter.isEmpty()) {
QString err = tr("No filter available. Try another %1.").arg(packet_list_->contextMenuActive() ? tr("column") : tr("item"));
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, err);
return;
}
setDisplayFilter(field_filter, action, filter_type);
}
void MainWindow::setDisplayFilter(QString filter, FilterAction::Action action, FilterAction::ActionType filterType)
{
emit filterAction(filter, action, filterType);
}
void MainWindow::on_actionAnalyzeDisplayFilters_triggered()
{
if (!display_filter_dlg_) {
display_filter_dlg_ = new FilterDialog(this, FilterDialog::DisplayFilter);
}
display_filter_dlg_->show();
display_filter_dlg_->raise();
display_filter_dlg_->activateWindow();
}
struct epan_uat;
void MainWindow::on_actionAnalyzeDisplayFilterMacros_triggered()
{
struct epan_uat* dfm_uat;
dfilter_macro_get_uat(&dfm_uat);
UatDialog *uat_dlg = new UatDialog(parentWidget(), dfm_uat);
connect(uat_dlg, SIGNAL(destroyed(QObject*)), wsApp, SLOT(flushAppSignals()));
uat_dlg->setWindowModality(Qt::ApplicationModal);
uat_dlg->setAttribute(Qt::WA_DeleteOnClose);
uat_dlg->show();
}
void MainWindow::on_actionAnalyzeCreateAColumn_triggered()
{
if (capture_file_.capFile() != 0 && capture_file_.capFile()->finfo_selected != 0) {
header_field_info *hfinfo = capture_file_.capFile()->finfo_selected->hfinfo;
int col = column_prefs_has_custom(hfinfo->abbrev);
if (col == -1) {
insertColumn(hfinfo->name, hfinfo->abbrev);
} else {
QString status;
if (QString(hfinfo->name) == get_column_title(col)) {
status = tr("The \"%1\" column already exists.").arg(hfinfo->name);
} else {
status = tr("The \"%1\" column already exists as \"%2\".").arg(hfinfo->name).arg(get_column_title(col));
}
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, status);
if (!get_column_visible(col)) {
packet_list_->setColumnHidden(col, false);
set_column_visible(col, TRUE);
prefs_main_write();
}
}
}
}
void MainWindow::applyConversationFilter()
{
ConversationAction *conv_action = qobject_cast<ConversationAction*>(sender());
if (!conv_action) return;
packet_info *pinfo = capture_file_.packetInfo();
if (!pinfo) return;
QByteArray conv_filter = conv_action->filter();
if (conv_filter.isEmpty()) return;
if (conv_action->isFilterValid(pinfo)) {
df_combo_box_->lineEdit()->setText(conv_filter);
df_combo_box_->applyDisplayFilter();
}
}
void MainWindow::applyExportObject()
{
ExportObjectAction *export_action = qobject_cast<ExportObjectAction*>(sender());
if (!export_action)
return;
ExportObjectDialog* export_dialog = new ExportObjectDialog(*this, capture_file_, export_action->exportObject());
export_dialog->setWindowModality(Qt::ApplicationModal);
export_dialog->setAttribute(Qt::WA_DeleteOnClose);
export_dialog->show();
}
void MainWindow::on_actionAnalyzeEnabledProtocols_triggered()
{
EnabledProtocolsDialog *enable_proto_dialog = new EnabledProtocolsDialog(this);
connect(enable_proto_dialog, SIGNAL(destroyed(QObject*)), wsApp, SLOT(flushAppSignals()));
enable_proto_dialog->setWindowModality(Qt::ApplicationModal);
enable_proto_dialog->setAttribute(Qt::WA_DeleteOnClose);
enable_proto_dialog->show();
}
void MainWindow::on_actionAnalyzeDecodeAs_triggered()
{
QAction *da_action = qobject_cast<QAction*>(sender());
bool create_new = da_action && da_action->property("create_new").toBool();
DecodeAsDialog *da_dialog = new DecodeAsDialog(this, capture_file_.capFile(), create_new);
connect(da_dialog, SIGNAL(destroyed(QObject*)), wsApp, SLOT(flushAppSignals()));
da_dialog->setWindowModality(Qt::ApplicationModal);
da_dialog->setAttribute(Qt::WA_DeleteOnClose);
da_dialog->show();
}
void MainWindow::on_actionAnalyzeReloadLuaPlugins_triggered()
{
reloadLuaPlugins();
}
void MainWindow::openFollowStreamDialog(follow_type_t type, guint stream_num, guint sub_stream_num, bool use_stream_index) {
FollowStreamDialog *fsd = new FollowStreamDialog(*this, capture_file_, type);
connect(fsd, SIGNAL(updateFilter(QString, bool)), this, SLOT(filterPackets(QString, bool)));
connect(fsd, SIGNAL(goToPacket(int)), packet_list_, SLOT(goToPacket(int)));
fsd->addCodecs(text_codec_map_);
fsd->show();
if (use_stream_index) {
// If a specific conversation was requested, then ignore any previous
// display filters and display all related packets.
fsd->follow("", true, stream_num, sub_stream_num);
} else {
fsd->follow(getFilter());
}
}
void MainWindow::openFollowStreamDialogForType(follow_type_t type) {
openFollowStreamDialog(type, 0, 0, false);
}
void MainWindow::on_actionAnalyzeFollowTCPStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_TCP);
}
void MainWindow::on_actionAnalyzeFollowUDPStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_UDP);
}
void MainWindow::on_actionAnalyzeFollowDCCPStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_DCCP);
}
void MainWindow::on_actionAnalyzeFollowTLSStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_TLS);
}
void MainWindow::on_actionAnalyzeFollowHTTPStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_HTTP);
}
void MainWindow::on_actionAnalyzeFollowHTTP2Stream_triggered()
{
openFollowStreamDialogForType(FOLLOW_HTTP2);
}
void MainWindow::on_actionAnalyzeFollowQUICStream_triggered()
{
openFollowStreamDialogForType(FOLLOW_QUIC);
}
void MainWindow::on_actionAnalyzeFollowSIPCall_triggered()
{
openFollowStreamDialogForType(FOLLOW_SIP);
}
void MainWindow::openSCTPAllAssocsDialog()
{
SCTPAllAssocsDialog *sctp_dialog = new SCTPAllAssocsDialog(this, capture_file_.capFile());
connect(sctp_dialog, SIGNAL(filterPackets(QString, bool)),
this, SLOT(filterPackets(QString, bool)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
sctp_dialog, SLOT(setCaptureFile(capture_file*)));
sctp_dialog->fillTable();
if (sctp_dialog->isMinimized() == true)
{
sctp_dialog->showNormal();
}
else
{
sctp_dialog->show();
}
sctp_dialog->raise();
sctp_dialog->activateWindow();
}
void MainWindow::on_actionSCTPShowAllAssociations_triggered()
{
openSCTPAllAssocsDialog();
}
void MainWindow::on_actionSCTPAnalyseThisAssociation_triggered()
{
const sctp_assoc_info_t* assoc = SCTPAssocAnalyseDialog::findAssocForPacket(capture_file_.capFile());
if (!assoc) {
return;
}
SCTPAssocAnalyseDialog *sctp_analyse = new SCTPAssocAnalyseDialog(this, assoc, capture_file_.capFile());
connect(sctp_analyse, SIGNAL(filterPackets(QString, bool)),
this, SLOT(filterPackets(QString, bool)));
if (sctp_analyse->isMinimized() == true)
{
sctp_analyse->showNormal();
}
else
{
sctp_analyse->show();
}
sctp_analyse->raise();
sctp_analyse->activateWindow();
}
void MainWindow::on_actionSCTPFilterThisAssociation_triggered()
{
const sctp_assoc_info_t* assoc = SCTPAssocAnalyseDialog::findAssocForPacket(capture_file_.capFile());
if (assoc) {
QString newFilter = QString("sctp.assoc_index==%1").arg(assoc->assoc_id);
assoc = NULL;
emit filterPackets(newFilter, false);
}
}
// -z wlan,stat
void MainWindow::statCommandWlanStatistics(const char *arg, void *)
{
WlanStatisticsDialog *wlan_stats_dlg = new WlanStatisticsDialog(*this, capture_file_, arg);
connect(wlan_stats_dlg, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
wlan_stats_dlg->show();
}
void MainWindow::on_actionWirelessWlanStatistics_triggered()
{
statCommandWlanStatistics(NULL, NULL);
}
// -z expert
void MainWindow::statCommandExpertInfo(const char *, void *)
{
ExpertInfoDialog *expert_dialog = new ExpertInfoDialog(*this, capture_file_);
const DisplayFilterEdit *df_edit = dynamic_cast<DisplayFilterEdit *>(df_combo_box_->lineEdit());
expert_dialog->setDisplayFilter(df_edit->text());
connect(expert_dialog->getExpertInfoView(), SIGNAL(goToPacket(int, int)),
packet_list_, SLOT(goToPacket(int, int)));
connect(expert_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
expert_dialog->show();
}
void MainWindow::on_actionAnalyzeExpertInfo_triggered()
{
statCommandExpertInfo(NULL, NULL);
}
// Next / previous / first / last slots in packet_list
// Statistics Menu
void MainWindow::on_actionStatisticsFlowGraph_triggered()
{
SequenceDialog *sequence_dialog = new SequenceDialog(*this, capture_file_);
sequence_dialog->show();
}
void MainWindow::openTcpStreamDialog(int graph_type)
{
TCPStreamDialog *stream_dialog = new TCPStreamDialog(this, capture_file_.capFile(), (tcp_graph_type)graph_type);
connect(stream_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
stream_dialog, SLOT(setCaptureFile(capture_file*)));
if (stream_dialog->result() == QDialog::Accepted) {
stream_dialog->show();
}
}
void MainWindow::on_actionStatisticsTcpStreamStevens_triggered()
{
openTcpStreamDialog(GRAPH_TSEQ_STEVENS);
}
void MainWindow::on_actionStatisticsTcpStreamTcptrace_triggered()
{
openTcpStreamDialog(GRAPH_TSEQ_TCPTRACE);
}
void MainWindow::on_actionStatisticsTcpStreamThroughput_triggered()
{
openTcpStreamDialog(GRAPH_THROUGHPUT);
}
void MainWindow::on_actionStatisticsTcpStreamRoundTripTime_triggered()
{
openTcpStreamDialog(GRAPH_RTT);
}
void MainWindow::on_actionStatisticsTcpStreamWindowScaling_triggered()
{
openTcpStreamDialog(GRAPH_WSCALE);
}
// -z mcast,stat
void MainWindow::statCommandMulticastStatistics(const char *arg, void *)
{
MulticastStatisticsDialog *mcast_stats_dlg = new MulticastStatisticsDialog(*this, capture_file_, arg);
connect(mcast_stats_dlg, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
mcast_stats_dlg->show();
}
void MainWindow::on_actionStatisticsUdpMulticastStreams_triggered()
{
statCommandMulticastStatistics(NULL, NULL);
}
void MainWindow::openStatisticsTreeDialog(const gchar *abbr)
{
StatsTreeDialog *st_dialog = new StatsTreeDialog(*this, capture_file_, abbr);
// connect(st_dialog, SIGNAL(goToPacket(int)),
// packet_list_, SLOT(goToPacket(int)));
st_dialog->show();
}
void MainWindow::on_actionStatistics29WestTopics_Advertisements_by_Topic_triggered()
{
openStatisticsTreeDialog("lbmr_topic_ads_topic");
}
void MainWindow::on_actionStatistics29WestTopics_Advertisements_by_Source_triggered()
{
openStatisticsTreeDialog("lbmr_topic_ads_source");
}
void MainWindow::on_actionStatistics29WestTopics_Advertisements_by_Transport_triggered()
{
openStatisticsTreeDialog("lbmr_topic_ads_transport");
}
void MainWindow::on_actionStatistics29WestTopics_Queries_by_Topic_triggered()
{
openStatisticsTreeDialog("lbmr_topic_queries_topic");
}
void MainWindow::on_actionStatistics29WestTopics_Queries_by_Receiver_triggered()
{
openStatisticsTreeDialog("lbmr_topic_queries_receiver");
}
void MainWindow::on_actionStatistics29WestTopics_Wildcard_Queries_by_Pattern_triggered()
{
openStatisticsTreeDialog("lbmr_topic_queries_pattern");
}
void MainWindow::on_actionStatistics29WestTopics_Wildcard_Queries_by_Receiver_triggered()
{
openStatisticsTreeDialog("lbmr_topic_queries_pattern_receiver");
}
void MainWindow::on_actionStatistics29WestQueues_Advertisements_by_Queue_triggered()
{
openStatisticsTreeDialog("lbmr_queue_ads_queue");
}
void MainWindow::on_actionStatistics29WestQueues_Advertisements_by_Source_triggered()
{
openStatisticsTreeDialog("lbmr_queue_ads_source");
}
void MainWindow::on_actionStatistics29WestQueues_Queries_by_Queue_triggered()
{
openStatisticsTreeDialog("lbmr_queue_queries_queue");
}
void MainWindow::on_actionStatistics29WestQueues_Queries_by_Receiver_triggered()
{
openStatisticsTreeDialog("lbmr_queue_queries_receiver");
}
void MainWindow::on_actionStatistics29WestUIM_Streams_triggered()
{
LBMStreamDialog *stream_dialog = new LBMStreamDialog(this, capture_file_.capFile());
// connect(stream_dialog, SIGNAL(goToPacket(int)),
// packet_list_, SLOT(goToPacket(int)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
stream_dialog, SLOT(setCaptureFile(capture_file*)));
stream_dialog->show();
}
void MainWindow::on_actionStatistics29WestLBTRM_triggered()
{
LBMLBTRMTransportDialog * lbtrm_dialog = new LBMLBTRMTransportDialog(this, capture_file_.capFile());
connect(lbtrm_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
lbtrm_dialog, SLOT(setCaptureFile(capture_file*)));
lbtrm_dialog->show();
}
void MainWindow::on_actionStatistics29WestLBTRU_triggered()
{
LBMLBTRUTransportDialog * lbtru_dialog = new LBMLBTRUTransportDialog(this, capture_file_.capFile());
connect(lbtru_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(this, SIGNAL(setCaptureFile(capture_file*)),
lbtru_dialog, SLOT(setCaptureFile(capture_file*)));
lbtru_dialog->show();
}
void MainWindow::on_actionStatisticsANCP_triggered()
{
openStatisticsTreeDialog("ancp");
}
void MainWindow::on_actionStatisticsBACappInstanceId_triggered()
{
openStatisticsTreeDialog("bacapp_instanceid");
}
void MainWindow::on_actionStatisticsBACappIP_triggered()
{
openStatisticsTreeDialog("bacapp_ip");
}
void MainWindow::on_actionStatisticsBACappObjectId_triggered()
{
openStatisticsTreeDialog("bacapp_objectid");
}
void MainWindow::on_actionStatisticsBACappService_triggered()
{
openStatisticsTreeDialog("bacapp_service");
}
void MainWindow::on_actionStatisticsCollectd_triggered()
{
openStatisticsTreeDialog("collectd");
}
// -z conv,...
void MainWindow::statCommandConversations(const char *arg, void *userdata)
{
ConversationDialog *conv_dialog = new ConversationDialog(*this, capture_file_, GPOINTER_TO_INT(userdata), arg);
connect(conv_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
connect(conv_dialog, SIGNAL(openFollowStreamDialog(follow_type_t, guint, guint)),
this, SLOT(openFollowStreamDialog(follow_type_t, guint, guint)));
connect(conv_dialog, SIGNAL(openTcpStreamGraph(int)),
this, SLOT(openTcpStreamDialog(int)));
conv_dialog->show();
}
void MainWindow::on_actionStatisticsConversations_triggered()
{
statCommandConversations(NULL, NULL);
}
// -z endpoints,...
void MainWindow::statCommandEndpoints(const char *arg, void *userdata)
{
EndpointDialog *endp_dialog = new EndpointDialog(*this, capture_file_, GPOINTER_TO_INT(userdata), arg);
connect(endp_dialog, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
connect(endp_dialog, SIGNAL(openFollowStreamDialog(follow_type_t)),
this, SLOT(openFollowStreamDialogForType(follow_type_t)));
connect(endp_dialog, SIGNAL(openTcpStreamGraph(int)),
this, SLOT(openTcpStreamDialog(int)));
endp_dialog->show();
}
void MainWindow::on_actionStatisticsEndpoints_triggered()
{
statCommandEndpoints(NULL, NULL);
}
void MainWindow::on_actionStatisticsHART_IP_triggered()
{
openStatisticsTreeDialog("hart_ip");
}
void MainWindow::on_actionStatisticsHTTPPacketCounter_triggered()
{
openStatisticsTreeDialog("http");
}
void MainWindow::on_actionStatisticsHTTPRequests_triggered()
{
openStatisticsTreeDialog("http_req");
}
void MainWindow::on_actionStatisticsHTTPLoadDistribution_triggered()
{
openStatisticsTreeDialog("http_srv");
}
void MainWindow::on_actionStatisticsHTTPRequestSequences_triggered()
{
openStatisticsTreeDialog("http_seq");
}
void MainWindow::on_actionStatisticsPacketLengths_triggered()
{
openStatisticsTreeDialog("plen");
}
// -z io,stat
void MainWindow::statCommandIOGraph(const char *, void *)
{
const DisplayFilterEdit *df_edit = qobject_cast<DisplayFilterEdit *>(df_combo_box_->lineEdit());
QString displayFilter;
if (df_edit)
displayFilter = df_edit->text();
IOGraphDialog *iog_dialog = new IOGraphDialog(*this, capture_file_, displayFilter);
connect(iog_dialog, SIGNAL(goToPacket(int)), packet_list_, SLOT(goToPacket(int)));
connect(this, SIGNAL(reloadFields()), iog_dialog, SLOT(reloadFields()));
iog_dialog->show();
}
void MainWindow::on_actionStatisticsIOGraph_triggered()
{
statCommandIOGraph(NULL, NULL);
}
void MainWindow::on_actionStatisticsSametime_triggered()
{
openStatisticsTreeDialog("sametime");
}
void MainWindow::on_actionStatisticsDNS_triggered()
{
openStatisticsTreeDialog("dns");
}
void MainWindow::actionStatisticsPlugin_triggered()
{
QAction* action = qobject_cast<QAction*>(sender());
if (action) {
openStatisticsTreeDialog(action->data().toString().toUtf8());
}
}
void MainWindow::on_actionStatisticsHTTP2_triggered()
{
openStatisticsTreeDialog("http2");
}
// Telephony Menu
RtpPlayerDialog *MainWindow::openTelephonyRtpPlayerDialog()
{
RtpPlayerDialog *dialog;
#ifdef HAVE_LIBPCAP
dialog = RtpPlayerDialog::openRtpPlayerDialog(*this, capture_file_, packet_list_, captureSession()->state != CAPTURE_STOPPED);
#else
dialog = RtpPlayerDialog::openRtpPlayerDialog(*this, capture_file_, packet_list_, false);
#endif
dialog->show();
return dialog;
}
VoipCallsDialog *MainWindow::openTelephonyVoipCallsDialogVoip()
{
VoipCallsDialog *dialog;
dialog = VoipCallsDialog::openVoipCallsDialogVoip(*this, capture_file_, packet_list_);
dialog->show();
return dialog;
}
VoipCallsDialog *MainWindow::openTelephonyVoipCallsDialogSip()
{
VoipCallsDialog *dialog;
dialog = VoipCallsDialog::openVoipCallsDialogSip(*this, capture_file_, packet_list_);
dialog->show();
return dialog;
}
RtpAnalysisDialog *MainWindow::openTelephonyRtpAnalysisDialog()
{
RtpAnalysisDialog *dialog;
dialog = RtpAnalysisDialog::openRtpAnalysisDialog(*this, capture_file_, packet_list_);
dialog->show();
return dialog;
}
void MainWindow::on_actionTelephonyVoipCalls_triggered()
{
openTelephonyVoipCallsDialogVoip();
}
void MainWindow::on_actionTelephonyGsmMapSummary_triggered()
{
GsmMapSummaryDialog *gms_dialog = new GsmMapSummaryDialog(*this, capture_file_);
gms_dialog->show();
}
void MainWindow::on_actionTelephonyIax2StreamAnalysis_triggered()
{
Iax2AnalysisDialog *iax2_analysis_dialog = new Iax2AnalysisDialog(*this, capture_file_);
connect(iax2_analysis_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
iax2_analysis_dialog->show();
}
void MainWindow::on_actionTelephonyISUPMessages_triggered()
{
openStatisticsTreeDialog("isup_msg");
}
// -z mac-lte,stat
void MainWindow::statCommandLteMacStatistics(const char *arg, void *)
{
LteMacStatisticsDialog *lte_mac_stats_dlg = new LteMacStatisticsDialog(*this, capture_file_, arg);
connect(lte_mac_stats_dlg, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
lte_mac_stats_dlg->show();
}
void MainWindow::on_actionTelephonyLteMacStatistics_triggered()
{
statCommandLteMacStatistics(NULL, NULL);
}
void MainWindow::statCommandLteRlcStatistics(const char *arg, void *)
{
LteRlcStatisticsDialog *lte_rlc_stats_dlg = new LteRlcStatisticsDialog(*this, capture_file_, arg);
connect(lte_rlc_stats_dlg, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
// N.B. It is necessary for the RLC Statistics window to launch the RLC graph in this way, to ensure
// that the goToPacket() signal/slot connection gets set up...
connect(lte_rlc_stats_dlg, SIGNAL(launchRLCGraph(bool, guint16, guint8, guint16, guint16, guint8)),
this, SLOT(launchRLCGraph(bool, guint16, guint8, guint16, guint16, guint8)));
lte_rlc_stats_dlg->show();
}
void MainWindow::on_actionTelephonyLteRlcStatistics_triggered()
{
statCommandLteRlcStatistics(NULL, NULL);
}
void MainWindow::launchRLCGraph(bool channelKnown,
guint16 ueid, guint8 rlcMode,
guint16 channelType, guint16 channelId, guint8 direction)
{
LteRlcGraphDialog *lrg_dialog = new LteRlcGraphDialog(*this, capture_file_, channelKnown);
connect(lrg_dialog, SIGNAL(goToPacket(int)), packet_list_, SLOT(goToPacket(int)));
// This is a bit messy, but wanted to hide these parameters from users of
// on_actionTelephonyLteRlcGraph_triggered().
if (channelKnown) {
lrg_dialog->setChannelInfo(ueid, rlcMode, channelType, channelId, direction);
}
lrg_dialog->show();
}
void MainWindow::on_actionTelephonyLteRlcGraph_triggered()
{
// We don't yet know the channel.
launchRLCGraph(false, 0, 0, 0, 0, 0);
}
void MainWindow::on_actionTelephonyMtp3Summary_triggered()
{
Mtp3SummaryDialog *mtp3s_dialog = new Mtp3SummaryDialog(*this, capture_file_);
mtp3s_dialog->show();
}
void MainWindow::on_actionTelephonyOsmuxPacketCounter_triggered()
{
openStatisticsTreeDialog("osmux");
}
RtpStreamDialog *MainWindow::openTelephonyRtpStreamsDialog()
{
RtpStreamDialog *dialog;
dialog = RtpStreamDialog::openRtpStreamDialog(*this, capture_file_, packet_list_);
dialog->show();
return dialog;
}
void MainWindow::on_actionTelephonyRtpStreams_triggered()
{
openTelephonyRtpStreamsDialog();
}
void MainWindow::on_actionTelephonyRtpStreamAnalysis_triggered()
{
QVector<rtpstream_id_t *> stream_ids;
QString err;
if (QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier)) {
err = findRtpStreams(&stream_ids, true);
} else {
err = findRtpStreams(&stream_ids, false);
}
if (err != NULL) {
QMessageBox::warning(this, tr("RTP packet search failed"),
err,
QMessageBox::Ok);
} else {
openTelephonyRtpAnalysisDialog()->addRtpStreams(stream_ids);
}
foreach(rtpstream_id_t *id, stream_ids) {
rtpstream_id_free(id);
}
}
void MainWindow::on_actionTelephonyRtpPlayer_triggered()
{
QVector<rtpstream_id_t *> stream_ids;
QString err;
if (QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier)) {
err = findRtpStreams(&stream_ids, true);
} else {
err = findRtpStreams(&stream_ids, false);
}
if (err != NULL) {
QMessageBox::warning(this, tr("RTP packet search failed"),
err,
QMessageBox::Ok);
} else {
openTelephonyRtpPlayerDialog()->addRtpStreams(stream_ids);
}
foreach(rtpstream_id_t *id, stream_ids) {
rtpstream_id_free(id);
}
}
void MainWindow::on_actionTelephonyRTSPPacketCounter_triggered()
{
openStatisticsTreeDialog("rtsp");
}
void MainWindow::on_actionTelephonySMPPOperations_triggered()
{
openStatisticsTreeDialog("smpp_commands");
}
void MainWindow::on_actionTelephonyUCPMessages_triggered()
{
openStatisticsTreeDialog("ucp_messages");
}
void MainWindow::on_actionTelephonyF1APMessages_triggered()
{
openStatisticsTreeDialog("f1ap");
}
void MainWindow::on_actionTelephonySipFlows_triggered()
{
openTelephonyVoipCallsDialogSip();
}
// Wireless Menu
void MainWindow::on_actionBluetoothATT_Server_Attributes_triggered()
{
BluetoothAttServerAttributesDialog *bluetooth_att_sever_attributes_dialog = new BluetoothAttServerAttributesDialog(*this, capture_file_);
connect(bluetooth_att_sever_attributes_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(bluetooth_att_sever_attributes_dialog, SIGNAL(updateFilter(QString, bool)),
this, SLOT(filterPackets(QString, bool)));
bluetooth_att_sever_attributes_dialog->show();
}
void MainWindow::on_actionBluetoothDevices_triggered()
{
BluetoothDevicesDialog *bluetooth_devices_dialog = new BluetoothDevicesDialog(*this, capture_file_, packet_list_);
connect(bluetooth_devices_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(bluetooth_devices_dialog, SIGNAL(updateFilter(QString, bool)),
this, SLOT(filterPackets(QString, bool)));
bluetooth_devices_dialog->show();
}
void MainWindow::on_actionBluetoothHCI_Summary_triggered()
{
BluetoothHciSummaryDialog *bluetooth_hci_summary_dialog = new BluetoothHciSummaryDialog(*this, capture_file_);
connect(bluetooth_hci_summary_dialog, SIGNAL(goToPacket(int)),
packet_list_, SLOT(goToPacket(int)));
connect(bluetooth_hci_summary_dialog, SIGNAL(updateFilter(QString, bool)),
this, SLOT(filterPackets(QString, bool)));
bluetooth_hci_summary_dialog->show();
}
// Tools Menu
void MainWindow::on_actionToolsFirewallAclRules_triggered()
{
FirewallRulesDialog *firewall_rules_dialog = new FirewallRulesDialog(*this, capture_file_);
firewall_rules_dialog->show();
}
void MainWindow::on_actionToolsCredentials_triggered()
{
CredentialsDialog *credentials_dialog = new CredentialsDialog(*this, capture_file_, packet_list_);
credentials_dialog->show();
}
// Help Menu
void MainWindow::on_actionHelpContents_triggered() {
wsApp->helpTopicAction(HELP_CONTENT);
}
void MainWindow::on_actionHelpMPWireshark_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_WIRESHARK);
}
void MainWindow::on_actionHelpMPWireshark_Filter_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_WIRESHARK_FILTER);
}
void MainWindow::on_actionHelpMPCapinfos_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_CAPINFOS);
}
void MainWindow::on_actionHelpMPDumpcap_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_DUMPCAP);
}
void MainWindow::on_actionHelpMPEditcap_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_EDITCAP);
}
void MainWindow::on_actionHelpMPMergecap_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_MERGECAP);
}
void MainWindow::on_actionHelpMPRawShark_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_RAWSHARK);
}
void MainWindow::on_actionHelpMPReordercap_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_REORDERCAP);
}
void MainWindow::on_actionHelpMPText2cap_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_TEXT2PCAP);
}
void MainWindow::on_actionHelpMPTShark_triggered() {
wsApp->helpTopicAction(LOCALPAGE_MAN_TSHARK);
}
void MainWindow::on_actionHelpWebsite_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_HOME);
}
void MainWindow::on_actionHelpFAQ_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_FAQ);
}
void MainWindow::on_actionHelpAsk_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_ASK);
}
void MainWindow::on_actionHelpDownloads_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_DOWNLOAD);
}
void MainWindow::on_actionHelpWiki_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_WIKI);
}
void MainWindow::on_actionHelpSampleCaptures_triggered() {
wsApp->helpTopicAction(ONLINEPAGE_SAMPLE_FILES);
}
#ifdef HAVE_SOFTWARE_UPDATE
void MainWindow::checkForUpdates()
{
software_update_check();
}
#endif
void MainWindow::on_actionHelpAbout_triggered()
{
AboutDialog *about_dialog = new AboutDialog(this);
if (about_dialog->isMinimized() == true)
{
about_dialog->showNormal();
}
else
{
about_dialog->show();
}
about_dialog->raise();
about_dialog->activateWindow();
}
void MainWindow::on_actionGoGoToPacket_triggered() {
if (! packet_list_->model() || packet_list_->model()->rowCount() < 1) {
return;
}
previous_focus_ = wsApp->focusWidget();
connect(previous_focus_, SIGNAL(destroyed()), this, SLOT(resetPreviousFocus()));
showAccordionFrame(main_ui_->goToFrame, true);
if (main_ui_->goToFrame->isVisible()) {
main_ui_->goToLineEdit->clear();
main_ui_->goToLineEdit->setFocus();
}
}
void MainWindow::on_actionGoGoToLinkedPacket_triggered()
{
QAction *gta = qobject_cast<QAction*>(sender());
if (!gta) return;
bool ok = false;
int packet_num = gta->data().toInt(&ok);
if (!ok) return;
packet_list_->goToPacket(packet_num);
}
// gtk/main_menubar.c:goto_conversation_frame
void MainWindow::goToConversationFrame(bool go_next) {
gchar *filter = NULL;
dfilter_t *dfcode = NULL;
gboolean found_packet = FALSE;
packet_info *pi = capture_file_.packetInfo();
if (!pi) {
// No packet was selected, or multiple packets were selected.
return;
}
/* Try to build a conversation
* filter in the order TCP, UDP, IP, Ethernet and apply the
* coloring */
filter = conversation_filter_from_packet(pi);
if (filter == NULL) {
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, tr("Unable to build conversation filter."));
g_free(filter);
return;
}
if (!dfilter_compile(filter, &dfcode, NULL)) {
/* The attempt failed; report an error. */
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, tr("Error compiling filter for this conversation."));
g_free(filter);
return;
}
found_packet = cf_find_packet_dfilter(capture_file_.capFile(), dfcode, go_next ? SD_FORWARD : SD_BACKWARD);
if (!found_packet) {
/* We didn't find a packet */
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, tr("No previous/next packet in conversation."));
}
dfilter_free(dfcode);
g_free(filter);
}
void MainWindow::on_actionGoNextConversationPacket_triggered()
{
goToConversationFrame(true);
}
void MainWindow::on_actionGoPreviousConversationPacket_triggered()
{
goToConversationFrame(false);
}
void MainWindow::on_actionGoAutoScroll_toggled(bool checked)
{
packet_list_->setVerticalAutoScroll(checked);
}
void MainWindow::resetPreviousFocus() {
previous_focus_ = NULL;
}
void MainWindow::on_goToCancel_clicked()
{
main_ui_->goToFrame->animatedHide();
if (previous_focus_) {
disconnect(previous_focus_, SIGNAL(destroyed()), this, SLOT(resetPreviousFocus()));
previous_focus_->setFocus();
resetPreviousFocus();
}
}
void MainWindow::on_goToGo_clicked()
{
gotoFrame(main_ui_->goToLineEdit->text().toInt());
on_goToCancel_clicked();
}
void MainWindow::on_goToLineEdit_returnPressed()
{
on_goToGo_clicked();
}
void MainWindow::on_actionCaptureStart_triggered()
{
//#ifdef HAVE_AIRPCAP
// airpcap_if_active = airpcap_if_selected;
// if (airpcap_if_active)
// airpcap_set_toolbar_start_capture(airpcap_if_active);
//#endif
// if (cap_open_w) {
// /*
// * There's an options dialog; get the values from it and close it.
// */
// gboolean success;
// /* Determine if "capture start" while building of the "capture options" window */
// /* is in progress. If so, ignore the "capture start. */
// /* XXX: Would it be better/cleaner for the "capture options" window code to */
// /* disable the capture start button temporarily ? */
// if (cap_open_complete == FALSE) {
// return; /* Building options window: ignore "capture start" */
// }
// success = capture_dlg_prep(cap_open_w);
// window_destroy(GTK_WIDGET(cap_open_w));
// if (!success)
// return; /* error in options dialog */
// }
#ifdef HAVE_LIBPCAP
if (global_capture_opts.num_selected == 0) {
QString err_msg = tr("No Interface Selected.");
wsApp->pushStatus(WiresharkApplication::TemporaryStatus, err_msg);
main_ui_->actionCaptureStart->setChecked(false);
return;
}
/* XXX - will closing this remove a temporary file? */
QString before_what(tr(" before starting a new capture"));
if (testCaptureFileClose(before_what)) {
startCapture();
} else {
// simply clicking the button sets it to 'checked' even though we've
// decided to do nothing, so undo that
main_ui_->actionCaptureStart->setChecked(false);
}
#endif // HAVE_LIBPCAP
}
void MainWindow::on_actionCaptureStop_triggered()
{
stopCapture();
}
void MainWindow::on_actionCaptureRestart_triggered()
{
#ifdef HAVE_LIBPCAP
QString before_what(tr(" before restarting the capture"));
cap_session_.capture_opts->restart = TRUE;
if (!testCaptureFileClose(before_what, Restart))
return;
startCapture();
#endif // HAVE_LIBPCAP
}
void MainWindow::on_actionCaptureCaptureFilters_triggered()
{
if (!capture_filter_dlg_) {
capture_filter_dlg_ = new FilterDialog(this, FilterDialog::CaptureFilter);
}
capture_filter_dlg_->show();
capture_filter_dlg_->raise();
capture_filter_dlg_->activateWindow();
}
void MainWindow::on_actionStatisticsCaptureFileProperties_triggered()
{
CaptureFilePropertiesDialog *capture_file_properties_dialog = new CaptureFilePropertiesDialog(*this, capture_file_);
connect(capture_file_properties_dialog, SIGNAL(captureCommentChanged()),
this, SLOT(updateForUnsavedChanges()));
capture_file_properties_dialog->show();
}
void MainWindow::on_actionStatisticsResolvedAddresses_triggered()
{
QString capFileName;
wtap* wth = Q_NULLPTR;
if (capture_file_.isValid())
{
capFileName = capture_file_.capFile()->filename;
wth = capture_file_.capFile()->provider.wth;
}
ResolvedAddressesDialog *resolved_addresses_dialog =
new ResolvedAddressesDialog(this, capFileName, wth);
resolved_addresses_dialog->show();
}
void MainWindow::on_actionStatisticsProtocolHierarchy_triggered()
{
ProtocolHierarchyDialog *phd = new ProtocolHierarchyDialog(*this, capture_file_);
connect(phd, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)),
this, SIGNAL(filterAction(QString, FilterAction::Action, FilterAction::ActionType)));
phd->show();
}
void MainWindow::on_actionCaptureOptions_triggered()
{
#ifdef HAVE_LIBPCAP
if (!capture_options_dialog_) {
capture_options_dialog_ = new CaptureOptionsDialog(this);
connect(capture_options_dialog_, SIGNAL(startCapture()), this, SLOT(startCapture()));
connect(capture_options_dialog_, SIGNAL(stopCapture()), this, SLOT(stopCapture()));
connect(capture_options_dialog_, SIGNAL(getPoints(int, PointList*)),
this->welcome_page_->getInterfaceFrame(), SLOT(getPoints(int, PointList*)));
connect(capture_options_dialog_, SIGNAL(interfacesChanged()),
this->welcome_page_, SLOT(interfaceSelected()));
connect(capture_options_dialog_, SIGNAL(interfacesChanged()),
this->welcome_page_->getInterfaceFrame(), SLOT(updateSelectedInterfaces()));
connect(capture_options_dialog_, SIGNAL(interfaceListChanged()),
this->welcome_page_->getInterfaceFrame(), SLOT(interfaceListChanged()));
connect(capture_options_dialog_, SIGNAL(captureFilterTextEdited(QString)),
this->welcome_page_, SLOT(setCaptureFilterText(QString)));
// Propagate selection changes from main UI to dialog.
connect(this->welcome_page_, SIGNAL(interfacesChanged()),
capture_options_dialog_, SLOT(interfaceSelected()));
connect(capture_options_dialog_, SIGNAL(setFilterValid(bool, const QString)),
this, SLOT(startInterfaceCapture(bool, const QString)));
}
capture_options_dialog_->setTab(0);
capture_options_dialog_->updateInterfaces();
if (capture_options_dialog_->isMinimized()) {
capture_options_dialog_->showNormal();
} else {
capture_options_dialog_->show();
}
capture_options_dialog_->raise();
capture_options_dialog_->activateWindow();
#endif
}
#ifdef HAVE_LIBPCAP
void MainWindow::on_actionCaptureRefreshInterfaces_triggered()
{
main_ui_->actionCaptureRefreshInterfaces->setEnabled(false);
wsApp->refreshLocalInterfaces();
main_ui_->actionCaptureRefreshInterfaces->setEnabled(true);
}
#endif
void MainWindow::externalMenuItem_triggered()
{
QAction * triggerAction = NULL;
QVariant v;
ext_menubar_t * entry = NULL;
if (QObject::sender()) {
triggerAction = (QAction *)QObject::sender();
v = triggerAction->data();
if (v.canConvert<void *>()) {
entry = (ext_menubar_t *)v.value<void *>();
if (entry->type == EXT_MENUBAR_ITEM) {
entry->callback(EXT_MENUBAR_QT_GUI, (gpointer)((void *)main_ui_), entry->user_data);
} else {
QDesktopServices::openUrl(QUrl(QString((gchar *)entry->user_data)));
}
}
}
}
void MainWindow::gotoFrame(int packet_num)
{
if (packet_num > 0) {
packet_list_->goToPacket(packet_num);
}
}
void MainWindow::extcap_options_finished(int result)
{
if (result == QDialog::Accepted) {
QString before_what(tr(" before starting a new capture"));
if (testCaptureFileClose(before_what)) {
startCapture();
}
}
this->welcome_page_->getInterfaceFrame()->interfaceListChanged();
}
void MainWindow::showExtcapOptionsDialog(QString &device_name)
{
ExtcapOptionsDialog * extcap_options_dialog = ExtcapOptionsDialog::createForDevice(device_name, this);
/* The dialog returns null, if the given device name is not a valid extcap device */
if (extcap_options_dialog) {
extcap_options_dialog->setModal(true);
extcap_options_dialog->setAttribute(Qt::WA_DeleteOnClose);
connect(extcap_options_dialog, SIGNAL(finished(int)),
this, SLOT(extcap_options_finished(int)));
extcap_options_dialog->show();
}
}
void MainWindow::insertColumn(QString name, QString abbrev, gint pos)
{
gint colnr = 0;
if (name.length() > 0 && abbrev.length() > 0)
{
colnr = column_prefs_add_custom(COL_CUSTOM, name.toStdString().c_str(), abbrev.toStdString().c_str(), pos);
packet_list_->columnsChanged();
packet_list_->resizeColumnToContents(colnr);
prefs_main_write();
}
}
void MainWindow::on_actionContextWikiProtocolPage_triggered()
{
QAction *wa = qobject_cast<QAction*>(sender());
if (!wa) return;
bool ok = false;
int field_id = wa->data().toInt(&ok);
if (!ok) return;
const QString proto_abbrev = proto_registrar_get_abbrev(field_id);
int ret = QMessageBox::question(this, wsApp->windowTitleString(tr("Wiki Page for %1").arg(proto_abbrev)),
tr("<p>The Wireshark Wiki is maintained by the community.</p>"
"<p>The page you are about to load might be wonderful, "
"incomplete, wrong, or nonexistent.</p>"
"<p>Proceed to the wiki?</p>"),
QMessageBox::Yes | QMessageBox::No, QMessageBox::Yes);
if (ret != QMessageBox::Yes) return;
QUrl wiki_url = QString(WS_WIKI_URL("Protocols/%1")).arg(proto_abbrev);
QDesktopServices::openUrl(wiki_url);
}
void MainWindow::on_actionContextFilterFieldReference_triggered()
{
QAction *wa = qobject_cast<QAction*>(sender());
if (!wa) return;
bool ok = false;
int field_id = wa->data().toInt(&ok);
if (!ok) return;
const QString proto_abbrev = proto_registrar_get_abbrev(field_id);
QUrl dfref_url = QString(WS_DOCS_URL "/dfref/%1/%2")
.arg(proto_abbrev[0])
.arg(proto_abbrev);
QDesktopServices::openUrl(dfref_url);
}
void MainWindow::on_actionViewFullScreen_triggered(bool checked)
{
if (checked) {
// Save the state for future restore
was_maximized_ = this->isMaximized();
this->showFullScreen();
} else {
// Restore the previous state
if (was_maximized_)
this->showMaximized();
else
this->showNormal();
}
}
void MainWindow::activatePluginIFToolbar(bool)
{
QAction *sendingAction = dynamic_cast<QAction *>(sender());
if (!sendingAction || !sendingAction->data().isValid())
return;
ext_toolbar_t *toolbar = VariantPointer<ext_toolbar_t>::asPtr(sendingAction->data());
QList<QToolBar *> toolbars = findChildren<QToolBar *>();
foreach(QToolBar *bar, toolbars) {
AdditionalToolBar *iftoolbar = dynamic_cast<AdditionalToolBar *>(bar);
if (iftoolbar && iftoolbar->menuName().compare(toolbar->name) == 0) {
if (iftoolbar->isVisible()) {
iftoolbar->setVisible(false);
sendingAction->setChecked(true);
} else {
iftoolbar->setVisible(true);
sendingAction->setChecked(true);
}
}
}
}
void MainWindow::rtpPlayerDialogReplaceRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpPlayerDialog()->replaceRtpStreams(stream_ids);
}
void MainWindow::rtpPlayerDialogAddRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpPlayerDialog()->addRtpStreams(stream_ids);
}
void MainWindow::rtpPlayerDialogRemoveRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpPlayerDialog()->removeRtpStreams(stream_ids);
}
void MainWindow::rtpAnalysisDialogReplaceRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpAnalysisDialog()->replaceRtpStreams(stream_ids);
}
void MainWindow::rtpAnalysisDialogAddRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpAnalysisDialog()->addRtpStreams(stream_ids);
}
void MainWindow::rtpAnalysisDialogRemoveRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpAnalysisDialog()->removeRtpStreams(stream_ids);
}
void MainWindow::rtpStreamsDialogSelectRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpStreamsDialog()->selectRtpStream(stream_ids);
}
void MainWindow::rtpStreamsDialogDeselectRtpStreams(QVector<rtpstream_id_t *> stream_ids)
{
openTelephonyRtpStreamsDialog()->deselectRtpStream(stream_ids);
}
#ifdef _MSC_VER
#pragma warning(pop)
#endif