| 
									
										
										
										
											2015-01-04 09:36:57 -08:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 09:56:32 +00:00
										 |  |  | #include <cinttypes>
 | 
					
						
							| 
									
										
										
										
											2016-03-06 14:04:47 +01:00
										 |  |  | #include <clocale>
 | 
					
						
							| 
									
										
										
										
											2016-04-05 13:29:55 +01:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  | #include <glad/glad.h>
 | 
					
						
							|  |  |  | #define QT_NO_OPENGL
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | #include <QDesktopWidget>
 | 
					
						
							|  |  |  | #include <QFileDialog>
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | #include <QMessageBox>
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <QtGui>
 | 
					
						
							| 
									
										
										
										
											2017-01-22 18:46:48 +01:00
										 |  |  | #include <QtWidgets>
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "citra_qt/bootmanager.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-21 23:49:36 -05:00
										 |  |  | #include "citra_qt/configuration/config.h"
 | 
					
						
							|  |  |  | #include "citra_qt/configuration/configure_dialog.h"
 | 
					
						
							| 
									
										
										
										
											2016-12-21 17:19:12 -05:00
										 |  |  | #include "citra_qt/debugger/graphics/graphics.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/graphics/graphics_breakpoints.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/graphics/graphics_cmdlists.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/graphics/graphics_surface.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/graphics/graphics_tracing.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/graphics/graphics_vertex_shader.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "citra_qt/debugger/profiler.h"
 | 
					
						
							|  |  |  | #include "citra_qt/debugger/registers.h"
 | 
					
						
							| 
									
										
										
										
											2016-04-08 19:28:54 +03:00
										 |  |  | #include "citra_qt/debugger/wait_tree.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-21 00:21:23 +09:00
										 |  |  | #include "citra_qt/game_list.h"
 | 
					
						
							|  |  |  | #include "citra_qt/hotkeys.h"
 | 
					
						
							|  |  |  | #include "citra_qt/main.h"
 | 
					
						
							|  |  |  | #include "citra_qt/ui_settings.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "common/logging/backend.h"
 | 
					
						
							|  |  |  | #include "common/logging/filter.h"
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/logging/text_formatter.h"
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include "common/platform.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:58:59 +01:00
										 |  |  | #include "common/scm_rev.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include "common/scope_exit.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2015-10-21 22:19:55 -04:00
										 |  |  | #include "core/gdbstub/gdbstub.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "core/loader/loader.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2015-05-18 21:21:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-01 15:13:35 -06:00
										 |  |  | #ifdef QT_STATICPLUGIN
 | 
					
						
							|  |  |  | Q_IMPORT_PLUGIN(QWindowsIntegrationPlugin); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-08 20:06:25 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * "Callouts" are one-time instructional messages shown to the user. In the config settings, there | 
					
						
							|  |  |  |  * is a bitfield "callout_flags" options, used to track if a message has already been shown to the | 
					
						
							|  |  |  |  * user. This is 32-bits - if we have more than 32 callouts, we should retire and recyle old ones. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | enum class CalloutFlag : uint32_t { | 
					
						
							|  |  |  |     Telemetry = 0x1, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void ShowCalloutMessage(const QString& message, CalloutFlag flag) { | 
					
						
							|  |  |  |     if (UISettings::values.callout_flags & static_cast<uint32_t>(flag)) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UISettings::values.callout_flags |= static_cast<uint32_t>(flag); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMessageBox msg; | 
					
						
							|  |  |  |     msg.setText(message); | 
					
						
							|  |  |  |     msg.setStandardButtons(QMessageBox::Ok); | 
					
						
							|  |  |  |     msg.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding); | 
					
						
							|  |  |  |     msg.setStyleSheet("QLabel{min-width: 900px;}"); | 
					
						
							|  |  |  |     msg.exec(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::ShowCallouts() { | 
					
						
							|  |  |  |     static const QString telemetry_message = | 
					
						
							|  |  |  |         tr("To help improve Citra, the Citra Team collects anonymous usage data. No private or " | 
					
						
							|  |  |  |            "personally identifying information is collected. This data helps us to understand how " | 
					
						
							|  |  |  |            "people use Citra and prioritize our efforts. Furthermore, it helps us to more easily " | 
					
						
							|  |  |  |            "identify emulation bugs and performance issues. This data includes:<ul><li>Information" | 
					
						
							|  |  |  |            " about the version of Citra you are using</li><li>Performance data about the games you " | 
					
						
							|  |  |  |            "play</li><li>Your configuration settings</li><li>Information about your computer " | 
					
						
							|  |  |  |            "hardware</li><li>Emulation errors and crash information</li></ul>By default, this " | 
					
						
							|  |  |  |            "feature is enabled. To disable this feature, click 'Emulation' from the menu and then " | 
					
						
							|  |  |  |            "select 'Configure...'. Then, on the 'Web' tab, uncheck 'Share anonymous usage data with" | 
					
						
							|  |  |  |            " the Citra team'. <br/><br/>By using this software, you agree to the above terms.<br/>" | 
					
						
							|  |  |  |            "<br/><a href='https://citra-emu.org/entry/telemetry-and-why-thats-a-good-thing/'>Learn " | 
					
						
							|  |  |  |            "more</a>"); | 
					
						
							|  |  |  |     ShowCalloutMessage(telemetry_message, CalloutFlag::Telemetry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | GMainWindow::GMainWindow() : config(new Config()), emu_thread(nullptr) { | 
					
						
							| 
									
										
										
										
											2014-10-25 18:02:26 +02:00
										 |  |  |     Pica::g_debug_context = Pica::DebugContext::Construct(); | 
					
						
							| 
									
										
										
										
											2017-02-15 21:23:30 -06:00
										 |  |  |     setAcceptDrops(true); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     ui.setupUi(this); | 
					
						
							|  |  |  |     statusBar()->hide(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     InitializeWidgets(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     InitializeDebugWidgets(); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     InitializeRecentFileMenuActions(); | 
					
						
							|  |  |  |     InitializeHotkeys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SetDefaultUIGeometry(); | 
					
						
							|  |  |  |     RestoreUIState(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  |     ConnectMenuEvents(); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     ConnectWidgetEvents(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-15 11:43:22 -07:00
										 |  |  |     setWindowTitle(QString("Citra %1| %2-%3") | 
					
						
							|  |  |  |                        .arg(Common::g_build_name, Common::g_scm_branch, Common::g_scm_desc)); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     show(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     game_list->PopulateAsync(UISettings::values.gamedir, UISettings::values.gamedir_deepscan); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-24 02:41:11 +02:00
										 |  |  |     UpdateUITheme(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-08 20:06:25 -04:00
										 |  |  |     // Show one-time "callout" messages to the user
 | 
					
						
							|  |  |  |     ShowCallouts(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     QStringList args = QApplication::arguments(); | 
					
						
							|  |  |  |     if (args.length() >= 2) { | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |         BootGame(args[1]); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GMainWindow::~GMainWindow() { | 
					
						
							|  |  |  |     // will get automatically deleted otherwise
 | 
					
						
							|  |  |  |     if (render_window->parent() == nullptr) | 
					
						
							|  |  |  |         delete render_window; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Pica::g_debug_context.reset(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::InitializeWidgets() { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     render_window = new GRenderWindow(this, emu_thread.get()); | 
					
						
							| 
									
										
										
										
											2014-04-21 23:15:17 -04:00
										 |  |  |     render_window->hide(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  |     game_list = new GameList(this); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  |     ui.horizontalLayout->addWidget(game_list); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Create status bar
 | 
					
						
							| 
									
										
										
										
											2017-04-13 01:10:19 -04:00
										 |  |  |     message_label = new QLabel(); | 
					
						
							|  |  |  |     // Configured separately for left alignment
 | 
					
						
							|  |  |  |     message_label->setVisible(false); | 
					
						
							|  |  |  |     message_label->setFrameStyle(QFrame::NoFrame); | 
					
						
							|  |  |  |     message_label->setContentsMargins(4, 0, 4, 0); | 
					
						
							|  |  |  |     message_label->setAlignment(Qt::AlignLeft); | 
					
						
							|  |  |  |     statusBar()->addPermanentWidget(message_label, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     emu_speed_label = new QLabel(); | 
					
						
							| 
									
										
										
										
											2017-02-19 19:09:46 -08:00
										 |  |  |     emu_speed_label->setToolTip(tr("Current emulation speed. Values higher or lower than 100% " | 
					
						
							|  |  |  |                                    "indicate emulation is running faster or slower than a 3DS.")); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     game_fps_label = new QLabel(); | 
					
						
							| 
									
										
										
										
											2017-02-19 19:09:46 -08:00
										 |  |  |     game_fps_label->setToolTip(tr("How many frames per second the game is currently displaying. " | 
					
						
							|  |  |  |                                   "This will vary from game to game and scene to scene.")); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     emu_frametime_label = new QLabel(); | 
					
						
							| 
									
										
										
										
											2017-02-19 19:09:46 -08:00
										 |  |  |     emu_frametime_label->setToolTip( | 
					
						
							|  |  |  |         tr("Time taken to emulate a 3DS frame, not counting framelimiting or v-sync. For " | 
					
						
							|  |  |  |            "full-speed emulation this should be at most 16.67 ms.")); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (auto& label : {emu_speed_label, game_fps_label, emu_frametime_label}) { | 
					
						
							|  |  |  |         label->setVisible(false); | 
					
						
							| 
									
										
										
										
											2017-02-20 16:53:40 -08:00
										 |  |  |         label->setFrameStyle(QFrame::NoFrame); | 
					
						
							|  |  |  |         label->setContentsMargins(4, 0, 4, 0); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:10:19 -04:00
										 |  |  |         statusBar()->addPermanentWidget(label, 0); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     } | 
					
						
							|  |  |  |     statusBar()->setVisible(true); | 
					
						
							| 
									
										
										
										
											2017-04-12 22:14:52 -03:00
										 |  |  |     setStyleSheet("QStatusBar::item{border: none;}"); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::InitializeDebugWidgets() { | 
					
						
							|  |  |  |     connect(ui.action_Create_Pica_Surface_Viewer, &QAction::triggered, this, | 
					
						
							|  |  |  |             &GMainWindow::OnCreateGraphicsSurfaceViewer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QMenu* debug_menu = ui.menu_View_Debugging; | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #if MICROPROFILE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     microProfileDialog = new MicroProfileDialog(this); | 
					
						
							|  |  |  |     microProfileDialog->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(microProfileDialog->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-18 18:30:53 -04:00
										 |  |  |     registersWidget = new RegistersWidget(this); | 
					
						
							|  |  |  |     addDockWidget(Qt::RightDockWidgetArea, registersWidget); | 
					
						
							|  |  |  |     registersWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(registersWidget->toggleViewAction()); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStarting, registersWidget, | 
					
						
							|  |  |  |             &RegistersWidget::OnEmulationStarting); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStopping, registersWidget, | 
					
						
							|  |  |  |             &RegistersWidget::OnEmulationStopping); | 
					
						
							| 
									
										
										
										
											2014-04-18 18:30:53 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-17 22:38:10 +02:00
										 |  |  |     graphicsWidget = new GPUCommandStreamWidget(this); | 
					
						
							|  |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsWidget); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     graphicsWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(graphicsWidget->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2014-05-17 22:38:10 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-18 17:52:22 +02:00
										 |  |  |     graphicsCommandsWidget = new GPUCommandListWidget(this); | 
					
						
							|  |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsCommandsWidget); | 
					
						
							| 
									
										
										
										
											2014-08-14 19:21:55 +02:00
										 |  |  |     graphicsCommandsWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(graphicsCommandsWidget->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2014-05-18 17:52:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     graphicsBreakpointsWidget = new GraphicsBreakPointsWidget(Pica::g_debug_context, this); | 
					
						
							| 
									
										
										
										
											2014-10-25 20:28:24 +02:00
										 |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsBreakpointsWidget); | 
					
						
							|  |  |  |     graphicsBreakpointsWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(graphicsBreakpointsWidget->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2014-10-25 20:28:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     graphicsVertexShaderWidget = new GraphicsVertexShaderWidget(Pica::g_debug_context, this); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsVertexShaderWidget); | 
					
						
							|  |  |  |     graphicsVertexShaderWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(graphicsVertexShaderWidget->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2014-12-10 19:24:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     graphicsTracingWidget = new GraphicsTracingWidget(Pica::g_debug_context, this); | 
					
						
							| 
									
										
										
										
											2015-04-04 12:57:31 +02:00
										 |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsTracingWidget); | 
					
						
							|  |  |  |     graphicsTracingWidget->hide(); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     debug_menu->addAction(graphicsTracingWidget->toggleViewAction()); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStarting, graphicsTracingWidget, | 
					
						
							|  |  |  |             &GraphicsTracingWidget::OnEmulationStarting); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStopping, graphicsTracingWidget, | 
					
						
							|  |  |  |             &GraphicsTracingWidget::OnEmulationStopping); | 
					
						
							| 
									
										
										
										
											2015-04-04 12:57:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-08 19:28:54 +03:00
										 |  |  |     waitTreeWidget = new WaitTreeWidget(this); | 
					
						
							|  |  |  |     addDockWidget(Qt::LeftDockWidgetArea, waitTreeWidget); | 
					
						
							|  |  |  |     waitTreeWidget->hide(); | 
					
						
							|  |  |  |     debug_menu->addAction(waitTreeWidget->toggleViewAction()); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:16:24 -08:00
										 |  |  |     connect(this, &GMainWindow::EmulationStarting, waitTreeWidget, | 
					
						
							|  |  |  |             &WaitTreeWidget::OnEmulationStarting); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStopping, waitTreeWidget, | 
					
						
							|  |  |  |             &WaitTreeWidget::OnEmulationStopping); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::InitializeRecentFileMenuActions() { | 
					
						
							|  |  |  |     for (int i = 0; i < max_recent_files_item; ++i) { | 
					
						
							|  |  |  |         actions_recent_files[i] = new QAction(this); | 
					
						
							|  |  |  |         actions_recent_files[i]->setVisible(false); | 
					
						
							|  |  |  |         connect(actions_recent_files[i], SIGNAL(triggered()), this, SLOT(OnMenuRecentFile())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ui.menu_recent_files->addAction(actions_recent_files[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UpdateRecentFiles(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::InitializeHotkeys() { | 
					
						
							|  |  |  |     RegisterHotkey("Main Window", "Load File", QKeySequence::Open); | 
					
						
							|  |  |  |     RegisterHotkey("Main Window", "Swap Screens", QKeySequence::NextChild); | 
					
						
							|  |  |  |     RegisterHotkey("Main Window", "Start Emulation"); | 
					
						
							|  |  |  |     LoadHotkeys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(GetHotkey("Main Window", "Load File", this), SIGNAL(activated()), this, | 
					
						
							|  |  |  |             SLOT(OnMenuLoadFile())); | 
					
						
							|  |  |  |     connect(GetHotkey("Main Window", "Start Emulation", this), SIGNAL(activated()), this, | 
					
						
							|  |  |  |             SLOT(OnStartGame())); | 
					
						
							|  |  |  |     connect(GetHotkey("Main Window", "Swap Screens", render_window), SIGNAL(activated()), this, | 
					
						
							|  |  |  |             SLOT(OnSwapScreens())); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | void GMainWindow::SetDefaultUIGeometry() { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // geometry: 55% of the window contents are in the upper screen half, 45% in the lower half
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     const QRect screenRect = QApplication::desktop()->screenGeometry(this); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const int w = screenRect.width() * 2 / 3; | 
					
						
							|  |  |  |     const int h = screenRect.height() / 2; | 
					
						
							|  |  |  |     const int x = (screenRect.x() + screenRect.width()) / 2 - w / 2; | 
					
						
							|  |  |  |     const int y = (screenRect.y() + screenRect.height()) / 2 - h * 55 / 100; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     setGeometry(x, y, w, h); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | void GMainWindow::RestoreUIState() { | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     restoreGeometry(UISettings::values.geometry); | 
					
						
							|  |  |  |     restoreState(UISettings::values.state); | 
					
						
							|  |  |  |     render_window->restoreGeometry(UISettings::values.renderwindow_geometry); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #if MICROPROFILE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     microProfileDialog->restoreGeometry(UISettings::values.microprofile_geometry); | 
					
						
							|  |  |  |     microProfileDialog->setVisible(UISettings::values.microprofile_visible); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-09-06 23:51:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     game_list->LoadInterfaceLayout(); | 
					
						
							| 
									
										
										
										
											2015-09-02 08:56:38 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     ui.action_Single_Window_Mode->setChecked(UISettings::values.single_window_mode); | 
					
						
							| 
									
										
										
										
											2014-04-21 23:15:17 -04:00
										 |  |  |     ToggleWindowMode(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 01:11:57 -08:00
										 |  |  |     ui.action_Display_Dock_Widget_Headers->setChecked(UISettings::values.display_titlebar); | 
					
						
							|  |  |  |     OnDisplayTitleBars(ui.action_Display_Dock_Widget_Headers->isChecked()); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  |     ui.action_Show_Filter_Bar->setChecked(UISettings::values.show_filter_bar); | 
					
						
							|  |  |  |     game_list->setFilterVisible(ui.action_Show_Filter_Bar->isChecked()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     ui.action_Show_Status_Bar->setChecked(UISettings::values.show_status_bar); | 
					
						
							|  |  |  |     statusBar()->setVisible(ui.action_Show_Status_Bar->isChecked()); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-01-06 16:09:30 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | void GMainWindow::ConnectWidgetEvents() { | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  |     connect(game_list, SIGNAL(GameChosen(QString)), this, SLOT(OnGameListLoadFile(QString))); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:56:32 +00:00
										 |  |  |     connect(game_list, SIGNAL(OpenSaveFolderRequested(u64)), this, | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  |             SLOT(OnGameListOpenSaveFolder(u64))); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     connect(this, SIGNAL(EmulationStarting(EmuThread*)), render_window, | 
					
						
							|  |  |  |             SLOT(OnEmulationStarting(EmuThread*))); | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |     connect(this, SIGNAL(EmulationStopping()), render_window, SLOT(OnEmulationStopping())); | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     connect(&status_bar_update_timer, &QTimer::timeout, this, &GMainWindow::UpdateStatusBar); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  | void GMainWindow::ConnectMenuEvents() { | 
					
						
							|  |  |  |     // File
 | 
					
						
							|  |  |  |     connect(ui.action_Load_File, &QAction::triggered, this, &GMainWindow::OnMenuLoadFile); | 
					
						
							|  |  |  |     connect(ui.action_Select_Game_List_Root, &QAction::triggered, this, | 
					
						
							|  |  |  |             &GMainWindow::OnMenuSelectGameListRoot); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:30:29 -08:00
										 |  |  |     connect(ui.action_Exit, &QAction::triggered, this, &QMainWindow::close); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Emulation
 | 
					
						
							|  |  |  |     connect(ui.action_Start, &QAction::triggered, this, &GMainWindow::OnStartGame); | 
					
						
							|  |  |  |     connect(ui.action_Pause, &QAction::triggered, this, &GMainWindow::OnPauseGame); | 
					
						
							|  |  |  |     connect(ui.action_Stop, &QAction::triggered, this, &GMainWindow::OnStopGame); | 
					
						
							|  |  |  |     connect(ui.action_Configure, &QAction::triggered, this, &GMainWindow::OnConfigure); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // View
 | 
					
						
							|  |  |  |     connect(ui.action_Single_Window_Mode, &QAction::triggered, this, | 
					
						
							|  |  |  |             &GMainWindow::ToggleWindowMode); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:30:29 -08:00
										 |  |  |     connect(ui.action_Display_Dock_Widget_Headers, &QAction::triggered, this, | 
					
						
							|  |  |  |             &GMainWindow::OnDisplayTitleBars); | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  |     ui.action_Show_Filter_Bar->setShortcut(tr("CTRL+F")); | 
					
						
							|  |  |  |     connect(ui.action_Show_Filter_Bar, &QAction::triggered, this, &GMainWindow::OnToggleFilterBar); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     connect(ui.action_Show_Status_Bar, &QAction::triggered, statusBar(), &QStatusBar::setVisible); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GMainWindow::OnDisplayTitleBars(bool show) { | 
					
						
							| 
									
										
										
										
											2015-01-06 16:09:30 +01:00
										 |  |  |     QList<QDockWidget*> widgets = findChildren<QDockWidget*>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (show) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         for (QDockWidget* widget : widgets) { | 
					
						
							| 
									
										
										
										
											2015-01-06 16:09:30 +01:00
										 |  |  |             QWidget* old = widget->titleBarWidget(); | 
					
						
							|  |  |  |             widget->setTitleBarWidget(nullptr); | 
					
						
							|  |  |  |             if (old != nullptr) | 
					
						
							|  |  |  |                 delete old; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         for (QDockWidget* widget : widgets) { | 
					
						
							| 
									
										
										
										
											2015-01-06 16:09:30 +01:00
										 |  |  |             QWidget* old = widget->titleBarWidget(); | 
					
						
							|  |  |  |             widget->setTitleBarWidget(new QWidget()); | 
					
						
							|  |  |  |             if (old != nullptr) | 
					
						
							|  |  |  |                 delete old; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  | bool GMainWindow::LoadROM(const QString& filename) { | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  |     // Shutdown previous session if the emu thread is still active...
 | 
					
						
							|  |  |  |     if (emu_thread != nullptr) | 
					
						
							|  |  |  |         ShutdownGame(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  |     render_window->InitRenderTarget(); | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  |     render_window->MakeCurrent(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  |     if (!gladLoadGL()) { | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |         QMessageBox::critical(this, tr("Error while initializing OpenGL 3.3 Core!"), | 
					
						
							| 
									
										
										
										
											2017-08-04 23:11:27 -04:00
										 |  |  |                               tr("Your GPU may not support OpenGL 3.3, or you do not " | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |                                  "have the latest graphics driver.")); | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |     Core::System& system{Core::System::GetInstance()}; | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |     const Core::System::ResultStatus result{system.Load(render_window, filename.toStdString())}; | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-22 22:47:56 -04:00
										 |  |  |     Core::Telemetry().AddField(Telemetry::FieldType::App, "Frontend", "Qt"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     if (result != Core::System::ResultStatus::Success) { | 
					
						
							|  |  |  |         switch (result) { | 
					
						
							|  |  |  |         case Core::System::ResultStatus::ErrorGetLoader: | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |             LOG_CRITICAL(Frontend, "Failed to obtain loader for %s!", | 
					
						
							|  |  |  |                          filename.toStdString().c_str()); | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             QMessageBox::critical(this, tr("Error while loading ROM!"), | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |                                   tr("The ROM format is not supported.")); | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2016-11-19 20:40:04 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |         case Core::System::ResultStatus::ErrorSystemMode: | 
					
						
							|  |  |  |             LOG_CRITICAL(Frontend, "Failed to load ROM!"); | 
					
						
							|  |  |  |             QMessageBox::critical(this, tr("Error while loading ROM!"), | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |                                   tr("Could not determine the system mode.")); | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2016-01-07 18:36:10 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |         case Core::System::ResultStatus::ErrorLoader_ErrorEncrypted: { | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             QMessageBox::critical( | 
					
						
							|  |  |  |                 this, tr("Error while loading ROM!"), | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |                 tr("The game that you are trying to load must be decrypted before being used with " | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |                    "Citra. A real 3DS is required.<br/><br/>" | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |                    "For more information on dumping and decrypting games, please see the following " | 
					
						
							|  |  |  |                    "wiki pages: <ul>" | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |                    "<li><a href='https://citra-emu.org/wiki/dumping-game-cartridges/'>Dumping Game " | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |                    "Cartridges</a></li>" | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |                    "<li><a href='https://citra-emu.org/wiki/dumping-installed-titles/'>Dumping " | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |                    "Installed Titles</a></li>" | 
					
						
							|  |  |  |                    "</ul>")); | 
					
						
							| 
									
										
										
										
											2016-01-07 18:36:10 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |         case Core::System::ResultStatus::ErrorLoader_ErrorInvalidFormat: | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |             QMessageBox::critical(this, tr("Error while loading ROM!"), | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |                                   tr("The ROM format is not supported.")); | 
					
						
							| 
									
										
										
										
											2016-01-07 18:36:10 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |         case Core::System::ResultStatus::ErrorVideoCore: | 
					
						
							|  |  |  |             QMessageBox::critical( | 
					
						
							|  |  |  |                 this, tr("An error occured in the video core."), | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |                 tr("Citra has encountered an error while running the video core, please see the " | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |                    "log for more details." | 
					
						
							|  |  |  |                    "For more information on accessing the log, please see the following page: " | 
					
						
							|  |  |  |                    "<a href='https://community.citra-emu.org/t/how-to-upload-the-log-file/296'>How " | 
					
						
							|  |  |  |                    "to " | 
					
						
							|  |  |  |                    "Upload the Log File</a>." | 
					
						
							|  |  |  |                    "Ensure that you have the latest graphics drivers for your GPU.")); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-07 18:36:10 +01:00
										 |  |  |         default: | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             QMessageBox::critical( | 
					
						
							|  |  |  |                 this, tr("Error while loading ROM!"), | 
					
						
							|  |  |  |                 tr("An unknown error occured. Please see the log for more details.")); | 
					
						
							| 
									
										
										
										
											2016-01-07 18:36:10 +01:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  | void GMainWindow::BootGame(const QString& filename) { | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |     LOG_INFO(Frontend, "Citra starting..."); | 
					
						
							| 
									
										
										
										
											2016-03-06 11:22:45 +01:00
										 |  |  |     StoreRecentFile(filename); // Put the filename on top of the list
 | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-19 20:40:04 -05:00
										 |  |  |     if (!LoadROM(filename)) | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     // Create and start the emulation thread
 | 
					
						
							| 
									
										
										
										
											2016-04-05 13:29:55 +01:00
										 |  |  |     emu_thread = std::make_unique<EmuThread>(render_window); | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |     emit EmulationStarting(emu_thread.get()); | 
					
						
							| 
									
										
										
										
											2015-05-18 21:24:43 -07:00
										 |  |  |     render_window->moveContext(); | 
					
						
							| 
									
										
										
										
											2015-04-16 18:35:09 -04:00
										 |  |  |     emu_thread->start(); | 
					
						
							| 
									
										
										
										
											2014-04-21 23:15:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-05 12:29:44 +02:00
										 |  |  |     connect(render_window, SIGNAL(Closed()), this, SLOT(OnStopGame())); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // BlockingQueuedConnection is important here, it makes sure we've finished refreshing our views
 | 
					
						
							|  |  |  |     // before the CPU continues
 | 
					
						
							|  |  |  |     connect(emu_thread.get(), SIGNAL(DebugModeEntered()), registersWidget, | 
					
						
							|  |  |  |             SLOT(OnDebugModeEntered()), Qt::BlockingQueuedConnection); | 
					
						
							| 
									
										
										
										
											2016-04-08 19:28:54 +03:00
										 |  |  |     connect(emu_thread.get(), SIGNAL(DebugModeEntered()), waitTreeWidget, | 
					
						
							|  |  |  |             SLOT(OnDebugModeEntered()), Qt::BlockingQueuedConnection); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     connect(emu_thread.get(), SIGNAL(DebugModeLeft()), registersWidget, SLOT(OnDebugModeLeft()), | 
					
						
							|  |  |  |             Qt::BlockingQueuedConnection); | 
					
						
							| 
									
										
										
										
											2016-04-08 19:28:54 +03:00
										 |  |  |     connect(emu_thread.get(), SIGNAL(DebugModeLeft()), waitTreeWidget, SLOT(OnDebugModeLeft()), | 
					
						
							|  |  |  |             Qt::BlockingQueuedConnection); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Update the GUI
 | 
					
						
							|  |  |  |     registersWidget->OnDebugModeEntered(); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  |     if (ui.action_Single_Window_Mode->isChecked()) { | 
					
						
							|  |  |  |         game_list->hide(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-02-20 16:36:44 -08:00
										 |  |  |     status_bar_update_timer.start(2000); | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-21 23:15:17 -04:00
										 |  |  |     render_window->show(); | 
					
						
							| 
									
										
										
										
											2016-12-18 14:58:28 +05:30
										 |  |  |     render_window->setFocus(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |     emulation_running = true; | 
					
						
							| 
									
										
										
										
											2014-10-30 22:44:51 -07:00
										 |  |  |     OnStartGame(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-27 23:13:57 -04:00
										 |  |  | void GMainWindow::ShutdownGame() { | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |     emu_thread->RequestStop(); | 
					
						
							| 
									
										
										
										
											2015-04-27 23:13:57 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     // Release emu threads from any breakpoints
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |     // This belongs after RequestStop() and before wait() because if emulation stops on a GPU
 | 
					
						
							|  |  |  |     // breakpoint after (or before) RequestStop() is called, the emulation would never be able
 | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     // to continue out to the main loop and terminate. Thus wait() would hang forever.
 | 
					
						
							|  |  |  |     // TODO(bunnei): This function is not thread safe, but it's being used as if it were
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     Pica::g_debug_context->ClearBreakpoints(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |     emit EmulationStopping(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Wait for emulation thread to complete and delete it
 | 
					
						
							|  |  |  |     emu_thread->wait(); | 
					
						
							|  |  |  |     emu_thread = nullptr; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-05 12:29:44 +02:00
										 |  |  |     // The emulation is stopped, so closing the window or not does not matter anymore
 | 
					
						
							|  |  |  |     disconnect(render_window, SIGNAL(Closed()), this, SLOT(OnStopGame())); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     // Update the GUI
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:58:26 -04:00
										 |  |  |     ui.action_Start->setEnabled(false); | 
					
						
							| 
									
										
										
										
											2015-07-26 16:38:51 +02:00
										 |  |  |     ui.action_Start->setText(tr("Start")); | 
					
						
							| 
									
										
										
										
											2015-04-27 23:13:57 -04:00
										 |  |  |     ui.action_Pause->setEnabled(false); | 
					
						
							|  |  |  |     ui.action_Stop->setEnabled(false); | 
					
						
							|  |  |  |     render_window->hide(); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  |     game_list->show(); | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  |     game_list->setFilterFocus(); | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  |     // Disable status bar updates
 | 
					
						
							|  |  |  |     status_bar_update_timer.stop(); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:10:19 -04:00
										 |  |  |     message_label->setVisible(false); | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  |     emu_speed_label->setVisible(false); | 
					
						
							|  |  |  |     game_fps_label->setVisible(false); | 
					
						
							|  |  |  |     emu_frametime_label->setVisible(false); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |     emulation_running = false; | 
					
						
							| 
									
										
										
										
											2015-04-27 23:13:57 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  | void GMainWindow::StoreRecentFile(const QString& filename) { | 
					
						
							|  |  |  |     UISettings::values.recent_files.prepend(filename); | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.recent_files.removeDuplicates(); | 
					
						
							|  |  |  |     while (UISettings::values.recent_files.size() > max_recent_files_item) { | 
					
						
							|  |  |  |         UISettings::values.recent_files.removeLast(); | 
					
						
							| 
									
										
										
										
											2015-09-07 22:00:08 -03:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 22:50:52 +02:00
										 |  |  |     UpdateRecentFiles(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::UpdateRecentFiles() { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     unsigned int num_recent_files = | 
					
						
							|  |  |  |         std::min(UISettings::values.recent_files.size(), static_cast<int>(max_recent_files_item)); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned int i = 0; i < num_recent_files; i++) { | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         QString text = QString("&%1. %2").arg(i + 1).arg( | 
					
						
							|  |  |  |             QFileInfo(UISettings::values.recent_files[i]).fileName()); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  |         actions_recent_files[i]->setText(text); | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |         actions_recent_files[i]->setData(UISettings::values.recent_files[i]); | 
					
						
							|  |  |  |         actions_recent_files[i]->setToolTip(UISettings::values.recent_files[i]); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  |         actions_recent_files[i]->setVisible(true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (int j = num_recent_files; j < max_recent_files_item; ++j) { | 
					
						
							|  |  |  |         actions_recent_files[j]->setVisible(false); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Grey out the recent files menu if the list is empty
 | 
					
						
							|  |  |  |     if (num_recent_files == 0) { | 
					
						
							|  |  |  |         ui.menu_recent_files->setEnabled(false); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         ui.menu_recent_files->setEnabled(true); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | void GMainWindow::OnGameListLoadFile(QString game_path) { | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |     BootGame(game_path); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-15 09:56:32 +00:00
										 |  |  | void GMainWindow::OnGameListOpenSaveFolder(u64 program_id) { | 
					
						
							| 
									
										
										
										
											2017-10-12 21:21:49 -04:00
										 |  |  |     UNIMPLEMENTED(); | 
					
						
							| 
									
										
										
										
											2016-12-15 09:56:32 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::OnMenuLoadFile() { | 
					
						
							| 
									
										
										
										
											2017-02-12 21:28:56 +01:00
										 |  |  |     QString extensions; | 
					
						
							|  |  |  |     for (const auto& piece : game_list->supported_file_extensions) | 
					
						
							|  |  |  |         extensions += "*." + piece + " "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:38:05 -08:00
										 |  |  |     QString file_filter = tr("3DS Executable") + " (" + extensions + ")"; | 
					
						
							|  |  |  |     file_filter += ";;" + tr("All Files (*.*)"); | 
					
						
							| 
									
										
										
										
											2017-02-12 21:28:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     QString filename = QFileDialog::getOpenFileName(this, tr("Load File"), | 
					
						
							|  |  |  |                                                     UISettings::values.roms_path, file_filter); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  |     if (!filename.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |         UISettings::values.roms_path = QFileInfo(filename).path(); | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |         BootGame(filename); | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-06 15:33:57 -07:00
										 |  |  | void GMainWindow::OnMenuSelectGameListRoot() { | 
					
						
							|  |  |  |     QString dir_path = QFileDialog::getExistingDirectory(this, tr("Select Directory")); | 
					
						
							|  |  |  |     if (!dir_path.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2016-01-24 21:54:04 +01:00
										 |  |  |         UISettings::values.gamedir = dir_path; | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |         game_list->PopulateAsync(dir_path, UISettings::values.gamedir_deepscan); | 
					
						
							| 
									
										
										
										
											2015-09-06 15:33:57 -07:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::OnMenuRecentFile() { | 
					
						
							|  |  |  |     QAction* action = qobject_cast<QAction*>(sender()); | 
					
						
							|  |  |  |     assert(action); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QString filename = action->data().toString(); | 
					
						
							|  |  |  |     QFileInfo file_info(filename); | 
					
						
							|  |  |  |     if (file_info.exists()) { | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |         BootGame(filename); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // Display an error message and remove the file from the list.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         QMessageBox::information(this, tr("File not found"), | 
					
						
							|  |  |  |                                  tr("File \"%1\" not found").arg(filename)); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |         UISettings::values.recent_files.removeOne(filename); | 
					
						
							| 
									
										
										
										
											2015-08-17 22:50:52 +02:00
										 |  |  |         UpdateRecentFiles(); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::OnStartGame() { | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     emu_thread->SetRunning(true); | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |     qRegisterMetaType<Core::System::ResultStatus>("Core::System::ResultStatus"); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:18:54 -04:00
										 |  |  |     qRegisterMetaType<std::string>("std::string"); | 
					
						
							|  |  |  |     connect(emu_thread.get(), SIGNAL(ErrorThrown(Core::System::ResultStatus, std::string)), this, | 
					
						
							|  |  |  |             SLOT(OnCoreError(Core::System::ResultStatus, std::string))); | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ui.action_Start->setEnabled(false); | 
					
						
							| 
									
										
										
										
											2015-07-26 16:38:51 +02:00
										 |  |  |     ui.action_Start->setText(tr("Continue")); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  |     ui.action_Pause->setEnabled(true); | 
					
						
							|  |  |  |     ui.action_Stop->setEnabled(true); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::OnPauseGame() { | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |     emu_thread->SetRunning(false); | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ui.action_Start->setEnabled(true); | 
					
						
							|  |  |  |     ui.action_Pause->setEnabled(false); | 
					
						
							|  |  |  |     ui.action_Stop->setEnabled(true); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  | void GMainWindow::OnStopGame() { | 
					
						
							| 
									
										
										
										
											2015-04-27 23:13:57 -04:00
										 |  |  |     ShutdownGame(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 18:35:09 -04:00
										 |  |  | void GMainWindow::ToggleWindowMode() { | 
					
						
							|  |  |  |     if (ui.action_Single_Window_Mode->isChecked()) { | 
					
						
							|  |  |  |         // Render in the main window...
 | 
					
						
							| 
									
										
										
										
											2014-04-21 23:15:17 -04:00
										 |  |  |         render_window->BackupGeometry(); | 
					
						
							|  |  |  |         ui.horizontalLayout->addWidget(render_window); | 
					
						
							| 
									
										
										
										
											2014-12-26 19:42:27 +01:00
										 |  |  |         render_window->setFocusPolicy(Qt::ClickFocus); | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |         if (emulation_running) { | 
					
						
							|  |  |  |             render_window->setVisible(true); | 
					
						
							|  |  |  |             render_window->setFocus(); | 
					
						
							| 
									
										
										
										
											2015-10-06 12:20:26 -07:00
										 |  |  |             game_list->hide(); | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-04-16 18:35:09 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // Render in a separate window...
 | 
					
						
							|  |  |  |         ui.horizontalLayout->removeWidget(render_window); | 
					
						
							|  |  |  |         render_window->setParent(nullptr); | 
					
						
							|  |  |  |         render_window->setFocusPolicy(Qt::NoFocus); | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |         if (emulation_running) { | 
					
						
							|  |  |  |             render_window->setVisible(true); | 
					
						
							|  |  |  |             render_window->RestoreGeometry(); | 
					
						
							| 
									
										
										
										
											2015-08-31 21:35:33 -07:00
										 |  |  |             game_list->show(); | 
					
						
							| 
									
										
										
										
											2015-08-31 18:30:06 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::OnConfigure() { | 
					
						
							| 
									
										
										
										
											2016-09-02 20:18:45 +08:00
										 |  |  |     ConfigureDialog configureDialog(this); | 
					
						
							| 
									
										
										
										
											2016-01-24 18:34:05 +01:00
										 |  |  |     auto result = configureDialog.exec(); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     if (result == QDialog::Accepted) { | 
					
						
							| 
									
										
										
										
											2016-01-24 18:34:05 +01:00
										 |  |  |         configureDialog.applyConfiguration(); | 
					
						
							| 
									
										
										
										
											2017-06-24 02:41:11 +02:00
										 |  |  |         UpdateUITheme(); | 
					
						
							| 
									
										
										
										
											2016-01-24 21:54:04 +01:00
										 |  |  |         config->Save(); | 
					
						
							| 
									
										
										
										
											2016-01-24 18:34:05 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  | void GMainWindow::OnToggleFilterBar() { | 
					
						
							|  |  |  |     game_list->setFilterVisible(ui.action_Show_Filter_Bar->isChecked()); | 
					
						
							|  |  |  |     if (ui.action_Show_Filter_Bar->isChecked()) { | 
					
						
							|  |  |  |         game_list->setFilterFocus(); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         game_list->clearFilter(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 02:58:11 -06:00
										 |  |  | void GMainWindow::OnSwapScreens() { | 
					
						
							|  |  |  |     Settings::values.swap_screen = !Settings::values.swap_screen; | 
					
						
							|  |  |  |     Settings::Apply(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-09 18:23:15 +02:00
										 |  |  | void GMainWindow::OnCreateGraphicsSurfaceViewer() { | 
					
						
							|  |  |  |     auto graphicsSurfaceViewerWidget = new GraphicsSurfaceWidget(Pica::g_debug_context, this); | 
					
						
							|  |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsSurfaceViewerWidget); | 
					
						
							|  |  |  |     // TODO: Maybe graphicsSurfaceViewerWidget->setFloating(true);
 | 
					
						
							|  |  |  |     graphicsSurfaceViewerWidget->show(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  | void GMainWindow::UpdateStatusBar() { | 
					
						
							|  |  |  |     if (emu_thread == nullptr) { | 
					
						
							|  |  |  |         status_bar_update_timer.stop(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto results = Core::System::GetInstance().GetAndResetPerfStats(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-19 18:56:26 -08:00
										 |  |  |     emu_speed_label->setText(tr("Speed: %1%").arg(results.emulation_speed * 100.0, 0, 'f', 0)); | 
					
						
							|  |  |  |     game_fps_label->setText(tr("Game: %1 FPS").arg(results.game_fps, 0, 'f', 0)); | 
					
						
							| 
									
										
										
										
											2017-02-19 14:34:47 -08:00
										 |  |  |     emu_frametime_label->setText(tr("Frame: %1 ms").arg(results.frametime * 1000.0, 0, 'f', 2)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     emu_speed_label->setVisible(true); | 
					
						
							|  |  |  |     game_fps_label->setVisible(true); | 
					
						
							|  |  |  |     emu_frametime_label->setVisible(true); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-13 01:18:54 -04:00
										 |  |  | void GMainWindow::OnCoreError(Core::System::ResultStatus result, std::string details) { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     QMessageBox::StandardButton answer; | 
					
						
							|  |  |  |     QString status_message; | 
					
						
							|  |  |  |     const QString common_message = | 
					
						
							|  |  |  |         tr("The game you are trying to load requires additional files from your 3DS to be dumped " | 
					
						
							|  |  |  |            "before playing.<br/><br/>For more information on dumping these files, please see the " | 
					
						
							|  |  |  |            "following wiki page: <a " | 
					
						
							|  |  |  |            "href='https://citra-emu.org/wiki/" | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |            "dumping-system-archives-and-the-shared-fonts-from-a-3ds-console/'>Dumping System " | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |            "Archives and the Shared Fonts from a 3DS Console</a>.<br/><br/>Would you like to quit " | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |            "back to the game list? Continuing emulation may result in crashes, corrupted save " | 
					
						
							|  |  |  |            "data, or other bugs."); | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |     switch (result) { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     case Core::System::ResultStatus::ErrorSystemFiles: { | 
					
						
							|  |  |  |         QString message = "Citra was unable to locate a 3DS system archive"; | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         if (!details.empty()) { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:18:54 -04:00
										 |  |  |             message.append(tr(": %1. ").arg(details.c_str())); | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |             message.append(". "); | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |         message.append(common_message); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         answer = QMessageBox::question(this, tr("System Archive Not Found"), message, | 
					
						
							|  |  |  |                                        QMessageBox::Yes | QMessageBox::No, QMessageBox::No); | 
					
						
							|  |  |  |         status_message = "System Archive Missing"; | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     case Core::System::ResultStatus::ErrorSharedFont: { | 
					
						
							|  |  |  |         QString message = tr("Citra was unable to locate the 3DS shared fonts. "); | 
					
						
							|  |  |  |         message.append(common_message); | 
					
						
							|  |  |  |         answer = QMessageBox::question(this, tr("Shared Fonts Not Found"), message, | 
					
						
							|  |  |  |                                        QMessageBox::Yes | QMessageBox::No, QMessageBox::No); | 
					
						
							|  |  |  |         status_message = "Shared Font Missing"; | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |     default: | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |         answer = QMessageBox::question( | 
					
						
							|  |  |  |             this, tr("Fatal Error"), | 
					
						
							|  |  |  |             tr("Citra has encountered a fatal error, please see the log for more details. " | 
					
						
							|  |  |  |                "For more information on accessing the log, please see the following page: " | 
					
						
							|  |  |  |                "<a href='https://community.citra-emu.org/t/how-to-upload-the-log-file/296'>How to " | 
					
						
							| 
									
										
										
										
											2017-05-25 16:49:46 -04:00
										 |  |  |                "Upload the Log File</a>.<br/><br/>Would you like to quit back to the game list? " | 
					
						
							|  |  |  |                "Continuing emulation may result in crashes, corrupted save data, or other bugs."), | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |             QMessageBox::Yes | QMessageBox::No, QMessageBox::No); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:18:54 -04:00
										 |  |  |         status_message = "Fatal Error encountered"; | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (answer == QMessageBox::Yes) { | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         if (emu_thread) { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |             ShutdownGame(); | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |         // Only show the message if the game is still running.
 | 
					
						
							|  |  |  |         if (emu_thread) { | 
					
						
							|  |  |  |             message_label->setText(status_message); | 
					
						
							|  |  |  |             message_label->setVisible(true); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-10 13:31:20 +01:00
										 |  |  | bool GMainWindow::ConfirmClose() { | 
					
						
							| 
									
										
										
										
											2016-01-24 21:54:04 +01:00
										 |  |  |     if (emu_thread == nullptr || !UISettings::values.confirm_before_closing) | 
					
						
							| 
									
										
										
										
											2016-01-13 18:40:41 +01:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2016-01-10 13:31:20 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     QMessageBox::StandardButton answer = | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |         QMessageBox::question(this, tr("Citra"), tr("Are you sure you want to close Citra?"), | 
					
						
							|  |  |  |                               QMessageBox::Yes | QMessageBox::No, QMessageBox::No); | 
					
						
							|  |  |  |     return answer != QMessageBox::No; | 
					
						
							| 
									
										
										
										
											2016-01-10 13:31:20 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | void GMainWindow::closeEvent(QCloseEvent* event) { | 
					
						
							| 
									
										
										
										
											2016-01-10 13:31:20 +01:00
										 |  |  |     if (!ConfirmClose()) { | 
					
						
							|  |  |  |         event->ignore(); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.geometry = saveGeometry(); | 
					
						
							|  |  |  |     UISettings::values.state = saveState(); | 
					
						
							|  |  |  |     UISettings::values.renderwindow_geometry = render_window->saveGeometry(); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #if MICROPROFILE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.microprofile_geometry = microProfileDialog->saveGeometry(); | 
					
						
							|  |  |  |     UISettings::values.microprofile_visible = microProfileDialog->isVisible(); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.single_window_mode = ui.action_Single_Window_Mode->isChecked(); | 
					
						
							| 
									
										
										
										
											2017-02-18 01:11:57 -08:00
										 |  |  |     UISettings::values.display_titlebar = ui.action_Display_Dock_Widget_Headers->isChecked(); | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  |     UISettings::values.show_filter_bar = ui.action_Show_Filter_Bar->isChecked(); | 
					
						
							| 
									
										
										
										
											2017-02-18 12:09:14 -08:00
										 |  |  |     UISettings::values.show_status_bar = ui.action_Show_Status_Bar->isChecked(); | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.first_start = false; | 
					
						
							| 
									
										
										
										
											2015-09-07 22:11:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     game_list->SaveInterfaceLayout(); | 
					
						
							|  |  |  |     SaveHotkeys(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-12 23:14:24 -04:00
										 |  |  |     // Shutdown session if the emu thread is active...
 | 
					
						
							|  |  |  |     if (emu_thread != nullptr) | 
					
						
							|  |  |  |         ShutdownGame(); | 
					
						
							| 
									
										
										
										
											2015-05-01 16:53:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     render_window->close(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QWidget::closeEvent(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:41:04 -08:00
										 |  |  | static bool IsSingleFileDropEvent(QDropEvent* event) { | 
					
						
							| 
									
										
										
										
											2017-02-15 21:23:30 -06:00
										 |  |  |     const QMimeData* mimeData = event->mimeData(); | 
					
						
							|  |  |  |     return mimeData->hasUrls() && mimeData->urls().length() == 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::dropEvent(QDropEvent* event) { | 
					
						
							|  |  |  |     if (IsSingleFileDropEvent(event) && ConfirmChangeGame()) { | 
					
						
							|  |  |  |         const QMimeData* mimeData = event->mimeData(); | 
					
						
							|  |  |  |         QString filename = mimeData->urls().at(0).toLocalFile(); | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  |         BootGame(filename); | 
					
						
							| 
									
										
										
										
											2017-02-15 21:23:30 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::dragEnterEvent(QDragEnterEvent* event) { | 
					
						
							|  |  |  |     if (IsSingleFileDropEvent(event)) { | 
					
						
							|  |  |  |         event->acceptProposedAction(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::dragMoveEvent(QDragMoveEvent* event) { | 
					
						
							|  |  |  |     event->acceptProposedAction(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GMainWindow::ConfirmChangeGame() { | 
					
						
							|  |  |  |     if (emu_thread == nullptr) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     auto answer = QMessageBox::question( | 
					
						
							|  |  |  |         this, tr("Citra"), | 
					
						
							|  |  |  |         tr("Are you sure you want to stop the emulation? Any unsaved progress will be lost."), | 
					
						
							|  |  |  |         QMessageBox::Yes | QMessageBox::No, QMessageBox::No); | 
					
						
							|  |  |  |     return answer != QMessageBox::No; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-04-30 04:04:39 +02:00
										 |  |  | void GMainWindow::filterBarSetChecked(bool state) { | 
					
						
							|  |  |  |     ui.action_Show_Filter_Bar->setChecked(state); | 
					
						
							|  |  |  |     emit(OnToggleFilterBar()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-24 02:41:11 +02:00
										 |  |  | void GMainWindow::UpdateUITheme() { | 
					
						
							|  |  |  |     if (UISettings::values.theme != UISettings::themes[0].second) { | 
					
						
							|  |  |  |         QString theme_uri(":" + UISettings::values.theme + "/style.qss"); | 
					
						
							|  |  |  |         QFile f(theme_uri); | 
					
						
							|  |  |  |         if (!f.exists()) { | 
					
						
							|  |  |  |             LOG_ERROR(Frontend, "Unable to set style, stylesheet file not found"); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             f.open(QFile::ReadOnly | QFile::Text); | 
					
						
							|  |  |  |             QTextStream ts(&f); | 
					
						
							|  |  |  |             qApp->setStyleSheet(ts.readAll()); | 
					
						
							|  |  |  |             GMainWindow::setStyleSheet(ts.readAll()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         qApp->setStyleSheet(""); | 
					
						
							|  |  |  |         GMainWindow::setStyleSheet(""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | #ifdef main
 | 
					
						
							|  |  |  | #undef main
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | int main(int argc, char* argv[]) { | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  |     Log::Filter log_filter(Log::Level::Info); | 
					
						
							| 
									
										
										
										
											2015-03-06 19:15:02 +01:00
										 |  |  |     Log::SetFilter(&log_filter); | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("Frontend"); | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     SCOPE_EXIT({ MicroProfileShutdown(); }); | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-26 17:13:02 +02:00
										 |  |  |     // Init settings params
 | 
					
						
							|  |  |  |     QCoreApplication::setOrganizationName("Citra team"); | 
					
						
							|  |  |  |     QCoreApplication::setApplicationName("Citra"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     QApplication::setAttribute(Qt::AA_X11InitThreads); | 
					
						
							|  |  |  |     QApplication app(argc, argv); | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // Qt changes the locale and causes issues in float conversion using std::to_string() when
 | 
					
						
							|  |  |  |     // generating shaders
 | 
					
						
							| 
									
										
										
										
											2016-03-06 14:04:47 +01:00
										 |  |  |     setlocale(LC_ALL, "C"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     GMainWindow main_window; | 
					
						
							| 
									
										
										
										
											2014-12-06 20:00:08 -02:00
										 |  |  |     // After settings have been loaded by GMainWindow, apply the filter
 | 
					
						
							|  |  |  |     log_filter.ParseFilterString(Settings::values.log_filter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     main_window.show(); | 
					
						
							|  |  |  |     return app.exec(); | 
					
						
							|  |  |  | } |