From 9f43018df2a7d30194dd5e3d9e1c63a3d329a0ff Mon Sep 17 00:00:00 2001 From: grorp Date: Wed, 16 Oct 2024 21:37:00 +0200 Subject: [PATCH] Better UX when touch events aren't supported by Irrlicht device (#15288) --- builtin/mainmenu/settings/dlg_settings.lua | 10 ++++++---- builtin/settingtypes.txt | 2 ++ irr/include/IrrlichtDevice.h | 5 +++++ irr/src/CIrrDeviceLinux.cpp | 11 +++++++++++ irr/src/CIrrDeviceLinux.h | 3 +++ irr/src/CIrrDeviceSDL.cpp | 6 ++++++ irr/src/CIrrDeviceSDL.h | 3 +++ src/client/game.cpp | 3 +++ src/script/lua_api/l_mainmenu.cpp | 9 +++++++++ src/script/lua_api/l_mainmenu.h | 2 ++ 10 files changed, 50 insertions(+), 4 deletions(-) diff --git a/builtin/mainmenu/settings/dlg_settings.lua b/builtin/mainmenu/settings/dlg_settings.lua index 509a6a420..602d6894f 100644 --- a/builtin/mainmenu/settings/dlg_settings.lua +++ b/builtin/mainmenu/settings/dlg_settings.lua @@ -349,14 +349,16 @@ local function check_requirements(name, requires) local video_driver = core.get_active_driver() local shaders_support = video_driver == "opengl" or video_driver == "opengl3" or video_driver == "ogles2" + local touch_support = core.irrlicht_device_supports_touch() local touch_controls = core.settings:get("touch_controls") local special = { android = PLATFORM == "Android", desktop = PLATFORM ~= "Android", - -- When touch_controls is "auto", we don't which input method will be used, - -- so we show settings for both. - touchscreen = touch_controls == "auto" or core.is_yes(touch_controls), - keyboard_mouse = touch_controls == "auto" or not core.is_yes(touch_controls), + touch_support = touch_support, + -- When touch_controls is "auto", we don't know which input method will + -- be used, so we show settings for both. + touchscreen = touch_support and (touch_controls == "auto" or core.is_yes(touch_controls)), + keyboard_mouse = not touch_support or (touch_controls == "auto" or not core.is_yes(touch_controls)), shaders_support = shaders_support, shaders = core.settings:get_bool("enable_shaders") and shaders_support, opengl = video_driver == "opengl", diff --git a/builtin/settingtypes.txt b/builtin/settingtypes.txt index 15d0ea69a..99bfe9ca9 100644 --- a/builtin/settingtypes.txt +++ b/builtin/settingtypes.txt @@ -155,6 +155,8 @@ invert_hotbar_mouse_wheel (Hotbar: Invert mouse wheel direction) bool false # Enables the touchscreen controls, allowing you to play the game with a touchscreen. # "auto" means that the touchscreen controls will be enabled and disabled # automatically depending on the last used input method. +# +# Requires: touch_support touch_controls (Touchscreen controls) enum auto auto,true,false # Touchscreen sensitivity multiplier. diff --git a/irr/include/IrrlichtDevice.h b/irr/include/IrrlichtDevice.h index 777a23420..edc6ead61 100644 --- a/irr/include/IrrlichtDevice.h +++ b/irr/include/IrrlichtDevice.h @@ -193,6 +193,11 @@ public: /** If this returns false, you should not do any rendering. */ virtual bool isWindowVisible() const { return true; }; + //! Checks if the Irrlicht device supports touch events. + /** Intentionally doesn't check whether a touch input device is available + or similar. */ + virtual bool supportsTouchEvents() const { return false; } + //! Get the current color format of the window /** \return Color format of the window. */ virtual video::ECOLOR_FORMAT getColorFormat() const = 0; diff --git a/irr/src/CIrrDeviceLinux.cpp b/irr/src/CIrrDeviceLinux.cpp index 6ecb499b2..c00f52380 100644 --- a/irr/src/CIrrDeviceLinux.cpp +++ b/irr/src/CIrrDeviceLinux.cpp @@ -1205,6 +1205,17 @@ bool CIrrDeviceLinux::isWindowMaximized() const return WindowMaximized; } +//! Checks if the Irrlicht device supports touch events. +bool CIrrDeviceLinux::supportsTouchEvents() const +{ +#if defined(_IRR_LINUX_X11_XINPUT2_) + return true; +#else + return false; +#endif +} + + //! returns color format of the window. video::ECOLOR_FORMAT CIrrDeviceLinux::getColorFormat() const { diff --git a/irr/src/CIrrDeviceLinux.h b/irr/src/CIrrDeviceLinux.h index 4aac1cf6c..f5ee81c6b 100644 --- a/irr/src/CIrrDeviceLinux.h +++ b/irr/src/CIrrDeviceLinux.h @@ -66,6 +66,9 @@ public: //! returns last state from maximizeWindow() and restoreWindow() bool isWindowMaximized() const override; + //! Checks if the Irrlicht device supports touch events. + bool supportsTouchEvents() const override; + //! returns color format of the window. video::ECOLOR_FORMAT getColorFormat() const override; diff --git a/irr/src/CIrrDeviceSDL.cpp b/irr/src/CIrrDeviceSDL.cpp index 6d1b45886..0db598c74 100644 --- a/irr/src/CIrrDeviceSDL.cpp +++ b/irr/src/CIrrDeviceSDL.cpp @@ -1293,6 +1293,12 @@ bool CIrrDeviceSDL::isWindowVisible() const return !IsInBackground; } +//! Checks if the Irrlicht device supports touch events. +bool CIrrDeviceSDL::supportsTouchEvents() const +{ + return true; +} + //! returns if window is active. if not, nothing need to be drawn bool CIrrDeviceSDL::isWindowActive() const { diff --git a/irr/src/CIrrDeviceSDL.h b/irr/src/CIrrDeviceSDL.h index 7156c19b6..b52422423 100644 --- a/irr/src/CIrrDeviceSDL.h +++ b/irr/src/CIrrDeviceSDL.h @@ -93,6 +93,9 @@ public: //! Checks if the window could possibly be visible. bool isWindowVisible() const override; + //! Checks if the Irrlicht device supports touch events. + bool supportsTouchEvents() const override; + //! Get the position of this window on screen core::position2di getWindowPosition() override; diff --git a/src/client/game.cpp b/src/client/game.cpp index 1a125f492..9619a4a8a 100644 --- a/src/client/game.cpp +++ b/src/client/game.cpp @@ -1576,6 +1576,9 @@ bool Game::createClient(const GameStartData &start_data) bool Game::shouldShowTouchControls() { + if (!device->supportsTouchEvents()) + return false; + const std::string &touch_controls = g_settings->get("touch_controls"); if (touch_controls == "auto") return RenderingEngine::getLastPointerType() == PointerType::Touch; diff --git a/src/script/lua_api/l_mainmenu.cpp b/src/script/lua_api/l_mainmenu.cpp index a14944381..e4b9ddaf5 100644 --- a/src/script/lua_api/l_mainmenu.cpp +++ b/src/script/lua_api/l_mainmenu.cpp @@ -1028,6 +1028,14 @@ int ModApiMainMenu::l_get_active_irrlicht_device(lua_State *L) return 1; } +/******************************************************************************/ +int ModApiMainMenu::l_irrlicht_device_supports_touch(lua_State *L) +{ + lua_pushboolean(L, RenderingEngine::get_raw_device()->supportsTouchEvents()); + return 1; +} + + /******************************************************************************/ int ModApiMainMenu::l_get_min_supp_proto(lua_State *L) { @@ -1160,6 +1168,7 @@ void ModApiMainMenu::Initialize(lua_State *L, int top) API_FCT(get_active_driver); API_FCT(get_active_renderer); API_FCT(get_active_irrlicht_device); + API_FCT(irrlicht_device_supports_touch); API_FCT(get_min_supp_proto); API_FCT(get_max_supp_proto); API_FCT(get_formspec_version); diff --git a/src/script/lua_api/l_mainmenu.h b/src/script/lua_api/l_mainmenu.h index cb3e7f9ca..877aab2e8 100644 --- a/src/script/lua_api/l_mainmenu.h +++ b/src/script/lua_api/l_mainmenu.h @@ -116,6 +116,8 @@ private: static int l_get_active_irrlicht_device(lua_State *L); + static int l_irrlicht_device_supports_touch(lua_State *L); + //filesystem static int l_get_mainmenu_path(lua_State *L);