| 
									
										
										
										
											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>
 | 
					
						
							| 
									
										
										
										
											2018-08-15 05:38:37 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <fmt/ostream.h>
 | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  | #include <glad/glad.h>
 | 
					
						
							| 
									
										
										
										
											2018-08-15 05:38:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-19 01:31:01 +00:00
										 |  |  | #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>
 | 
					
						
							| 
									
										
										
										
											2018-06-14 17:25:40 -04:00
										 |  |  | #include "common/common_paths.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"
 | 
					
						
							| 
									
										
										
										
											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"
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | #include "core/crypto/key_manager.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  | #include "core/file_sys/card_image.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-03 11:51:48 -04:00
										 |  |  | #include "core/file_sys/vfs_real.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"
 | 
					
						
							| 
									
										
										
										
											2018-03-24 23:35:06 -05:00
										 |  |  | #include "video_core/debug_utils/debug_utils.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-14 19:15:45 +01:00
										 |  |  | #include "yuzu/about_dialog.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | #include "yuzu/bootmanager.h"
 | 
					
						
							|  |  |  | #include "yuzu/configuration/config.h"
 | 
					
						
							|  |  |  | #include "yuzu/configuration/configure_dialog.h"
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #include "yuzu/debugger/console.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-22 15:19:35 -05:00
										 |  |  | #include "yuzu/debugger/graphics/graphics_breakpoints.h"
 | 
					
						
							| 
									
										
										
										
											2018-03-22 15:30:43 -05:00
										 |  |  | #include "yuzu/debugger/graphics/graphics_surface.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | #include "yuzu/debugger/profiler.h"
 | 
					
						
							|  |  |  | #include "yuzu/debugger/wait_tree.h"
 | 
					
						
							|  |  |  | #include "yuzu/game_list.h"
 | 
					
						
							|  |  |  | #include "yuzu/hotkeys.h"
 | 
					
						
							|  |  |  | #include "yuzu/main.h"
 | 
					
						
							|  |  |  | #include "yuzu/ui_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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-19 20:22:26 +02:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  | extern "C" { | 
					
						
							|  |  |  | // tells Nvidia and AMD drivers to use the dedicated GPU by default on laptops with switchable
 | 
					
						
							|  |  |  | // graphics
 | 
					
						
							|  |  |  | __declspec(dllexport) unsigned long NvOptimusEnablement = 0x00000001; | 
					
						
							|  |  |  | __declspec(dllexport) int AmdPowerXpressRequestHighPerformance = 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #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(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-11 21:56:15 -07:00
										 |  |  | void GMainWindow::ShowCallouts() {} | 
					
						
							| 
									
										
										
										
											2017-08-08 20:06:25 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-08 21:44:59 -04:00
										 |  |  | const int GMainWindow::max_recent_files_item; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 11:51:48 -04:00
										 |  |  | GMainWindow::GMainWindow() | 
					
						
							|  |  |  |     : config(new Config()), emu_thread(nullptr), | 
					
						
							|  |  |  |       vfs(std::make_shared<FileSys::RealVfsFilesystem>()) { | 
					
						
							| 
									
										
										
										
											2018-03-22 15:19:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 23:35:06 -05:00
										 |  |  |     debug_context = Tegra::DebugContext::Construct(); | 
					
						
							| 
									
										
										
										
											2018-03-22 15:19:35 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-15 21:23:30 -06:00
										 |  |  |     setAcceptDrops(true); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     ui.setupUi(this); | 
					
						
							|  |  |  |     statusBar()->hide(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 17:38:34 +02:00
										 |  |  |     default_theme_paths = QIcon::themeSearchPaths(); | 
					
						
							|  |  |  |     UpdateUITheme(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2018-07-26 15:51:14 +02:00
										 |  |  |     LOG_INFO(Frontend, "yuzu Version: {} | {}-{}", Common::g_build_name, Common::g_scm_branch, | 
					
						
							|  |  |  |              Common::g_scm_desc); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-11 21:56:15 -07:00
										 |  |  |     setWindowTitle(QString("yuzu %1| %2-%3") | 
					
						
							| 
									
										
										
										
											2017-01-15 11:43:22 -07:00
										 |  |  |                        .arg(Common::g_build_name, Common::g_scm_branch, Common::g_scm_desc)); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     show(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |     // Necessary to load titles from nand in gamelist.
 | 
					
						
							|  |  |  |     Service::FileSystem::RegisterBIS(std::make_unique<FileSys::BISFactory>(vfs->OpenDirectory( | 
					
						
							|  |  |  |         FileUtil::GetUserPath(FileUtil::UserPath::NANDDir), FileSys::Mode::ReadWrite))); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  |     game_list->PopulateAsync(UISettings::values.gamedir, UISettings::values.gamedir_deepscan); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-03 11:51:48 -04:00
										 |  |  |     game_list = new GameList(vfs, 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(); | 
					
						
							| 
									
										
										
										
											2018-01-16 18:05:21 +00:00
										 |  |  |     emu_speed_label->setToolTip( | 
					
						
							|  |  |  |         tr("Current emulation speed. Values higher or lower than 100% " | 
					
						
							|  |  |  |            "indicate emulation is running faster or slower than a Switch.")); | 
					
						
							| 
									
										
										
										
											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( | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |         tr("Time taken to emulate a Switch frame, not counting framelimiting or v-sync. For " | 
					
						
							| 
									
										
										
										
											2017-02-19 19:09:46 -08:00
										 |  |  |            "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() { | 
					
						
							|  |  |  |     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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 23:35:06 -05:00
										 |  |  |     graphicsBreakpointsWidget = new GraphicsBreakPointsWidget(debug_context, this); | 
					
						
							| 
									
										
										
										
											2018-03-22 15:19:35 -05:00
										 |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsBreakpointsWidget); | 
					
						
							|  |  |  |     graphicsBreakpointsWidget->hide(); | 
					
						
							|  |  |  |     debug_menu->addAction(graphicsBreakpointsWidget->toggleViewAction()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 23:35:06 -05:00
										 |  |  |     graphicsSurfaceWidget = new GraphicsSurfaceWidget(debug_context, this); | 
					
						
							| 
									
										
										
										
											2018-03-22 15:30:43 -05:00
										 |  |  |     addDockWidget(Qt::RightDockWidgetArea, graphicsSurfaceWidget); | 
					
						
							|  |  |  |     graphicsSurfaceWidget->hide(); | 
					
						
							|  |  |  |     debug_menu->addAction(graphicsSurfaceWidget->toggleViewAction()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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); | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |         connect(actions_recent_files[i], &QAction::triggered, this, &GMainWindow::OnMenuRecentFile); | 
					
						
							| 
									
										
										
										
											2016-12-19 14:50:58 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ui.menu_recent_files->addAction(actions_recent_files[i]); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UpdateRecentFiles(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::InitializeHotkeys() { | 
					
						
							| 
									
										
										
										
											2018-08-07 00:43:07 -04:00
										 |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Load File", QKeySequence::Open); | 
					
						
							|  |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Start Emulation"); | 
					
						
							|  |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Continue/Pause", QKeySequence(Qt::Key_F4)); | 
					
						
							|  |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Fullscreen", QKeySequence::FullScreen); | 
					
						
							|  |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Exit Fullscreen", QKeySequence(Qt::Key_Escape), | 
					
						
							|  |  |  |                                    Qt::ApplicationShortcut); | 
					
						
							|  |  |  |     hotkey_registry.RegisterHotkey("Main Window", "Toggle Speed Limit", QKeySequence("CTRL+Z"), | 
					
						
							|  |  |  |                                    Qt::ApplicationShortcut); | 
					
						
							|  |  |  |     hotkey_registry.LoadHotkeys(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Load File", this), &QShortcut::activated, | 
					
						
							|  |  |  |             this, &GMainWindow::OnMenuLoadFile); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Start Emulation", this), | 
					
						
							|  |  |  |             &QShortcut::activated, this, &GMainWindow::OnStartGame); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Continue/Pause", this), &QShortcut::activated, | 
					
						
							|  |  |  |             this, [&] { | 
					
						
							|  |  |  |                 if (emulation_running) { | 
					
						
							|  |  |  |                     if (emu_thread->IsRunning()) { | 
					
						
							|  |  |  |                         OnPauseGame(); | 
					
						
							|  |  |  |                     } else { | 
					
						
							|  |  |  |                         OnStartGame(); | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Fullscreen", render_window), | 
					
						
							|  |  |  |             &QShortcut::activated, ui.action_Fullscreen, &QAction::trigger); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Fullscreen", render_window), | 
					
						
							|  |  |  |             &QShortcut::activatedAmbiguously, ui.action_Fullscreen, &QAction::trigger); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Exit Fullscreen", this), | 
					
						
							|  |  |  |             &QShortcut::activated, this, [&] { | 
					
						
							|  |  |  |                 if (emulation_running) { | 
					
						
							|  |  |  |                     ui.action_Fullscreen->setChecked(false); | 
					
						
							|  |  |  |                     ToggleFullscreen(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |     connect(hotkey_registry.GetHotkey("Main Window", "Toggle Speed Limit", this), | 
					
						
							|  |  |  |             &QShortcut::activated, this, [&] { | 
					
						
							|  |  |  |                 Settings::values.toggle_framelimit = !Settings::values.toggle_framelimit; | 
					
						
							|  |  |  |                 UpdateStatusBar(); | 
					
						
							|  |  |  |             }); | 
					
						
							| 
									
										
										
										
											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::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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  |     ui.action_Fullscreen->setChecked(UISettings::values.fullscreen); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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()); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     Debugger::ToggleConsole(); | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(game_list, &GameList::GameChosen, this, &GMainWindow::OnGameListLoadFile); | 
					
						
							|  |  |  |     connect(game_list, &GameList::OpenSaveFolderRequested, this, | 
					
						
							|  |  |  |             &GMainWindow::OnGameListOpenSaveFolder); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(this, &GMainWindow::EmulationStarting, render_window, | 
					
						
							|  |  |  |             &GRenderWindow::OnEmulationStarting); | 
					
						
							|  |  |  |     connect(this, &GMainWindow::EmulationStopping, render_window, | 
					
						
							|  |  |  |             &GRenderWindow::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); | 
					
						
							| 
									
										
										
										
											2018-06-14 12:27:29 -04:00
										 |  |  |     connect(ui.action_Load_Folder, &QAction::triggered, this, &GMainWindow::OnMenuLoadFolder); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |     connect(ui.action_Install_File_NAND, &QAction::triggered, this, | 
					
						
							|  |  |  |             &GMainWindow::OnMenuInstallToNAND); | 
					
						
							| 
									
										
										
										
											2017-02-18 02:26:57 -08:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2018-01-14 19:15:45 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  |     // Fullscreen
 | 
					
						
							| 
									
										
										
										
											2018-08-07 00:43:07 -04:00
										 |  |  |     ui.action_Fullscreen->setShortcut( | 
					
						
							|  |  |  |         hotkey_registry.GetHotkey("Main Window", "Fullscreen", this)->key()); | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  |     connect(ui.action_Fullscreen, &QAction::triggered, this, &GMainWindow::ToggleFullscreen); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 19:15:45 +01:00
										 |  |  |     // Help
 | 
					
						
							|  |  |  |     connect(ui.action_About, &QAction::triggered, this, &GMainWindow::OnAbout); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-30 15:38:22 -05:00
										 |  |  | bool GMainWindow::SupportsRequiredGLExtensions() { | 
					
						
							|  |  |  |     QStringList unsupported_ext; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_program_interface_query) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_program_interface_query"); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_separate_shader_objects) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_separate_shader_objects"); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_vertex_attrib_binding) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_vertex_attrib_binding"); | 
					
						
							| 
									
										
										
										
											2018-07-22 12:05:38 -05:00
										 |  |  |     if (!GLAD_GL_ARB_vertex_type_10f_11f_11f_rev) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_vertex_type_10f_11f_11f_rev"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Extensions required to support some texture formats.
 | 
					
						
							|  |  |  |     if (!GLAD_GL_EXT_texture_compression_s3tc) | 
					
						
							|  |  |  |         unsupported_ext.append("EXT_texture_compression_s3tc"); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_texture_compression_rgtc) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_texture_compression_rgtc"); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_texture_compression_bptc) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_texture_compression_bptc"); | 
					
						
							|  |  |  |     if (!GLAD_GL_ARB_depth_buffer_float) | 
					
						
							|  |  |  |         unsupported_ext.append("ARB_depth_buffer_float"); | 
					
						
							| 
									
										
										
										
											2018-05-30 15:38:22 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (const QString& ext : unsupported_ext) | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_CRITICAL(Frontend, "Unsupported GL extension: {}", ext.toStdString()); | 
					
						
							| 
									
										
										
										
											2018-05-30 15:38:22 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return unsupported_ext.empty(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-05-30 15:38:22 -05:00
										 |  |  |     if (!SupportsRequiredGLExtensions()) { | 
					
						
							|  |  |  |         QMessageBox::critical( | 
					
						
							|  |  |  |             this, tr("Error while initializing OpenGL Core!"), | 
					
						
							|  |  |  |             tr("Your GPU may not support one or more required OpenGL extensions. Please " | 
					
						
							|  |  |  |                "ensure you have the latest graphics driver. See the log for more details.")); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-17 01:20:47 -05:00
										 |  |  |     Core::System& system{Core::System::GetInstance()}; | 
					
						
							| 
									
										
										
										
											2018-08-03 11:51:48 -04:00
										 |  |  |     system.SetFilesystem(vfs); | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-24 23:35:06 -05:00
										 |  |  |     system.SetGPUDebugContext(debug_context); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-01 20:59:42 -04:00
										 |  |  |     const Core::System::ResultStatus result{system.Load(*render_window, filename.toStdString())}; | 
					
						
							| 
									
										
										
										
											2014-04-03 21:24:07 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 14:10:16 -04:00
										 |  |  |     render_window->DoneCurrent(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     if (result != Core::System::ResultStatus::Success) { | 
					
						
							|  |  |  |         switch (result) { | 
					
						
							|  |  |  |         case Core::System::ResultStatus::ErrorGetLoader: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |             LOG_CRITICAL(Frontend, "Failed to obtain loader for {}!", filename.toStdString()); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  |         case Core::System::ResultStatus::ErrorSystemMode: | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |             LOG_CRITICAL(Frontend, "Failed to load ROM!"); | 
					
						
							| 
									
										
										
										
											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("Could not determine the system mode.")); | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |         case Core::System::ResultStatus::ErrorVideoCore: | 
					
						
							|  |  |  |             QMessageBox::critical( | 
					
						
							| 
									
										
										
										
											2018-08-03 12:55:58 -04:00
										 |  |  |                 this, tr("An error occurred initializing the video core."), | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |                 tr("yuzu 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: | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |             if (static_cast<u32>(result) > | 
					
						
							|  |  |  |                 static_cast<u32>(Core::System::ResultStatus::ErrorLoader)) { | 
					
						
							|  |  |  |                 LOG_CRITICAL(Frontend, "Failed to load ROM!"); | 
					
						
							|  |  |  |                 const u16 loader_id = static_cast<u16>(Core::System::ResultStatus::ErrorLoader); | 
					
						
							|  |  |  |                 const u16 error_id = static_cast<u16>(result) - loader_id; | 
					
						
							|  |  |  |                 QMessageBox::critical( | 
					
						
							|  |  |  |                     this, tr("Error while loading ROM!"), | 
					
						
							|  |  |  |                     QString::fromStdString(fmt::format( | 
					
						
							|  |  |  |                         "While attempting to load the ROM requested, an error occured. Please " | 
					
						
							|  |  |  |                         "refer to the yuzu wiki for more information or the yuzu discord for " | 
					
						
							|  |  |  |                         "additional help.\n\nError Code: {:04X}-{:04X}\nError Description: {}", | 
					
						
							| 
									
										
										
										
											2018-08-15 05:38:37 -04:00
										 |  |  |                         loader_id, error_id, static_cast<Loader::ResultStatus>(error_id)))); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:06:44 -04:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 QMessageBox::critical( | 
					
						
							|  |  |  |                     this, tr("Error while loading ROM!"), | 
					
						
							|  |  |  |                     tr("An unknown error occurred. 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
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-04-07 07:25:14 -06:00
										 |  |  |     Core::Telemetry().AddField(Telemetry::FieldType::App, "Frontend", "Qt"); | 
					
						
							| 
									
										
										
										
											2016-01-07 20:33:54 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-02-16 22:32:22 -08:00
										 |  |  | void GMainWindow::BootGame(const QString& filename) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |     LOG_INFO(Frontend, "yuzu 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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(render_window, &GRenderWindow::Closed, this, &GMainWindow::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
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(emu_thread.get(), &EmuThread::DebugModeEntered, waitTreeWidget, | 
					
						
							|  |  |  |             &WaitTreeWidget::OnDebugModeEntered, Qt::BlockingQueuedConnection); | 
					
						
							|  |  |  |     connect(emu_thread.get(), &EmuThread::DebugModeLeft, waitTreeWidget, | 
					
						
							|  |  |  |             &WaitTreeWidget::OnDebugModeLeft, Qt::BlockingQueuedConnection); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // Update the GUI
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2018-01-16 15:59:30 +01:00
										 |  |  |     if (ui.action_Fullscreen->isChecked()) { | 
					
						
							|  |  |  |         ShowFullscreen(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											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-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
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     disconnect(render_window, &GRenderWindow::Closed, this, &GMainWindow::OnStopGame); | 
					
						
							| 
									
										
										
										
											2015-09-05 12:29:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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() { | 
					
						
							| 
									
										
										
										
											2018-08-06 14:30:04 -04:00
										 |  |  |     const int num_recent_files = | 
					
						
							|  |  |  |         std::min(UISettings::values.recent_files.size(), max_recent_files_item); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 14:30:04 -04:00
										 |  |  |     for (int i = 0; i < num_recent_files; i++) { | 
					
						
							|  |  |  |         const QString text = QString("&%1. %2").arg(i + 1).arg( | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |             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); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 14:28:01 -04:00
										 |  |  |     // Enable the recent files menu if the list isn't empty
 | 
					
						
							|  |  |  |     ui.menu_recent_files->setEnabled(num_recent_files != 0); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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 + " "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 12:16:56 -04:00
										 |  |  |     extensions += "main "; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-11 21:56:15 -07:00
										 |  |  |     QString file_filter = tr("Switch Executable") + " (" + extensions + ")"; | 
					
						
							| 
									
										
										
										
											2017-02-16 22:38:05 -08:00
										 |  |  |     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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-14 12:27:29 -04:00
										 |  |  | void GMainWindow::OnMenuLoadFolder() { | 
					
						
							| 
									
										
										
										
											2018-08-06 14:02:31 -04:00
										 |  |  |     const QString dir_path = | 
					
						
							|  |  |  |         QFileDialog::getExistingDirectory(this, tr("Open Extracted ROM Directory")); | 
					
						
							| 
									
										
										
										
											2018-06-14 12:27:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 14:02:31 -04:00
										 |  |  |     if (dir_path.isNull()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const QDir dir{dir_path}; | 
					
						
							|  |  |  |     const QStringList matching_main = dir.entryList(QStringList("main"), QDir::Files); | 
					
						
							| 
									
										
										
										
											2018-06-14 17:25:40 -04:00
										 |  |  |     if (matching_main.size() == 1) { | 
					
						
							|  |  |  |         BootGame(dir.path() + DIR_SEP + matching_main[0]); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         QMessageBox::warning(this, tr("Invalid Directory Selected"), | 
					
						
							|  |  |  |                              tr("The directory you have selected does not contain a 'main' file.")); | 
					
						
							| 
									
										
										
										
											2018-06-14 12:27:29 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  | void GMainWindow::OnMenuInstallToNAND() { | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |     const QString file_filter = | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |         tr("Installable Switch File (*.nca *.xci);;Nintendo Content Archive (*.nca);;NX Cartridge " | 
					
						
							|  |  |  |            "Image (*.xci)"); | 
					
						
							|  |  |  |     QString filename = QFileDialog::getOpenFileName(this, tr("Install File"), | 
					
						
							|  |  |  |                                                     UISettings::values.roms_path, file_filter); | 
					
						
							| 
									
										
										
										
											2018-08-10 15:07:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const auto qt_raw_copy = [this](FileSys::VirtualFile src, FileSys::VirtualFile dest) { | 
					
						
							|  |  |  |         if (src == nullptr || dest == nullptr) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         if (!dest->Resize(src->GetSize())) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QProgressDialog progress(fmt::format("Installing file \"{}\"...", src->GetName()).c_str(), | 
					
						
							|  |  |  |                                  "Cancel", 0, src->GetSize() / 0x1000, this); | 
					
						
							|  |  |  |         progress.setWindowModality(Qt::WindowModal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         std::array<u8, 0x1000> buffer{}; | 
					
						
							|  |  |  |         for (size_t i = 0; i < src->GetSize(); i += 0x1000) { | 
					
						
							|  |  |  |             if (progress.wasCanceled()) { | 
					
						
							|  |  |  |                 dest->Resize(0); | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             progress.setValue(i / 0x1000); | 
					
						
							|  |  |  |             const auto read = src->Read(buffer.data(), buffer.size(), i); | 
					
						
							|  |  |  |             dest->Write(buffer.data(), read, i); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |     const auto success = [this]() { | 
					
						
							|  |  |  |         QMessageBox::information(this, tr("Successfully Installed"), | 
					
						
							|  |  |  |                                  tr("The file was successfully installed.")); | 
					
						
							|  |  |  |         game_list->PopulateAsync(UISettings::values.gamedir, UISettings::values.gamedir_deepscan); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto failed = [this]() { | 
					
						
							|  |  |  |         QMessageBox::warning( | 
					
						
							|  |  |  |             this, tr("Failed to Install"), | 
					
						
							|  |  |  |             tr("There was an error while attempting to install the provided file. It " | 
					
						
							|  |  |  |                "could have an incorrect format or be missing metadata. Please " | 
					
						
							|  |  |  |                "double-check your file and try again.")); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto overwrite = [this]() { | 
					
						
							|  |  |  |         return QMessageBox::question(this, "Failed to Install", | 
					
						
							|  |  |  |                                      "The file you are attempting to install already exists " | 
					
						
							|  |  |  |                                      "in the cache. Would you like to overwrite it?") == | 
					
						
							|  |  |  |                QMessageBox::Yes; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |     if (!filename.isEmpty()) { | 
					
						
							|  |  |  |         if (filename.endsWith("xci", Qt::CaseInsensitive)) { | 
					
						
							|  |  |  |             const auto xci = std::make_shared<FileSys::XCI>( | 
					
						
							|  |  |  |                 vfs->OpenFile(filename.toStdString(), FileSys::Mode::Read)); | 
					
						
							|  |  |  |             if (xci->GetStatus() != Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                 failed(); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |             const auto res = | 
					
						
							|  |  |  |                 Service::FileSystem::GetUserNANDContents()->InstallEntry(xci, false, qt_raw_copy); | 
					
						
							|  |  |  |             if (res == FileSys::InstallResult::Success) { | 
					
						
							|  |  |  |                 success(); | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                 if (res == FileSys::InstallResult::ErrorAlreadyExists) { | 
					
						
							|  |  |  |                     if (overwrite()) { | 
					
						
							|  |  |  |                         const auto res2 = Service::FileSystem::GetUserNANDContents()->InstallEntry( | 
					
						
							|  |  |  |                             xci, true, qt_raw_copy); | 
					
						
							|  |  |  |                         if (res2 == FileSys::InstallResult::Success) { | 
					
						
							|  |  |  |                             success(); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             failed(); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     failed(); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             const auto nca = std::make_shared<FileSys::NCA>( | 
					
						
							|  |  |  |                 vfs->OpenFile(filename.toStdString(), FileSys::Mode::Read)); | 
					
						
							|  |  |  |             if (nca->GetStatus() != Loader::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                 failed(); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |             static const QStringList tt_options{"System Application", | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |                                                 "System Archive", | 
					
						
							|  |  |  |                                                 "System Application Update", | 
					
						
							|  |  |  |                                                 "Firmware Package (Type A)", | 
					
						
							|  |  |  |                                                 "Firmware Package (Type B)", | 
					
						
							|  |  |  |                                                 "Game", | 
					
						
							|  |  |  |                                                 "Game Update", | 
					
						
							|  |  |  |                                                 "Game DLC", | 
					
						
							|  |  |  |                                                 "Delta Title"}; | 
					
						
							|  |  |  |             bool ok; | 
					
						
							|  |  |  |             const auto item = QInputDialog::getItem( | 
					
						
							|  |  |  |                 this, tr("Select NCA Install Type..."), | 
					
						
							|  |  |  |                 tr("Please select the type of title you would like to install this NCA as:\n(In " | 
					
						
							|  |  |  |                    "most instances, the default 'Game' is fine.)"), | 
					
						
							|  |  |  |                 tt_options, 5, false, &ok); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             auto index = tt_options.indexOf(item); | 
					
						
							|  |  |  |             if (!ok || index == -1) { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                 QMessageBox::warning(this, tr("Failed to Install"), | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |                                      tr("The title type you selected for the NCA is invalid.")); | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  |                 return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (index >= 5) | 
					
						
							| 
									
										
										
										
											2018-08-10 11:11:33 -04:00
										 |  |  |                 index += 0x7B; | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |             const auto res = Service::FileSystem::GetUserNANDContents()->InstallEntry( | 
					
						
							|  |  |  |                 nca, static_cast<FileSys::TitleType>(index), false, qt_raw_copy); | 
					
						
							|  |  |  |             if (res == FileSys::InstallResult::Success) { | 
					
						
							|  |  |  |                 success(); | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2018-08-11 23:01:38 -04:00
										 |  |  |                 if (res == FileSys::InstallResult::ErrorAlreadyExists) { | 
					
						
							|  |  |  |                     if (overwrite()) { | 
					
						
							|  |  |  |                         const auto res2 = Service::FileSystem::GetUserNANDContents()->InstallEntry( | 
					
						
							|  |  |  |                             nca, static_cast<FileSys::TitleType>(index), true, qt_raw_copy); | 
					
						
							|  |  |  |                         if (res2 == FileSys::InstallResult::Success) { | 
					
						
							|  |  |  |                             success(); | 
					
						
							|  |  |  |                         } else { | 
					
						
							|  |  |  |                             failed(); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } else { | 
					
						
							|  |  |  |                     failed(); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2018-08-09 21:33:13 -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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-06 14:12:57 -04:00
										 |  |  |     const QString filename = action->data().toString(); | 
					
						
							|  |  |  |     if (QFileInfo::exists(filename)) { | 
					
						
							| 
									
										
										
										
											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"); | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(emu_thread.get(), &EmuThread::ErrorThrown, this, &GMainWindow::OnCoreError); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  | void GMainWindow::ToggleFullscreen() { | 
					
						
							|  |  |  |     if (!emulation_running) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (ui.action_Fullscreen->isChecked()) { | 
					
						
							| 
									
										
										
										
											2018-01-16 15:59:30 +01:00
										 |  |  |         ShowFullscreen(); | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2018-01-16 15:59:30 +01:00
										 |  |  |         HideFullscreen(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::ShowFullscreen() { | 
					
						
							|  |  |  |     if (ui.action_Single_Window_Mode->isChecked()) { | 
					
						
							|  |  |  |         UISettings::values.geometry = saveGeometry(); | 
					
						
							|  |  |  |         ui.menubar->hide(); | 
					
						
							|  |  |  |         statusBar()->hide(); | 
					
						
							|  |  |  |         showFullScreen(); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         UISettings::values.renderwindow_geometry = render_window->saveGeometry(); | 
					
						
							|  |  |  |         render_window->showFullScreen(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GMainWindow::HideFullscreen() { | 
					
						
							|  |  |  |     if (ui.action_Single_Window_Mode->isChecked()) { | 
					
						
							|  |  |  |         statusBar()->setVisible(ui.action_Show_Status_Bar->isChecked()); | 
					
						
							|  |  |  |         ui.menubar->show(); | 
					
						
							|  |  |  |         showNormal(); | 
					
						
							|  |  |  |         restoreGeometry(UISettings::values.geometry); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         render_window->showNormal(); | 
					
						
							|  |  |  |         render_window->restoreGeometry(UISettings::values.renderwindow_geometry); | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01: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() { | 
					
						
							| 
									
										
										
										
											2018-08-07 00:43:07 -04:00
										 |  |  |     ConfigureDialog configureDialog(this, hotkey_registry); | 
					
						
							| 
									
										
										
										
											2018-07-29 14:37:18 +02:00
										 |  |  |     auto old_theme = UISettings::values.theme; | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2018-07-29 14:37:18 +02:00
										 |  |  |         if (UISettings::values.theme != old_theme) | 
					
						
							|  |  |  |             UpdateUITheme(); | 
					
						
							| 
									
										
										
										
											2018-07-28 12:32:16 -04:00
										 |  |  |         game_list->PopulateAsync(UISettings::values.gamedir, UISettings::values.gamedir_deepscan); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-14 19:15:45 +01:00
										 |  |  | void GMainWindow::OnAbout() { | 
					
						
							|  |  |  |     AboutDialog aboutDialog(this); | 
					
						
							|  |  |  |     aboutDialog.exec(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							| 
									
										
										
										
											2018-03-23 15:24:21 +01:00
										 |  |  |     const QString common_message = tr( | 
					
						
							|  |  |  |         "The game you are trying to load requires additional files from your Switch to be dumped " | 
					
						
							|  |  |  |         "before playing.<br/><br/>For more information on dumping these files, please see the " | 
					
						
							|  |  |  |         "following wiki page: <a " | 
					
						
							| 
									
										
										
										
											2018-03-25 11:28:04 +02:00
										 |  |  |         "href='https://yuzu-emu.org/wiki/" | 
					
						
							|  |  |  |         "dumping-system-archives-and-the-shared-fonts-from-a-switch-console/'>Dumping System " | 
					
						
							| 
									
										
										
										
											2018-03-23 15:24:21 +01:00
										 |  |  |         "Archives and the Shared Fonts from a Switch Console</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-03-08 16:28:30 -05:00
										 |  |  |     switch (result) { | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |     case Core::System::ResultStatus::ErrorSystemFiles: { | 
					
						
							| 
									
										
										
										
											2018-03-23 15:24:21 +01:00
										 |  |  |         QString message = "yuzu was unable to locate a Switch 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: { | 
					
						
							| 
									
										
										
										
											2018-03-23 15:24:21 +01:00
										 |  |  |         QString message = tr("yuzu was unable to locate the Switch shared fonts. "); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |         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"), | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |             tr("yuzu has encountered a fatal error, please see the log for more details. " | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |                "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) { | 
					
						
							| 
									
										
										
										
											2018-01-16 17:32:27 +01:00
										 |  |  |             emu_thread->SetRunning(true); | 
					
						
							| 
									
										
										
										
											2017-06-02 17:03:38 -04:00
										 |  |  |             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 = | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |         QMessageBox::question(this, tr("yuzu"), tr("Are you sure you want to close yuzu?"), | 
					
						
							| 
									
										
										
										
											2017-03-08 20:21:31 -05:00
										 |  |  |                               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; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-14 16:35:47 +05:30
										 |  |  |     if (ui.action_Fullscreen->isChecked()) { | 
					
						
							|  |  |  |         UISettings::values.geometry = saveGeometry(); | 
					
						
							|  |  |  |         UISettings::values.renderwindow_geometry = render_window->saveGeometry(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-01-24 21:23:55 +01:00
										 |  |  |     UISettings::values.state = saveState(); | 
					
						
							| 
									
										
										
										
											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(); | 
					
						
							| 
									
										
										
										
											2018-01-16 15:50:33 +01:00
										 |  |  |     UISettings::values.fullscreen = ui.action_Fullscreen->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(); | 
					
						
							| 
									
										
										
										
											2018-08-07 00:43:07 -04:00
										 |  |  |     hotkey_registry.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( | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |         this, tr("yuzu"), | 
					
						
							| 
									
										
										
										
											2017-02-15 21:23:30 -06:00
										 |  |  |         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()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-30 11:50:10 +02:00
										 |  |  | void GMainWindow::UpdateUITheme() { | 
					
						
							|  |  |  |     QStringList theme_paths(default_theme_paths); | 
					
						
							|  |  |  |     if (UISettings::values.theme != UISettings::themes[0].second && | 
					
						
							|  |  |  |         !UISettings::values.theme.isEmpty()) { | 
					
						
							| 
									
										
										
										
											2018-08-06 14:12:57 -04:00
										 |  |  |         const QString theme_uri(":" + UISettings::values.theme + "/style.qss"); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:50:10 +02:00
										 |  |  |         QFile f(theme_uri); | 
					
						
							| 
									
										
										
										
											2018-08-06 14:12:57 -04:00
										 |  |  |         if (f.open(QFile::ReadOnly | QFile::Text)) { | 
					
						
							| 
									
										
										
										
											2018-03-30 11:50:10 +02:00
										 |  |  |             QTextStream ts(&f); | 
					
						
							|  |  |  |             qApp->setStyleSheet(ts.readAll()); | 
					
						
							|  |  |  |             GMainWindow::setStyleSheet(ts.readAll()); | 
					
						
							| 
									
										
										
										
											2018-08-06 14:12:57 -04:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             LOG_ERROR(Frontend, "Unable to set style, stylesheet file not found"); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:50:10 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         theme_paths.append(QStringList{":/icons/default", ":/icons/" + UISettings::values.theme}); | 
					
						
							|  |  |  |         QIcon::setThemeName(":/icons/" + UISettings::values.theme); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         qApp->setStyleSheet(""); | 
					
						
							|  |  |  |         GMainWindow::setStyleSheet(""); | 
					
						
							|  |  |  |         theme_paths.append(QStringList{":/icons/default"}); | 
					
						
							|  |  |  |         QIcon::setThemeName(":/icons/default"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     QIcon::setThemeSearchPaths(theme_paths); | 
					
						
							| 
									
										
										
										
											2018-03-30 17:38:34 +02:00
										 |  |  |     emit UpdateThemedIcons(); | 
					
						
							| 
									
										
										
										
											2018-03-30 11:50:10 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | #ifdef main
 | 
					
						
							|  |  |  | #undef main
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-21 15:52:42 -04:00
										 |  |  | static void InitializeLogging() { | 
					
						
							|  |  |  |     Log::Filter log_filter; | 
					
						
							|  |  |  |     log_filter.ParseFilterString(Settings::values.log_filter); | 
					
						
							|  |  |  |     Log::SetGlobalFilter(log_filter); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const std::string& log_dir = FileUtil::GetUserPath(FileUtil::UserPath::LogDir); | 
					
						
							|  |  |  |     FileUtil::CreateFullPath(log_dir); | 
					
						
							|  |  |  |     Log::AddBackend(std::make_unique<Log::FileBackend>(log_dir + LOG_FILE)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:43:18 +02:00
										 |  |  | int main(int argc, char* argv[]) { | 
					
						
							| 
									
										
										
										
											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
 | 
					
						
							| 
									
										
										
										
											2018-01-13 23:49:16 +00:00
										 |  |  |     QCoreApplication::setOrganizationName("yuzu team"); | 
					
						
							|  |  |  |     QCoreApplication::setApplicationName("yuzu"); | 
					
						
							| 
									
										
										
										
											2015-07-26 17:13:02 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-29 14:10:16 -04:00
										 |  |  |     QApplication::setAttribute(Qt::AA_DontCheckOpenGLContextThreadAffinity); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     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
 | 
					
						
							| 
									
										
										
										
											2018-07-21 15:52:42 -04:00
										 |  |  |     InitializeLogging(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     main_window.show(); | 
					
						
							|  |  |  |     return app.exec(); | 
					
						
							|  |  |  | } |