| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include <QApplication>
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | #include <QHBoxLayout>
 | 
					
						
							|  |  |  | #include <QKeyEvent>
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | #include <QOffscreenSurface>
 | 
					
						
							|  |  |  | #include <QOpenGLWindow>
 | 
					
						
							|  |  |  | #include <QPainter>
 | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | #include <QScreen>
 | 
					
						
							|  |  |  | #include <QWindow>
 | 
					
						
							| 
									
										
										
										
											2018-04-29 18:37:15 -04:00
										 |  |  | #include <fmt/format.h>
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | #include "common/microprofile.h"
 | 
					
						
							| 
									
										
										
										
											2015-09-11 00:23:00 -04:00
										 |  |  | #include "common/scm_rev.h"
 | 
					
						
							| 
									
										
										
										
											2014-04-10 20:50:10 -04:00
										 |  |  | #include "core/core.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:36:07 -05:00
										 |  |  | #include "core/frontend/framebuffer_layout.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-28 12:33:35 +02:00
										 |  |  | #include "core/settings.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  | #include "input_common/keyboard.h"
 | 
					
						
							|  |  |  | #include "input_common/main.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-07 00:04:06 +03:00
										 |  |  | #include "input_common/motion_emu.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  | #include "video_core/renderer_base.h"
 | 
					
						
							|  |  |  | #include "video_core/video_core.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | #include "yuzu/bootmanager.h"
 | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  | #include "yuzu/main.h"
 | 
					
						
							| 
									
										
										
										
											2018-01-11 20:33:56 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-17 18:34:58 -05:00
										 |  |  | EmuThread::EmuThread(GRenderWindow* render_window) : render_window(render_window) {} | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  | void EmuThread::run() { | 
					
						
							| 
									
										
										
										
											2019-01-23 22:09:22 -05:00
										 |  |  |     render_window->MakeCurrent(); | 
					
						
							| 
									
										
										
										
											2015-05-18 21:24:43 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadCreate("EmuThread"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-21 16:38:23 -03:00
										 |  |  |     emit LoadProgress(VideoCore::LoadCallbackStage::Prepare, 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Core::System::GetInstance().Renderer().Rasterizer().LoadDiskResources( | 
					
						
							|  |  |  |         stop_run, [this](VideoCore::LoadCallbackStage stage, std::size_t value, std::size_t total) { | 
					
						
							|  |  |  |             emit LoadProgress(stage, value, total); | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     emit LoadProgress(VideoCore::LoadCallbackStage::Complete, 0, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-23 22:09:22 -05:00
										 |  |  |     if (Settings::values.use_asynchronous_gpu_emulation) { | 
					
						
							|  |  |  |         // Release OpenGL context for the GPU thread
 | 
					
						
							|  |  |  |         render_window->DoneCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-10 18:06:49 -06:00
										 |  |  |     // Holds whether the cpu was running during the last iteration,
 | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |     // so that the DebugModeLeft signal can be emitted before the
 | 
					
						
							|  |  |  |     // next execution step
 | 
					
						
							|  |  |  |     bool was_active = false; | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  |     while (!stop_run) { | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |         if (running) { | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             if (!was_active) | 
					
						
							|  |  |  |                 emit DebugModeLeft(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             Core::System::ResultStatus result = Core::System::GetInstance().RunLoop(); | 
					
						
							|  |  |  |             if (result != Core::System::ResultStatus::Success) { | 
					
						
							| 
									
										
										
										
											2018-01-16 17:32:27 +01:00
										 |  |  |                 this->SetRunning(false); | 
					
						
							| 
									
										
										
										
											2017-04-13 01:15:23 -04:00
										 |  |  |                 emit ErrorThrown(result, Core::System::GetInstance().GetStatusDetails()); | 
					
						
							| 
									
										
										
										
											2017-03-08 16:28:30 -05:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |             was_active = running || exec_step; | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  |             if (!was_active && !stop_run) | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |                 emit DebugModeEntered(); | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |         } else if (exec_step) { | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             if (!was_active) | 
					
						
							|  |  |  |                 emit DebugModeLeft(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-28 19:03:01 -04:00
										 |  |  |             exec_step = false; | 
					
						
							| 
									
										
										
										
											2016-12-15 19:01:48 -05:00
										 |  |  |             Core::System::GetInstance().SingleStep(); | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             emit DebugModeEntered(); | 
					
						
							| 
									
										
										
										
											2014-11-09 16:56:29 -05:00
										 |  |  |             yieldCurrentThread(); | 
					
						
							| 
									
										
										
										
											2015-05-25 20:34:09 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 12:14:23 +01:00
										 |  |  |             was_active = false; | 
					
						
							| 
									
										
										
										
											2015-05-16 11:56:00 -06:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |             std::unique_lock lock{running_mutex}; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |             running_cv.wait(lock, [this] { return IsRunning() || exec_step || stop_run; }); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-04-16 23:31:14 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-30 08:47:50 -03:00
										 |  |  |     // Shutdown the core emulation
 | 
					
						
							| 
									
										
										
										
											2016-11-04 23:14:38 -04:00
										 |  |  |     Core::System::GetInstance().Shutdown(); | 
					
						
							| 
									
										
										
										
											2015-08-30 08:47:50 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #if MICROPROFILE_ENABLED
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  |     MicroProfileOnThreadExit(); | 
					
						
							| 
									
										
										
										
											2016-04-29 02:17:31 +02:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-08-17 18:25:21 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-25 01:49:34 +10:00
										 |  |  |     render_window->moveContext(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | class GGLContext : public Core::Frontend::GraphicsContext { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2019-04-17 00:02:28 -04:00
										 |  |  |     explicit GGLContext(QOpenGLContext* shared_context) | 
					
						
							|  |  |  |         : context{std::make_unique<QOpenGLContext>(shared_context)} { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |         surface.setFormat(shared_context->format()); | 
					
						
							|  |  |  |         surface.create(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void MakeCurrent() override { | 
					
						
							|  |  |  |         context->makeCurrent(&surface); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void DoneCurrent() override { | 
					
						
							|  |  |  |         context->doneCurrent(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void SwapBuffers() override {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     std::unique_ptr<QOpenGLContext> context; | 
					
						
							|  |  |  |     QOffscreenSurface surface; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | // This class overrides paintEvent and resizeEvent to prevent the GUI thread from stealing GL
 | 
					
						
							|  |  |  | // context.
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | // The corresponding functionality is handled in EmuThread instead
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | class GGLWidgetInternal : public QOpenGLWindow { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     GGLWidgetInternal(GRenderWindow* parent, QOpenGLContext* shared_context) | 
					
						
							|  |  |  |         : QOpenGLWindow(shared_context), parent(parent) {} | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     void paintEvent(QPaintEvent* ev) override { | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  |         if (do_painting) { | 
					
						
							|  |  |  |             QPainter painter(this); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-26 02:56:13 -02:00
										 |  |  |     void resizeEvent(QResizeEvent* ev) override { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |         parent->OnClientAreaResized(ev->size().width(), ev->size().height()); | 
					
						
							|  |  |  |         parent->OnFramebufferSizeChanged(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  |     void keyPressEvent(QKeyEvent* event) override { | 
					
						
							|  |  |  |         InputCommon::GetKeyboard()->PressKey(event->key()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void keyReleaseEvent(QKeyEvent* event) override { | 
					
						
							|  |  |  |         InputCommon::GetKeyboard()->ReleaseKey(event->key()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mousePressEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchBeginEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto pos{event->pos()}; | 
					
						
							|  |  |  |         if (event->button() == Qt::LeftButton) { | 
					
						
							|  |  |  |             const auto [x, y] = parent->ScaleTouch(pos); | 
					
						
							|  |  |  |             parent->TouchPressed(x, y); | 
					
						
							|  |  |  |         } else if (event->button() == Qt::RightButton) { | 
					
						
							|  |  |  |             InputCommon::GetMotionEmu()->BeginTilt(pos.x(), pos.y()); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mouseMoveEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchUpdateEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const auto pos{event->pos()}; | 
					
						
							|  |  |  |         const auto [x, y] = parent->ScaleTouch(pos); | 
					
						
							|  |  |  |         parent->TouchMoved(x, y); | 
					
						
							|  |  |  |         InputCommon::GetMotionEmu()->Tilt(pos.x(), pos.y()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void mouseReleaseEvent(QMouseEvent* event) override { | 
					
						
							|  |  |  |         if (event->source() == Qt::MouseEventSynthesizedBySystem) | 
					
						
							|  |  |  |             return; // touch input is handled in TouchEndEvent
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (event->button() == Qt::LeftButton) | 
					
						
							|  |  |  |             parent->TouchReleased(); | 
					
						
							|  |  |  |         else if (event->button() == Qt::RightButton) | 
					
						
							|  |  |  |             InputCommon::GetMotionEmu()->EndTilt(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     void DisablePainting() { | 
					
						
							|  |  |  |         do_painting = false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     void EnablePainting() { | 
					
						
							|  |  |  |         do_painting = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     GRenderWindow* parent; | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  |     bool do_painting; | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | GRenderWindow::GRenderWindow(QWidget* parent, EmuThread* emu_thread) | 
					
						
							| 
									
										
										
										
											2019-04-16 23:49:24 -04:00
										 |  |  |     : QWidget(parent), emu_thread(emu_thread) { | 
					
						
							| 
									
										
										
										
											2018-10-24 08:10:56 -04:00
										 |  |  |     setWindowTitle(QStringLiteral("yuzu %1 | %2-%3") | 
					
						
							|  |  |  |                        .arg(Common::g_build_name, Common::g_scm_branch, Common::g_scm_desc)); | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  |     setAttribute(Qt::WA_AcceptTouchEvents); | 
					
						
							| 
									
										
										
										
											2014-11-13 18:17:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  |     InputCommon::Init(); | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     connect(this, &GRenderWindow::FirstFrameDisplayed, static_cast<GMainWindow*>(parent), | 
					
						
							|  |  |  |             &GMainWindow::OnLoadComplete); | 
					
						
							| 
									
										
										
										
											2017-01-21 11:53:03 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GRenderWindow::~GRenderWindow() { | 
					
						
							|  |  |  |     InputCommon::Shutdown(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::moveContext() { | 
					
						
							| 
									
										
										
										
											2014-08-25 00:47:00 +10:00
										 |  |  |     DoneCurrent(); | 
					
						
							| 
									
										
										
										
											2018-01-17 23:19:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // If the thread started running, move the GL Context to the new thread. Otherwise, move it
 | 
					
						
							|  |  |  |     // back.
 | 
					
						
							|  |  |  |     auto thread = (QThread::currentThread() == qApp->thread() && emu_thread != nullptr) | 
					
						
							|  |  |  |                       ? emu_thread | 
					
						
							|  |  |  |                       : qApp->thread(); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->moveToThread(thread); | 
					
						
							| 
									
										
										
										
											2014-08-25 00:47:00 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::SwapBuffers() { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     // In our multi-threaded QWidget use case we shouldn't need to call `makeCurrent`,
 | 
					
						
							| 
									
										
										
										
											2018-07-07 14:11:49 +02:00
										 |  |  |     // since we never call `doneCurrent` in this thread.
 | 
					
						
							|  |  |  |     // However:
 | 
					
						
							|  |  |  |     // - The Qt debug runtime prints a bogus warning on the console if `makeCurrent` wasn't called
 | 
					
						
							|  |  |  |     // since the last time `swapBuffers` was executed;
 | 
					
						
							|  |  |  |     // - On macOS, if `makeCurrent` isn't called explicitely, resizing the buffer breaks.
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->makeCurrent(child); | 
					
						
							| 
									
										
										
										
											2018-07-07 14:11:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->swapBuffers(child); | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     if (!first_frame) { | 
					
						
							|  |  |  |         emit FirstFrameDisplayed(); | 
					
						
							|  |  |  |         first_frame = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::MakeCurrent() { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->makeCurrent(child); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::DoneCurrent() { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     context->doneCurrent(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  | void GRenderWindow::PollEvents() {} | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | // On Qt 5.0+, this correctly gets the size of the framebuffer (pixels).
 | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | //
 | 
					
						
							|  |  |  | // Older versions get the window size (density independent pixels),
 | 
					
						
							|  |  |  | // and hence, do not support DPI scaling ("retina" displays).
 | 
					
						
							|  |  |  | // The result will be a viewport that is smaller than the extent of the window.
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::OnFramebufferSizeChanged() { | 
					
						
							|  |  |  |     // Screen changes potentially incur a change in screen DPI, hence we should update the
 | 
					
						
							|  |  |  |     // framebuffer size
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  |     qreal pixelRatio = GetWindowPixelRatio(); | 
					
						
							| 
									
										
										
										
											2015-09-10 23:42:45 +02:00
										 |  |  |     unsigned width = child->QPaintDevice::width() * pixelRatio; | 
					
						
							|  |  |  |     unsigned height = child->QPaintDevice::height() * pixelRatio; | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |     UpdateCurrentFramebufferLayout(width, height); | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | void GRenderWindow::ForwardKeyPressEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     if (child) { | 
					
						
							|  |  |  |         child->keyPressEvent(event); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::ForwardKeyReleaseEvent(QKeyEvent* event) { | 
					
						
							|  |  |  |     if (child) { | 
					
						
							|  |  |  |         child->keyReleaseEvent(event); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::BackupGeometry() { | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     geometry = ((QWidget*)this)->saveGeometry(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::RestoreGeometry() { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // We don't want to back up the geometry here (obviously)
 | 
					
						
							|  |  |  |     QWidget::restoreGeometry(geometry); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::restoreGeometry(const QByteArray& geometry) { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // Make sure users of this class don't need to deal with backing up the geometry themselves
 | 
					
						
							|  |  |  |     QWidget::restoreGeometry(geometry); | 
					
						
							|  |  |  |     BackupGeometry(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | QByteArray GRenderWindow::saveGeometry() { | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     // If we are a top-level widget, store the current geometry
 | 
					
						
							|  |  |  |     // otherwise, store the last backup
 | 
					
						
							| 
									
										
										
										
											2014-12-03 12:57:57 -06:00
										 |  |  |     if (parent() == nullptr) | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |         return ((QWidget*)this)->saveGeometry(); | 
					
						
							| 
									
										
										
										
											2014-03-31 22:26:50 -04:00
										 |  |  |     else | 
					
						
							|  |  |  |         return geometry; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  | qreal GRenderWindow::GetWindowPixelRatio() const { | 
					
						
							| 
									
										
										
										
											2015-09-10 23:42:45 +02:00
										 |  |  |     // windowHandle() might not be accessible until the window is displayed to screen.
 | 
					
						
							|  |  |  |     return windowHandle() ? windowHandle()->screen()->devicePixelRatio() : 1.0f; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  | std::pair<unsigned, unsigned> GRenderWindow::ScaleTouch(const QPointF pos) const { | 
					
						
							| 
									
										
										
										
											2019-03-25 17:01:11 -03:00
										 |  |  |     const qreal pixel_ratio = GetWindowPixelRatio(); | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  |     return {static_cast<unsigned>(std::max(std::round(pos.x() * pixel_ratio), qreal{0.0})), | 
					
						
							|  |  |  |             static_cast<unsigned>(std::max(std::round(pos.y() * pixel_ratio), qreal{0.0}))}; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-09-05 12:29:44 +02:00
										 |  |  | void GRenderWindow::closeEvent(QCloseEvent* event) { | 
					
						
							|  |  |  |     emit Closed(); | 
					
						
							|  |  |  |     QWidget::closeEvent(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-01 22:42:49 +03:00
										 |  |  | void GRenderWindow::TouchBeginEvent(const QTouchEvent* event) { | 
					
						
							|  |  |  |     // TouchBegin always has exactly one touch point, so take the .first()
 | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(event->touchPoints().first().pos()); | 
					
						
							|  |  |  |     this->TouchPressed(x, y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::TouchUpdateEvent(const QTouchEvent* event) { | 
					
						
							|  |  |  |     QPointF pos; | 
					
						
							|  |  |  |     int active_points = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // average all active touch points
 | 
					
						
							|  |  |  |     for (const auto tp : event->touchPoints()) { | 
					
						
							|  |  |  |         if (tp.state() & (Qt::TouchPointPressed | Qt::TouchPointMoved | Qt::TouchPointStationary)) { | 
					
						
							|  |  |  |             active_points++; | 
					
						
							|  |  |  |             pos += tp.pos(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     pos /= active_points; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto [x, y] = ScaleTouch(pos); | 
					
						
							|  |  |  |     this->TouchMoved(x, y); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void GRenderWindow::TouchEndEvent() { | 
					
						
							|  |  |  |     this->TouchReleased(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool GRenderWindow::event(QEvent* event) { | 
					
						
							|  |  |  |     if (event->type() == QEvent::TouchBegin) { | 
					
						
							|  |  |  |         TouchBeginEvent(static_cast<QTouchEvent*>(event)); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (event->type() == QEvent::TouchUpdate) { | 
					
						
							|  |  |  |         TouchUpdateEvent(static_cast<QTouchEvent*>(event)); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (event->type() == QEvent::TouchEnd || event->type() == QEvent::TouchCancel) { | 
					
						
							|  |  |  |         TouchEndEvent(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return QWidget::event(event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-17 21:41:25 +02:00
										 |  |  | void GRenderWindow::focusOutEvent(QFocusEvent* event) { | 
					
						
							|  |  |  |     QWidget::focusOutEvent(event); | 
					
						
							|  |  |  |     InputCommon::GetKeyboard()->ReleaseAllKeys(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::OnClientAreaResized(unsigned width, unsigned height) { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     NotifyClientAreaSizeChanged(std::make_pair(width, height)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | std::unique_ptr<Core::Frontend::GraphicsContext> GRenderWindow::CreateSharedContext() const { | 
					
						
							|  |  |  |     return std::make_unique<GGLContext>(shared_context.get()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | void GRenderWindow::InitRenderTarget() { | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     shared_context.reset(); | 
					
						
							|  |  |  |     context.reset(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     delete child; | 
					
						
							|  |  |  |     child = nullptr; | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     delete container; | 
					
						
							|  |  |  |     container = nullptr; | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     delete layout(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-17 00:01:00 -07:00
										 |  |  |     first_frame = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     // TODO: One of these flags might be interesting: WA_OpaquePaintEvent, WA_NoBackground,
 | 
					
						
							|  |  |  |     // WA_DontShowOnScreen, WA_DeleteOnClose
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     QSurfaceFormat fmt; | 
					
						
							| 
									
										
										
										
											2018-11-17 18:04:50 -04:00
										 |  |  |     fmt.setVersion(4, 3); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     fmt.setProfile(QSurfaceFormat::CoreProfile); | 
					
						
							|  |  |  |     // TODO: expose a setting for buffer value (ie default/single/double/triple)
 | 
					
						
							|  |  |  |     fmt.setSwapBehavior(QSurfaceFormat::DefaultSwapBehavior); | 
					
						
							|  |  |  |     shared_context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |     shared_context->setFormat(fmt); | 
					
						
							|  |  |  |     shared_context->create(); | 
					
						
							|  |  |  |     context = std::make_unique<QOpenGLContext>(); | 
					
						
							|  |  |  |     context->setShareContext(shared_context.get()); | 
					
						
							|  |  |  |     context->setFormat(fmt); | 
					
						
							|  |  |  |     context->create(); | 
					
						
							| 
									
										
										
										
											2018-09-06 18:57:51 +02:00
										 |  |  |     fmt.setSwapInterval(false); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     child = new GGLWidgetInternal(this, shared_context.get()); | 
					
						
							| 
									
										
										
										
											2019-03-25 16:41:48 -03:00
										 |  |  |     container = QWidget::createWindowContainer(child, this); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     QBoxLayout* layout = new QHBoxLayout(this); | 
					
						
							| 
									
										
										
										
											2019-03-25 16:46:30 -03:00
										 |  |  |     layout->addWidget(container); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  |     layout->setMargin(0); | 
					
						
							|  |  |  |     setLayout(layout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-25 16:46:30 -03:00
										 |  |  |     // Reset minimum size to avoid unwanted resizes when this function is called for a second time.
 | 
					
						
							|  |  |  |     setMinimumSize(1, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Show causes the window to actually be created and the OpenGL context as well, but we don't
 | 
					
						
							|  |  |  |     // want the widget to be shown yet, so immediately hide it.
 | 
					
						
							|  |  |  |     show(); | 
					
						
							|  |  |  |     hide(); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:36:07 -05:00
										 |  |  |     resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     child->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							|  |  |  |     container->resize(Layout::ScreenUndocked::Width, Layout::ScreenUndocked::Height); | 
					
						
							| 
									
										
										
										
											2016-08-29 21:28:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     OnMinimalClientAreaChangeRequest(GetActiveConfig().min_client_area_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     OnFramebufferSizeChanged(); | 
					
						
							|  |  |  |     NotifyClientAreaSizeChanged(std::pair<unsigned, unsigned>(child->width(), child->height())); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     BackupGeometry(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-31 14:16:16 +08:00
										 |  |  | void GRenderWindow::CaptureScreenshot(u16 res_scale, const QString& screenshot_path) { | 
					
						
							|  |  |  |     auto& renderer = Core::System::GetInstance().Renderer(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!res_scale) | 
					
						
							|  |  |  |         res_scale = VideoCore::GetResolutionScaleFactor(renderer); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const Layout::FramebufferLayout layout{Layout::FrameLayoutFromResolutionScale(res_scale)}; | 
					
						
							|  |  |  |     screenshot_image = QImage(QSize(layout.width, layout.height), QImage::Format_RGB32); | 
					
						
							|  |  |  |     renderer.RequestScreenshot(screenshot_image.bits(), | 
					
						
							|  |  |  |                                [=] { | 
					
						
							|  |  |  |                                    screenshot_image.mirrored(false, true).save(screenshot_path); | 
					
						
							|  |  |  |                                    LOG_INFO(Frontend, "The screenshot is saved."); | 
					
						
							|  |  |  |                                }, | 
					
						
							|  |  |  |                                layout); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::OnMinimalClientAreaChangeRequest( | 
					
						
							|  |  |  |     const std::pair<unsigned, unsigned>& minimal_size) { | 
					
						
							| 
									
										
										
										
											2014-11-19 09:02:05 +00:00
										 |  |  |     setMinimumSize(minimal_size.first, minimal_size.second); | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  | void GRenderWindow::OnEmulationStarting(EmuThread* emu_thread) { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     this->emu_thread = emu_thread; | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  |     child->DisablePainting(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-30 19:46:50 -04:00
										 |  |  | void GRenderWindow::OnEmulationStopping() { | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  |     emu_thread = nullptr; | 
					
						
							| 
									
										
										
										
											2015-08-26 22:04:12 +02:00
										 |  |  |     child->EnablePainting(); | 
					
						
							| 
									
										
										
										
											2015-04-29 00:01:41 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void GRenderWindow::showEvent(QShowEvent* event) { | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  |     QWidget::showEvent(event); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 18:01:46 -07:00
										 |  |  |     // windowHandle() is not initialized until the Window is shown, so we connect it here.
 | 
					
						
							| 
									
										
										
										
											2018-01-18 20:03:13 -05:00
										 |  |  |     connect(windowHandle(), &QWindow::screenChanged, this, &GRenderWindow::OnFramebufferSizeChanged, | 
					
						
							|  |  |  |             Qt::UniqueConnection); | 
					
						
							| 
									
										
										
										
											2015-09-04 15:55:48 +02:00
										 |  |  | } |