#include "MainWindow.h" #include "ui_MainWindow.h" #include "dialogs/CreateNewDialog.h" #include "dialogs/CommentsDialog.h" #include "dialogs/AboutDialog.h" #include "dialogs/RenameDialog.h" #include "dialogs/AsmOptionsDialog.h" #include "utils/Helpers.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "utils/Highlighter.h" #include "utils/HexAsciiHighlighter.h" #include "utils/Helpers.h" #include "dialogs/NewFileDialog.h" #include "widgets/PreviewWidget.h" #include "widgets/FunctionsWidget.h" #include "widgets/SectionsWidget.h" #include "widgets/CommentsWidget.h" #include "widgets/ImportsWidget.h" #include "widgets/ExportsWidget.h" #include "widgets/SymbolsWidget.h" #include "widgets/StringsWidget.h" #include "widgets/SectionsDock.h" #include "widgets/RelocsWidget.h" #include "widgets/FlagsWidget.h" #include "widgets/CodeGraphic.h" #include "widgets/Dashboard.h" #include "widgets/Notepad.h" #include "widgets/Sidebar.h" #include "widgets/SdbDock.h" #include "widgets/Omnibar.h" #include "widgets/ConsoleWidget.h" #include "dialogs/OptionsDialog.h" #include "widgets/EntrypointWidget.h" #include "dialogs/SaveProjectDialog.h" // graphics #include #include #include #include static void registerCustomFonts() { int ret = QFontDatabase::addApplicationFont(":/fonts/Anonymous Pro.ttf"); assert(-1 != ret && "unable to register Anonymous Pro.ttf"); ret = QFontDatabase::addApplicationFont(":/fonts/Inconsolata-Regular.ttf"); assert(-1 != ret && "unable to register Inconsolata-Regular.ttf"); // do not issue a warning in release Q_UNUSED(ret) } MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), core(CutterCore::getInstance()), previewDock(nullptr), notepadDock(nullptr), asmDock(nullptr), calcDock(nullptr), omnibar(nullptr), ui(new Ui::MainWindow), highlighter(nullptr), hex_highlighter(nullptr), graphicsBar(nullptr), entrypointDock(nullptr), functionsDock(nullptr), importsDock(nullptr), exportsDock(nullptr), symbolsDock(nullptr), relocsDock(nullptr), commentsDock(nullptr), stringsDock(nullptr), flagsDock(nullptr), dashboardDock(nullptr), gotoEntry(nullptr), sdbDock(nullptr), sectionsDock(nullptr), consoleWidget(nullptr) { doLock = false; configuration = new Configuration(); } MainWindow::~MainWindow() { delete core; } void MainWindow::initUI() { ui->setupUi(this); registerCustomFonts(); /* * Toolbar */ // Hide central tab widget tabs QTabBar *centralbar = ui->centralTabWidget->tabBar(); centralbar->setVisible(false); consoleWidget = new ConsoleWidget(this); ui->tabVerticalLayout->addWidget(consoleWidget); // Sepparator between back/forward and undo/redo buttons QWidget *spacer4 = new QWidget(); spacer4->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding); spacer4->setMinimumSize(10, 10); ui->mainToolBar->insertWidget(ui->actionForward, spacer4); // Popup menu on theme toolbar button QToolButton *backButton = new QToolButton(this); backButton->setIcon(QIcon(":/img/icons/arrow_left.svg")); //backButton->setPopupMode(QToolButton::DelayedPopup); ui->mainToolBar->insertWidget(ui->actionForward, backButton); connect(backButton, SIGNAL(clicked()), this, SLOT(backButton_clicked())); // Sepparator between undo/redo and goto lineEdit QWidget *spacer3 = new QWidget(); spacer3->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); spacer3->setMinimumSize(20, 20); spacer3->setMaximumWidth(300); ui->mainToolBar->addWidget(spacer3); // Omnibar LineEdit this->omnibar = new Omnibar(this); ui->mainToolBar->addWidget(this->omnibar); // Add special separators to the toolbar that expand to separate groups of elements QWidget *spacer2 = new QWidget(); spacer2->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); spacer2->setMinimumSize(10, 10); spacer2->setMaximumWidth(300); ui->mainToolBar->addWidget(spacer2); // Separator between back/forward and undo/redo buttons QWidget *spacer = new QWidget(); spacer->setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Expanding); spacer->setMinimumSize(20, 20); ui->mainToolBar->addWidget(spacer); // codeGraphics tool bar this->graphicsBar = new GraphicsBar(this); this->graphicsBar->setMovable(false); addToolBarBreak(Qt::TopToolBarArea); addToolBar(graphicsBar); /* * Dock Widgets */ dockWidgets.reserve(14); // Add Memory DockWidget this->previewDock = new PreviewWidget(tr("Preview"), this); dockWidgets.push_back(previewDock); // To use in the future when we handle more than one memory views // this->previewDock->setAttribute(Qt::WA_DeleteOnClose); // this->add_debug_output( QString::number(this->dockList.length()) ); // Add disassembly view (dockable) this->disassemblyDock = new DisassemblyWidget(tr("Disassembly"), this); dockWidgets.push_back(disassemblyDock); sidebarDock = new SidebarWidget(tr("Sidebar"), this); dockWidgets.push_back(sidebarDock); hexdumpDock = new HexdumpWidget(tr("Hexdump"), this); dockWidgets.push_back(hexdumpDock); // Add graph view as dockable graphDock = new QDockWidget(tr("Graph"), this); graphDock->setObjectName("Graph"); graphDock->setAllowedAreas(Qt::AllDockWidgetAreas); graphView = new DisassemblerGraphView(graphDock); graphDock->setWidget(graphView); connect(graphDock, &QDockWidget::visibilityChanged, graphDock, [](bool visibility) { if (visibility) { Core()->setMemoryWidgetPriority(CutterCore::MemoryWidgetType::Graph); } }); connect(Core(), &CutterCore::raisePrioritizedMemoryWidget, graphDock, [ = ](CutterCore::MemoryWidgetType type) { if (type == CutterCore::MemoryWidgetType::Graph) { graphDock->raise(); graphView->setFocus(); } }); dockWidgets.push_back(graphDock); // Add Sections dock panel this->sectionsDock = new SectionsDock(this); dockWidgets.push_back(sectionsDock); // Add entrypoint DockWidget this->entrypointDock = new EntrypointWidget(this); dockWidgets.push_back(entrypointDock); // Add functions DockWidget this->functionsDock = new FunctionsWidget(this); dockWidgets.push_back(functionsDock); // Add imports DockWidget this->importsDock = new ImportsWidget(this); dockWidgets.push_back(importsDock); // Add exports DockWidget this->exportsDock = new ExportsWidget(this); dockWidgets.push_back(exportsDock); // Add symbols DockWidget this->symbolsDock = new SymbolsWidget(this); dockWidgets.push_back(symbolsDock); // Add relocs DockWidget this->relocsDock = new RelocsWidget(this); dockWidgets.push_back(relocsDock); // Add comments DockWidget this->commentsDock = new CommentsWidget(this); dockWidgets.push_back(commentsDock); // Add strings DockWidget this->stringsDock = new StringsWidget(this); dockWidgets.push_back(stringsDock); // Add flags DockWidget this->flagsDock = new FlagsWidget(this); dockWidgets.push_back(flagsDock); // Add Notepad Dock panel this->notepadDock = new Notepad(this); dockWidgets.push_back(notepadDock); connect(previewDock, SIGNAL(fontChanged(QFont)), notepadDock, SLOT(setFonts(QFont))); //Add Dashboard Dock panel this->dashboardDock = new Dashboard(this); dockWidgets.push_back(dashboardDock); // Set up dock widgets default layout resetToDefaultLayout(); // Restore saved settings this->readSettings(); // TODO: Allow the user to select this option visually in the GUI settings // Adjust the DockWidget areas setCorner(Qt::TopLeftCorner, Qt::LeftDockWidgetArea); //setCorner( Qt::TopRightCorner, Qt::RightDockWidgetArea ); setCorner(Qt::BottomLeftCorner, Qt::LeftDockWidgetArea); //setCorner( Qt::BottomRightCorner, Qt::RightDockWidgetArea ); // Show dashboard by default this->dashboardDock->raise(); //qDebug() << "FOLDER: " << QStandardPaths::writableLocation(QStandardPaths::GenericDataLocation); /* * Some global shortcuts */ // Period goes to command entry QShortcut *cmd_shortcut = new QShortcut(QKeySequence(Qt::Key_Period), this); connect(cmd_shortcut, SIGNAL(activated()), consoleWidget, SLOT(focusInputLineEdit())); // G and S goes to goto entry QShortcut *goto_shortcut = new QShortcut(QKeySequence(Qt::Key_G), this); connect(goto_shortcut, SIGNAL(activated()), this->omnibar, SLOT(setFocus())); QShortcut *seek_shortcut = new QShortcut(QKeySequence(Qt::Key_S), this); connect(seek_shortcut, SIGNAL(activated()), this->omnibar, SLOT(setFocus())); // : goes to goto entry QShortcut *commands_shortcut = new QShortcut(QKeySequence(Qt::Key_Colon), this); connect(commands_shortcut, SIGNAL(activated()), this->omnibar, SLOT(showCommands())); QShortcut *refresh_shortcut = new QShortcut(QKeySequence(QKeySequence::Refresh), this); connect(refresh_shortcut, SIGNAL(activated()), this, SLOT(refreshAll())); connect(core, SIGNAL(projectSaved(const QString &)), this, SLOT(projectSaved(const QString &))); } void MainWindow::openNewFile(const QString &fn, int anal_level, QList advanced) { setFilename(fn); OptionsDialog *o = new OptionsDialog(this); o->setAttribute(Qt::WA_DeleteOnClose); o->show(); if (anal_level >= 0) o->setupAndStartAnalysis(anal_level, advanced); } void MainWindow::openProject(const QString &project_name) { QString filename = core->cmd("Pi " + project_name); setFilename(filename.trimmed()); core->openProject(project_name); initUI(); finalizeOpen(); } void MainWindow::finalizeOpen() { core->getOpcodes(); // Set settings to override any incorrect saved in the project core->setSettings(); addOutput(tr(" > Populating UI")); // FIXME: initialization order frakup. the next line is needed so that the // comments widget displays the function names. core->cmd("fs sections"); refreshAll(); if (core->getNotes().isEmpty()) { core->setNotes(tr("# Binary information\n\n") + core->cmd("i") + "\n" + core->cmd("ie") + "\n" + core->cmd("iM") + "\n"); } //Get binary beginning/end addresses this->core->binStart = this->core->cmd("?v $M"); this->core->binEnd = this->core->cmd("?v $M+$s"); addOutput(tr(" > Finished, happy reversing :)")); // Add fortune message addOutput("\n" + core->cmd("fo")); //previewDock->setWindowTitle("entry0"); showMaximized(); // Initialize syntax highlighters notepadDock->highlightPreview(); } bool MainWindow::saveProject(bool quit) { QString projectName = core->getConfig("prj.name"); if (projectName.isEmpty()) { return saveProjectAs(quit); } else { core->saveProject(projectName); return true; } } bool MainWindow::saveProjectAs(bool quit) { SaveProjectDialog dialog(quit, this); int result = dialog.exec(); return !quit || result != SaveProjectDialog::Rejected; } void MainWindow::toggleTheme() { if (QSettings().value("dark").toBool()) { def_theme(); } else { dark(); } } void MainWindow::refreshOmniBar(const QStringList &flags) { omnibar->refresh(flags); } void MainWindow::setFilename(const QString &fn) { // Add file name to window title this->filename = fn; this->setWindowTitle(APPNAME" - " + fn); } void MainWindow::closeEvent(QCloseEvent *event) { QMessageBox::StandardButton ret = QMessageBox::question(this, APPNAME, tr("Do you really want to exit?\nSave your project before closing!"), (QMessageBox::StandardButtons)(QMessageBox::Save | QMessageBox::Discard | QMessageBox::Cancel)); //qDebug() << ret; if (ret == QMessageBox::Save) { if (saveProject(true)) { QSettings settings; settings.setValue("geometry", saveGeometry()); settings.setValue("size", size()); settings.setValue("pos", pos()); settings.setValue("state", saveState()); QMainWindow::closeEvent(event); } else { event->ignore(); } } else if (ret == QMessageBox::Discard) { QSettings settings; settings.setValue("geometry", saveGeometry()); settings.setValue("size", size()); settings.setValue("pos", pos()); settings.setValue("state", saveState()); } else { event->ignore(); } } void MainWindow::readSettings() { QSettings settings; QByteArray geo = settings.value("geometry", QByteArray()).toByteArray(); restoreGeometry(geo); QByteArray state = settings.value("state", QByteArray()).toByteArray(); restoreState(state); if (settings.value("dark").toBool()) { this->dark(); } this->responsive = settings.value("responsive").toBool(); } void MainWindow::dark() { qApp->setStyleSheet("QPlainTextEdit { background-color: rgb(64, 64, 64); color: rgb(222, 222, 222);} QTextEdit { background-color: rgb(64, 64, 64); color: rgb(222, 222, 222);} "); // Call Config() rather than QSettings QSettings settings; settings.setValue("dark", true); // TODO: emit a signal for theme } void MainWindow::def_theme() { qApp->setStyleSheet(""); QSettings settings; settings.setValue("dark", false); // TODO: emit a signal for theme } void MainWindow::refreshAll() { Core()->triggerRefreshAll(); } void MainWindow::on_actionLock_triggered() { doLock = !doLock; if (doLock) { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::NoDockWidgetFeatures); } } else { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::AllDockWidgetFeatures); } } } void MainWindow::lockUnlock_Docks(bool what) { if (what) { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::NoDockWidgetFeatures); } } else { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::AllDockWidgetFeatures); } } } void MainWindow::on_actionLockUnlock_triggered() { if (ui->actionLockUnlock->isChecked()) { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::NoDockWidgetFeatures); } ui->actionLockUnlock->setIcon(QIcon(":/lock")); } else { foreach (QDockWidget *dockWidget, findChildren()) { dockWidget->setFeatures(QDockWidget::AllDockWidgetFeatures); } ui->actionLockUnlock->setIcon(QIcon(":/unlock")); } } void MainWindow::on_actionTabs_triggered() { if (ui->centralTabWidget->tabPosition() == QTabWidget::South) { ui->centralTabWidget->setTabPosition(QTabWidget::North); this->setTabPosition(Qt::AllDockWidgetAreas, QTabWidget::North); } else { ui->centralTabWidget->setTabPosition(QTabWidget::South); this->setTabPosition(Qt::AllDockWidgetAreas, QTabWidget::South); } } void MainWindow::on_actionMem_triggered() { //this->previewDock->show(); //this->previewDock->raise(); PreviewWidget *newMemDock = new PreviewWidget(); this->dockWidgets << newMemDock; newMemDock->setAttribute(Qt::WA_DeleteOnClose); this->tabifyDockWidget(this->previewDock, newMemDock); //newMemDock->refreshDisasm(); } void MainWindow::on_actionEntry_points_triggered() { toggleDockWidget(entrypointDock); } void MainWindow::on_actionFunctions_triggered() { toggleDockWidget(functionsDock); } void MainWindow::on_actionImports_triggered() { toggleDockWidget(importsDock); } void MainWindow::on_actionExports_triggered() { toggleDockWidget(exportsDock); } void MainWindow::on_actionSymbols_triggered() { toggleDockWidget(symbolsDock); } void MainWindow::on_actionReloc_triggered() { toggleDockWidget(relocsDock); } void MainWindow::on_actionStrings_triggered() { toggleDockWidget(stringsDock); } void MainWindow::on_actionSections_triggered() { toggleDockWidget(sectionsDock); } void MainWindow::on_actionFlags_triggered() { toggleDockWidget(flagsDock); } void MainWindow::on_actionComents_triggered() { toggleDockWidget(commentsDock); } void MainWindow::on_actionNotepad_triggered() { toggleDockWidget(notepadDock); } void MainWindow::on_actionAbout_triggered() { AboutDialog *a = new AboutDialog(this); a->open(); } void MainWindow::on_actionRefresh_Panels_triggered() { this->refreshAll(); } void MainWindow::toggleDockWidget(QDockWidget *dock_widget) { if (dock_widget->isVisible()) { dock_widget->close(); } else { dock_widget->show(); dock_widget->raise(); } } void MainWindow::backButton_clicked() { core->seekPrev(); } void MainWindow::on_actionForward_triggered() { core->seekNext(); } void MainWindow::on_actionCreate_File_triggered() { CreateNewDialog *n = new CreateNewDialog(this); n->exec(); } void MainWindow::on_actionDisasAdd_comment_triggered() { CommentsDialog *c = new CommentsDialog(this); c->exec(); } void MainWindow::restoreDocks() { // bottom right addDockWidget(Qt::RightDockWidgetArea, sectionsDock); // left addDockWidget(Qt::TopDockWidgetArea, functionsDock); // center splitDockWidget(functionsDock, dashboardDock, Qt::Horizontal); // right (sidebar) splitDockWidget(dashboardDock, sidebarDock, Qt::Horizontal); // tabs for center (must be applied after splitDockWidget()) tabifyDockWidget(sectionsDock, commentsDock); tabifyDockWidget(dashboardDock, disassemblyDock); tabifyDockWidget(dashboardDock, graphDock); tabifyDockWidget(dashboardDock, hexdumpDock); tabifyDockWidget(dashboardDock, previewDock); tabifyDockWidget(dashboardDock, entrypointDock); tabifyDockWidget(dashboardDock, flagsDock); tabifyDockWidget(dashboardDock, stringsDock); tabifyDockWidget(dashboardDock, relocsDock); tabifyDockWidget(dashboardDock, importsDock); tabifyDockWidget(dashboardDock, exportsDock); tabifyDockWidget(dashboardDock, symbolsDock); tabifyDockWidget(dashboardDock, notepadDock); dashboardDock->raise(); sectionsDock->raise(); } void MainWindow::hideAllDocks() { for (auto w : dockWidgets) { w->hide(); } } void MainWindow::showDefaultDocks() { const QList defaultDocks = { sectionsDock, entrypointDock, functionsDock, previewDock, commentsDock, stringsDock, importsDock, symbolsDock, notepadDock, graphDock, disassemblyDock, sidebarDock, hexdumpDock, dashboardDock }; for (auto w : dockWidgets) { if (defaultDocks.contains(w)) { w->show(); } } } void MainWindow::resetToDefaultLayout() { restoreDocks(); hideAllDocks(); showDefaultDocks(); dashboardDock->raise(); // ugly workaround to set the default widths of functions and sidebar docks // if anyone finds a way to do this cleaner that also works, feel free to change it! auto restoreFunctionDock = qhelpers::forceWidth(functionsDock->widget(), 300); auto restoreSidebarDock = qhelpers::forceWidth(sidebarDock->widget(), 300); qApp->processEvents(); restoreFunctionDock.restoreWidth(functionsDock->widget()); restoreSidebarDock.restoreWidth(sidebarDock->widget()); Core()->setMemoryWidgetPriority(CutterCore::MemoryWidgetType::Disassembly); } void MainWindow::on_actionDefaut_triggered() { resetToDefaultLayout(); } void MainWindow::on_actionhide_bottomPannel_triggered() { if (ui->centralWidget->isVisible()) { ui->centralWidget->hide(); } else { ui->centralWidget->show(); } } void MainWindow::sendToNotepad(const QString &txt) { core->setNotes(core->getNotes() + "```\n" + txt + "\n```"); } void MainWindow::on_actionFunctionsRename_triggered() { RenameDialog *r = new RenameDialog(this); // Get function based on click position //r->setFunctionName(fcn_name); r->open(); } void MainWindow::addOutput(const QString &msg) { consoleWidget->addOutput(msg); } void MainWindow::addDebugOutput(const QString &msg) { printf("debug output: %s\n", msg.toLocal8Bit().constData()); consoleWidget->addDebugOutput(msg); } void MainWindow::on_actionNew_triggered() { if (close()) on_actionLoad_triggered(); } void MainWindow::on_actionSave_triggered() { saveProject(); } void MainWindow::on_actionSaveAs_triggered() { saveProjectAs(); } void MainWindow::on_actionRun_Script_triggered() { QFileDialog dialog(this); dialog.setFileMode(QFileDialog::ExistingFile); dialog.setViewMode(QFileDialog::Detail); dialog.setDirectory(QDir::home()); QString fileName; fileName = dialog.getOpenFileName(this, tr("Select radare2 script")); if (!fileName.length()) //cancel was pressed return; this->core->cmd(". " + fileName); } void MainWindow::on_actionDark_Theme_triggered() { this->dark(); } void MainWindow::on_actionWhite_Theme_triggered() { this->def_theme(); } void MainWindow::on_actionSDBBrowser_triggered() { this->sdbDock = new SdbDock(this); this->tabifyDockWidget(this->previewDock, this->sdbDock); this->sdbDock->setFloating(true); this->sdbDock->show(); } void MainWindow::on_actionLoad_triggered() { QProcess process(this); process.setEnvironment(QProcess::systemEnvironment()); process.startDetached(qApp->applicationFilePath()); } void MainWindow::on_actionDashboard_triggered() { if (this->dashboardDock->isVisible()) { this->dashboardDock->close(); } else { this->dashboardDock->show(); this->dashboardDock->raise(); } } void MainWindow::toggleResponsive(bool maybe) { this->responsive = maybe; // Save options in settings QSettings settings; settings.setValue("responsive", this->responsive); } void MainWindow::on_actionTabs_on_Top_triggered() { this->on_actionTabs_triggered(); } void MainWindow::on_actionReset_settings_triggered() { QMessageBox::StandardButton ret = (QMessageBox::StandardButton)QMessageBox::question(this, APPNAME, tr("Do you really want to clear all settings?"), QMessageBox::Ok | QMessageBox::Cancel); if (ret == QMessageBox::Ok) { // Save options in settings QSettings settings; settings.clear(); } } void MainWindow::on_actionQuit_triggered() { close(); } void MainWindow::on_actionRefresh_contents_triggered() { refreshAll(); } void MainWindow::on_actionAsmOptions_triggered() { auto dialog = new AsmOptionsDialog(this); dialog->show(); } void MainWindow::projectSaved(const QString &name) { this->addOutput(tr("Project saved: ") + name); }