| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-09 02:57:42 +03:00
										 |  |  | #include <memory>
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:02:11 +01:00
										 |  |  | #include <tuple>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2015-05-06 04:06:12 -03:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2017-05-27 16:10:25 -07:00
										 |  |  | #include "core/frontend/framebuffer_layout.h"
 | 
					
						
							| 
									
										
										
										
											2015-06-21 14:02:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 20:20:19 -04:00
										 |  |  | namespace Core::Frontend { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Represents a graphics context that can be used for background computation or drawing. If the | 
					
						
							|  |  |  |  * graphics backend doesn't require the context, then the implementation of these methods can be | 
					
						
							|  |  |  |  * stubs | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class GraphicsContext { | 
					
						
							|  |  |  | public: | 
					
						
							| 
									
										
										
										
											2019-05-04 01:47:16 -04:00
										 |  |  |     virtual ~GraphicsContext(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     /// Makes the graphics context current for the caller thread
 | 
					
						
							|  |  |  |     virtual void MakeCurrent() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Releases (dunno if this is the "right" word) the context from the caller thread
 | 
					
						
							|  |  |  |     virtual void DoneCurrent() = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Abstraction class used to provide an interface between emulation code and the frontend | 
					
						
							|  |  |  |  * (e.g. SDL, QGLWidget, GLFW, etc...). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Design notes on the interaction between EmuWindow and the emulation core: | 
					
						
							|  |  |  |  * - Generally, decisions on anything visible to the user should be left up to the GUI. | 
					
						
							|  |  |  |  *   For example, the emulation core should not try to dictate some window title or size. | 
					
						
							|  |  |  |  *   This stuff is not the core's business and only causes problems with regards to thread-safety | 
					
						
							|  |  |  |  *   anyway. | 
					
						
							|  |  |  |  * - Under certain circumstances, it may be desirable for the core to politely request the GUI | 
					
						
							|  |  |  |  *   to set e.g. a minimum window size. However, the GUI should always be free to ignore any | 
					
						
							|  |  |  |  *   such hints. | 
					
						
							|  |  |  |  * - EmuWindow may expose some of its state as read-only to the emulation core, however care | 
					
						
							|  |  |  |  *   should be taken to make sure the provided information is self-consistent. This requires | 
					
						
							|  |  |  |  *   some sort of synchronization (most of this is still a TODO). | 
					
						
							| 
									
										
										
										
											2014-11-13 20:45:37 +01:00
										 |  |  |  * - DO NOT TREAT THIS CLASS AS A GUI TOOLKIT ABSTRACTION LAYER. That's not what it is. Please | 
					
						
							|  |  |  |  *   re-read the upper points again and think about it if you don't see this. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  | class EmuWindow : public GraphicsContext { | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |     /// Data structure to store emuwindow configuration
 | 
					
						
							| 
									
										
										
										
											2014-09-12 17:06:13 -07:00
										 |  |  |     struct WindowConfig { | 
					
						
							| 
									
										
										
										
											2018-08-14 19:36:13 -04:00
										 |  |  |         bool fullscreen = false; | 
					
						
							|  |  |  |         int res_width = 0; | 
					
						
							|  |  |  |         int res_height = 0; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         std::pair<unsigned, unsigned> min_client_area_size; | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |     /// Polls window events
 | 
					
						
							|  |  |  |     virtual void PollEvents() = 0; | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-01-11 21:06:34 -07:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Returns a GraphicsContext that the frontend provides that is shared with the emu window. This | 
					
						
							|  |  |  |      * context can be used from other threads for background graphics computation. If the frontend | 
					
						
							|  |  |  |      * is using a graphics backend that doesn't need anything specific to run on a different thread, | 
					
						
							|  |  |  |      * then it can use a stubbed implemenation for GraphicsContext. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * If the return value is null, then the core should assume that the frontend cannot provide a | 
					
						
							|  |  |  |      * Shared Context | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     virtual std::unique_ptr<GraphicsContext> CreateSharedContext() const { | 
					
						
							|  |  |  |         return nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-21 16:40:53 -03:00
										 |  |  |     /// Returns if window is shown (not minimized)
 | 
					
						
							|  |  |  |     virtual bool IsShown() const = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Retrieves Vulkan specific handlers from the window
 | 
					
						
							|  |  |  |     virtual void RetrieveVulkanHandlers(void* get_instance_proc_addr, void* instance, | 
					
						
							|  |  |  |                                         void* surface) const = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-08 03:13:26 -04:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Signal that a touch pressed event has occurred (e.g. mouse click pressed) | 
					
						
							|  |  |  |      * @param framebuffer_x Framebuffer x-coordinate that was pressed | 
					
						
							|  |  |  |      * @param framebuffer_y Framebuffer y-coordinate that was pressed | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     void TouchPressed(unsigned framebuffer_x, unsigned framebuffer_y); | 
					
						
							| 
									
										
										
										
											2015-03-08 03:13:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     /// Signal that a touch released event has occurred (e.g. mouse click released)
 | 
					
						
							|  |  |  |     void TouchReleased(); | 
					
						
							| 
									
										
										
										
											2015-03-08 03:13:26 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							|  |  |  |      * Signal that a touch movement event has occurred (e.g. mouse was moved over the emu window) | 
					
						
							|  |  |  |      * @param framebuffer_x Framebuffer x-coordinate | 
					
						
							|  |  |  |      * @param framebuffer_y Framebuffer y-coordinate | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  |     void TouchMoved(unsigned framebuffer_x, unsigned framebuffer_y); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Returns currently active configuration. | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |      * @note Accesses to the returned object need not be consistent because it may be modified in | 
					
						
							|  |  |  |      * another thread | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  |     const WindowConfig& GetActiveConfig() const { | 
					
						
							|  |  |  |         return active_config; | 
					
						
							| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |      * Requests the internal configuration to be replaced by the specified argument at some point in | 
					
						
							|  |  |  |      * the future. | 
					
						
							|  |  |  |      * @note This method is thread-safe, because it delays configuration changes to the GUI event | 
					
						
							|  |  |  |      * loop. Hence there is no guarantee on when the requested configuration will be active. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2014-09-12 17:06:13 -07:00
										 |  |  |     void SetConfig(const WindowConfig& val) { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |         config = val; | 
					
						
							| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-08-29 22:23:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2018-01-09 22:36:07 -05:00
										 |  |  |      * Gets the framebuffer layout (width, height, and screen regions) | 
					
						
							|  |  |  |      * @note This method is thread-safe | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |     const Layout::FramebufferLayout& GetFramebufferLayout() const { | 
					
						
							| 
									
										
										
										
											2015-03-07 17:21:19 -05:00
										 |  |  |         return framebuffer_layout; | 
					
						
							| 
									
										
										
										
											2014-04-08 23:18:23 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2016-11-05 02:58:11 -06:00
										 |  |  |      * Convenience method to update the current frame layout | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |      * Read from the current settings to determine which layout to use. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     void UpdateCurrentFramebufferLayout(unsigned width, unsigned height); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | protected: | 
					
						
							| 
									
										
										
										
											2017-08-09 02:57:42 +03:00
										 |  |  |     EmuWindow(); | 
					
						
							|  |  |  |     virtual ~EmuWindow(); | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Processes any pending configuration changes from the last SetConfig call. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:24:15 +01:00
										 |  |  |      * This method invokes OnMinimalClientAreaChangeRequest if the corresponding configuration | 
					
						
							|  |  |  |      * field changed. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |      * @note Implementations will usually want to call this from the GUI thread. | 
					
						
							| 
									
										
										
										
											2014-11-13 20:31:34 +01:00
										 |  |  |      * @todo Actually call this in existing implementations. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |      */ | 
					
						
							|  |  |  |     void ProcessConfigurationChanges() { | 
					
						
							|  |  |  |         // TODO: For proper thread safety, we should eventually implement a proper
 | 
					
						
							|  |  |  |         // multiple-writer/single-reader queue...
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (config.min_client_area_size != active_config.min_client_area_size) { | 
					
						
							|  |  |  |             OnMinimalClientAreaChangeRequest(config.min_client_area_size); | 
					
						
							|  |  |  |             config.min_client_area_size = active_config.min_client_area_size; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /**
 | 
					
						
							| 
									
										
										
										
											2015-03-07 17:21:19 -05:00
										 |  |  |      * Update framebuffer layout with the given parameter. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |      * @note EmuWindow implementations will usually use this in window resize event handlers. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |     void NotifyFramebufferLayoutChanged(const Layout::FramebufferLayout& layout) { | 
					
						
							| 
									
										
										
										
											2015-03-07 17:21:19 -05:00
										 |  |  |         framebuffer_layout = layout; | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-13 18:12:27 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Update internal client area size with the given parameter. | 
					
						
							|  |  |  |      * @note EmuWindow implementations will usually use this in window resize event handlers. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     void NotifyClientAreaSizeChanged(const std::pair<unsigned, unsigned>& size) { | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  |         client_area_width = size.first; | 
					
						
							|  |  |  |         client_area_height = size.second; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2014-11-13 18:24:15 +01:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Handler called when the minimal client area was requested to be changed via SetConfig. | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |      * For the request to be honored, EmuWindow implementations will usually reimplement this | 
					
						
							|  |  |  |      * function. | 
					
						
							| 
									
										
										
										
											2014-11-13 18:24:15 +01:00
										 |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-05-25 17:50:20 -03:00
										 |  |  |     virtual void OnMinimalClientAreaChangeRequest(std::pair<unsigned, unsigned>) { | 
					
						
							| 
									
										
										
										
											2014-11-13 18:24:15 +01:00
										 |  |  |         // By default, ignore this request and do nothing.
 | 
					
						
							| 
									
										
										
										
											2014-10-12 22:46:33 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-03 00:07:17 -06:00
										 |  |  |     Layout::FramebufferLayout framebuffer_layout; ///< Current framebuffer layout
 | 
					
						
							| 
									
										
										
										
											2014-10-12 18:14:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     unsigned client_area_width;  ///< Current client width, should be set by window impl.
 | 
					
						
							|  |  |  |     unsigned client_area_height; ///< Current client height, should be set by window impl.
 | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     WindowConfig config;        ///< Internal configuration (changes pending for being applied in
 | 
					
						
							|  |  |  |                                 /// ProcessConfigurationChanges)
 | 
					
						
							|  |  |  |     WindowConfig active_config; ///< Internal active configuration
 | 
					
						
							| 
									
										
										
										
											2015-03-08 03:13:26 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-09 02:57:42 +03:00
										 |  |  |     class TouchState; | 
					
						
							|  |  |  |     std::shared_ptr<TouchState> touch_state; | 
					
						
							| 
									
										
										
										
											2015-03-09 00:14:59 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |     /**
 | 
					
						
							|  |  |  |      * Clip the provided coordinates to be inside the touchscreen area. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2019-02-27 08:54:40 -05:00
										 |  |  |     std::tuple<unsigned, unsigned> ClipToTouchScreen(unsigned new_x, unsigned new_y) const; | 
					
						
							| 
									
										
										
										
											2013-09-04 21:00:29 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2018-08-11 20:20:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | } // namespace Core::Frontend
 |