From 7dad916b363c77a763a3dcb9b231f0d6b0754a83 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Mon, 23 Sep 2024 11:12:55 +0200 Subject: [PATCH 01/16] WIP: input capture --- .gitmodules | 3 ++- CMakeLists.txt | 2 ++ src/managers/PointerManager.cpp | 4 ++++ src/managers/ProtocolManager.cpp | 3 +++ src/protocols/InputCapture.cpp | 33 ++++++++++++++++++++++++++++++++ src/protocols/InputCapture.hpp | 23 ++++++++++++++++++++++ src/protocols/core/Output.cpp | 2 ++ subprojects/hyprland-protocols | 2 +- 8 files changed, 70 insertions(+), 2 deletions(-) create mode 100644 src/protocols/InputCapture.cpp create mode 100644 src/protocols/InputCapture.hpp diff --git a/.gitmodules b/.gitmodules index 638f8ba9776..8c9b93366c8 100644 --- a/.gitmodules +++ b/.gitmodules @@ -1,6 +1,7 @@ [submodule "subprojects/hyprland-protocols"] path = subprojects/hyprland-protocols - url = https://github.com/hyprwm/hyprland-protocols + url = https://github.com/3l0w/hyprland-protocols + branch = feat/input-capture-impl [submodule "subprojects/udis86"] path = subprojects/udis86 url = https://github.com/canihavesomecoffee/udis86 diff --git a/CMakeLists.txt b/CMakeLists.txt index 3441039e10f..f48727c552c 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -351,6 +351,8 @@ protocolnew("staging/xdg-dialog" "xdg-dialog-v1" false) protocolnew("staging/single-pixel-buffer" "single-pixel-buffer-v1" false) protocolnew("staging/security-context" "security-context-v1" false) +protocolnew("subprojects/hyprland-protocols/protocols" "hyprland-input-capture-v1" + true) protocolwayland() # tools diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index bc166825a0f..33a54ef9a1b 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -10,6 +10,7 @@ #include "eventLoop/EventLoopManager.hpp" #include "../render/pass/TexPassElement.hpp" #include "SeatManager.hpp" +#include "protocols/InputCapture.hpp" #include #include @@ -699,6 +700,9 @@ void CPointerManager::move(const Vector2D& deltaLogical) { const auto oldPos = pointerPos; auto newPos = oldPos + Vector2D{std::isnan(deltaLogical.x) ? 0.0 : deltaLogical.x, std::isnan(deltaLogical.y) ? 0.0 : deltaLogical.y}; + + PROTO::inputCapture->sendAbsoluteMotion(newPos, deltaLogical); + warpTo(newPos); } diff --git a/src/managers/ProtocolManager.cpp b/src/managers/ProtocolManager.cpp index cf517b22a44..887dc95bec9 100644 --- a/src/managers/ProtocolManager.cpp +++ b/src/managers/ProtocolManager.cpp @@ -47,6 +47,7 @@ #include "../protocols/SinglePixel.hpp" #include "../protocols/SecurityContext.hpp" #include "../protocols/CTMControl.hpp" +#include "../protocols/InputCapture.hpp" #include "../protocols/core/Seat.hpp" #include "../protocols/core/DataDevice.hpp" @@ -159,6 +160,7 @@ CProtocolManager::CProtocolManager() { PROTO::singlePixel = std::make_unique(&wp_single_pixel_buffer_manager_v1_interface, 1, "SinglePixel"); PROTO::securityContext = std::make_unique(&wp_security_context_manager_v1_interface, 1, "SecurityContext"); PROTO::ctm = std::make_unique(&hyprland_ctm_control_manager_v1_interface, 1, "CTMControl"); + PROTO::inputCapture = std::make_unique(&hyprland_input_capture_manager_v1_interface, 1, "InputCapture"); for (auto const& b : g_pCompositor->m_pAqBackend->getImplementations()) { if (b->type() != Aquamarine::AQ_BACKEND_DRM) @@ -232,6 +234,7 @@ CProtocolManager::~CProtocolManager() { PROTO::singlePixel.reset(); PROTO::securityContext.reset(); PROTO::ctm.reset(); + PROTO::inputCapture.reset(); PROTO::lease.reset(); PROTO::sync.reset(); diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp new file mode 100644 index 00000000000..889f2599cd0 --- /dev/null +++ b/src/protocols/InputCapture.cpp @@ -0,0 +1,33 @@ +#include "InputCapture.hpp" +#include "hyprland-input-capture-v1.hpp" +#include +#include +#include + +CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name) : IWaylandProtocol(iface, ver, name) { + ; +} + +void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id) { + const auto RESOURCE = m_vManagers.emplace_back(std::make_unique(client, ver, id)).get(); + + RESOURCE->setOnDestroy([this](CHyprlandInputCaptureManagerV1* p) { this->onManagerResourceDestroy(p->resource()); }); + + RESOURCE->setCapture([this](CHyprlandInputCaptureManagerV1* p) { this->onCapture(p); }); + RESOURCE->setRelease([this](CHyprlandInputCaptureManagerV1* p) { this->onRelease(p); }); +} + +void CInputCaptureProtocol::onManagerResourceDestroy(wl_resource* res) { + std::erase_if(m_vManagers, [&](const auto& other) { return other->resource() == res; }); +} + +void CInputCaptureProtocol::onCapture(CHyprlandInputCaptureManagerV1* pMgr) {} + +void CInputCaptureProtocol::onRelease(CHyprlandInputCaptureManagerV1* pMgr) {} + +void CInputCaptureProtocol::sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta) { + for (const UP& manager : m_vManagers) { + manager->sendAbsoluteMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), + wl_fixed_from_double(delta.y)); + } +} diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp new file mode 100644 index 00000000000..172c17dcb5f --- /dev/null +++ b/src/protocols/InputCapture.hpp @@ -0,0 +1,23 @@ +#pragma once + +#include "WaylandProtocol.hpp" +#include "hyprland-input-capture-v1.hpp" +#include + +class CInputCaptureProtocol : public IWaylandProtocol { + public: + CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name); + virtual void bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id); + void sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta); + + private: + void onManagerResourceDestroy(wl_resource* res); + void onCapture(CHyprlandInputCaptureManagerV1* pMgr); + void onRelease(CHyprlandInputCaptureManagerV1* pMgr); + + std::vector> m_vManagers; +}; + +namespace PROTO { + inline UP inputCapture; +} diff --git a/src/protocols/core/Output.cpp b/src/protocols/core/Output.cpp index db580457a73..ea4183d7171 100644 --- a/src/protocols/core/Output.cpp +++ b/src/protocols/core/Output.cpp @@ -74,6 +74,8 @@ void CWLOutputResource::updateState() { if (resource->version() >= 2) resource->sendScale(std::ceil(monitor->scale)); + resource->sendGeometry(monitor->vecPosition.x, monitor->vecPosition.y, monitor->output->physicalSize.x, monitor->output->physicalSize.y, (wl_output_subpixel)monitor->output->subpixel, monitor->output->make.c_str(), + monitor->output->model.c_str(), monitor->transform); resource->sendMode((wl_output_mode)(WL_OUTPUT_MODE_CURRENT), monitor->vecPixelSize.x, monitor->vecPixelSize.y, monitor->refreshRate * 1000.0); resource->sendGeometry(0, 0, monitor->output->physicalSize.x, monitor->output->physicalSize.y, (wl_output_subpixel)monitor->output->subpixel, monitor->output->make.c_str(), diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index c7c3f4cd0fa..1c4de2b22b9 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit c7c3f4cd0faed21fc90ba6bd06fe4f3e0e057ea8 +Subproject commit 1c4de2b22b9a7100ca664e46e37e29b0b724ce97 From 5fb45626e5014cc9e3e99b0aa8fceec9e4cf9e2c Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Wed, 25 Sep 2024 18:03:12 +0200 Subject: [PATCH 02/16] input-capture: impl keyboard, mouse button & mouse wheel --- src/managers/PointerManager.cpp | 1 + src/managers/input/InputManager.cpp | 22 ++++++++++++++++++ src/managers/input/InputManager.hpp | 1 + src/protocols/InputCapture.cpp | 36 +++++++++++++++++++++++++++++ src/protocols/InputCapture.hpp | 8 +++++++ subprojects/hyprland-protocols | 2 +- 6 files changed, 69 insertions(+), 1 deletion(-) diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index 33a54ef9a1b..63bebc22256 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -703,6 +703,7 @@ void CPointerManager::move(const Vector2D& deltaLogical) { PROTO::inputCapture->sendAbsoluteMotion(newPos, deltaLogical); + //TODO: Inhibit inputs warpTo(newPos); } diff --git a/src/managers/input/InputManager.cpp b/src/managers/input/InputManager.cpp index cccaa43e040..bad91260c3c 100644 --- a/src/managers/input/InputManager.cpp +++ b/src/managers/input/InputManager.cpp @@ -19,6 +19,7 @@ #include "../../protocols/core/DataDevice.hpp" #include "../../protocols/core/Compositor.hpp" #include "../../protocols/XDGShell.hpp" +#include "../../protocols/InputCapture.hpp" #include "../../devices/Mouse.hpp" #include "../../devices/VirtualPointer.hpp" @@ -99,6 +100,7 @@ void CInputManager::onMouseMoved(IPointer::SMotionEvent e) { g_pPointerManager->move(DELTA); + //TODO: Inhibit inputs mouseMoveUnified(e.timeMs, false, e.mouse); m_tmrLastCursorMovement.reset(); @@ -545,6 +547,9 @@ void CInputManager::onMouseButton(IPointer::SButtonEvent e) { if (e.mouse) recheckMouseWarpOnMouseInput(); + PROTO::inputCapture->sendButton(e.button, (hyprlandInputCaptureManagerV1ButtonState)e.state); + + //TODO: Inhibit inputs m_tmrLastCursorMovement.reset(); if (e.state == WL_POINTER_BUTTON_STATE_PRESSED) { @@ -782,6 +787,13 @@ void CInputManager::onMouseWheel(IPointer::SAxisEvent e) { if (e.mouse) recheckMouseWarpOnMouseInput(); + PROTO::inputCapture->sendAxis((hyprlandInputCaptureManagerV1Axis)e.axis, e.delta); + if (e.source == 0) + PROTO::inputCapture->sendAxisValue120((hyprlandInputCaptureManagerV1Axis)e.axis, e.delta); + else if (e.delta == 0) + PROTO::inputCapture->sendAxisStop((hyprlandInputCaptureManagerV1Axis)e.axis); + + //TODO: Inhibit inputs bool passEvent = g_pKeybindManager->onAxisEvent(e); if (!passEvent) @@ -862,6 +874,13 @@ void CInputManager::onMouseWheel(IPointer::SAxisEvent e) { g_pSeatManager->sendPointerAxis(e.timeMs, e.axis, delta, deltaDiscrete, value120, e.source, WL_POINTER_AXIS_RELATIVE_DIRECTION_IDENTICAL); } +void CInputManager::onMouseFrame() { + PROTO::inputCapture->sendFrame(); + + //TODO: Inhibit inputs + g_pSeatManager->sendPointerFrame(); +} + Vector2D CInputManager::getMouseCoordsInternal() { return g_pPointerManager->position(); } @@ -1325,6 +1344,9 @@ void CInputManager::onKeyboardKey(std::any event, SP pKeyboard) { auto e = std::any_cast(event); + //TODO: Inhibit inputs + PROTO::inputCapture->sendKey(e.keycode, (hyprlandInputCaptureManagerV1KeyState)e.state); + if (passEvent) { const auto IME = m_sIMERelay.m_pIME.lock(); diff --git a/src/managers/input/InputManager.hpp b/src/managers/input/InputManager.hpp index 36c8144dee0..4ae8eb10d63 100644 --- a/src/managers/input/InputManager.hpp +++ b/src/managers/input/InputManager.hpp @@ -87,6 +87,7 @@ class CInputManager { void onMouseWarp(IPointer::SMotionAbsoluteEvent); void onMouseButton(IPointer::SButtonEvent); void onMouseWheel(IPointer::SAxisEvent); + void onMouseFrame(); void onKeyboardKey(std::any, SP); void onKeyboardMod(SP); diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 889f2599cd0..1d9bdecffa8 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -31,3 +31,39 @@ void CInputCaptureProtocol::sendAbsoluteMotion(const Vector2D& absolutePosition, wl_fixed_from_double(delta.y)); } } + +void CInputCaptureProtocol::sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state) { + for (const UP& manager : m_vManagers) { + manager->sendKey(keyCode, state); + } +} + +void CInputCaptureProtocol::sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state) { + for (const UP& manager : m_vManagers) { + manager->sendButton(button, state); + } +} + +void CInputCaptureProtocol::sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value) { + for (const UP& manager : m_vManagers) { + manager->sendAxis(axis, value); + } +} + +void CInputCaptureProtocol::sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120) { + for (const UP& manager : m_vManagers) { + manager->sendAxisValue120(axis, value120); + } +} + +void CInputCaptureProtocol::sendAxisStop(hyprlandInputCaptureManagerV1Axis axis) { + for (const UP& manager : m_vManagers) { + manager->sendAxisStop(axis); + } +} + +void CInputCaptureProtocol::sendFrame() { + for (const UP& manager : m_vManagers) { + manager->sendFrame(); + } +} diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 172c17dcb5f..639a21f157e 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -9,12 +9,20 @@ class CInputCaptureProtocol : public IWaylandProtocol { CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name); virtual void bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id); void sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta); + void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); + void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); + void sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value); + void sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120); + void sendAxisStop(hyprlandInputCaptureManagerV1Axis axis); + + void sendFrame(); private: void onManagerResourceDestroy(wl_resource* res); void onCapture(CHyprlandInputCaptureManagerV1* pMgr); void onRelease(CHyprlandInputCaptureManagerV1* pMgr); + // std::vector> m_vManagers; }; diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index 1c4de2b22b9..53a994b2efb 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit 1c4de2b22b9a7100ca664e46e37e29b0b724ce97 +Subproject commit 53a994b2efbcc19862125fc9a8d5a752a24a0f20 From fbf1f88d8af356ae6810fed1eddc9d3af90f6dbb Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Thu, 26 Sep 2024 08:47:21 +0200 Subject: [PATCH 03/16] input-capture: inhibit inputs --- src/managers/PointerManager.cpp | 4 +++- src/managers/input/InputManager.cpp | 18 +++++++++++------- src/protocols/InputCapture.cpp | 16 +++++++++++++--- src/protocols/InputCapture.hpp | 20 +++++++++++++------- 4 files changed, 40 insertions(+), 18 deletions(-) diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index 63bebc22256..4c981143fd5 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -703,7 +703,9 @@ void CPointerManager::move(const Vector2D& deltaLogical) { PROTO::inputCapture->sendAbsoluteMotion(newPos, deltaLogical); - //TODO: Inhibit inputs + if (PROTO::inputCapture->isCaptured()) + return; + warpTo(newPos); } diff --git a/src/managers/input/InputManager.cpp b/src/managers/input/InputManager.cpp index bad91260c3c..8ac0e505a30 100644 --- a/src/managers/input/InputManager.cpp +++ b/src/managers/input/InputManager.cpp @@ -100,7 +100,9 @@ void CInputManager::onMouseMoved(IPointer::SMotionEvent e) { g_pPointerManager->move(DELTA); - //TODO: Inhibit inputs + if (PROTO::inputCapture->isCaptured()) + return; + mouseMoveUnified(e.timeMs, false, e.mouse); m_tmrLastCursorMovement.reset(); @@ -549,7 +551,9 @@ void CInputManager::onMouseButton(IPointer::SButtonEvent e) { PROTO::inputCapture->sendButton(e.button, (hyprlandInputCaptureManagerV1ButtonState)e.state); - //TODO: Inhibit inputs + if (PROTO::inputCapture->isCaptured()) + return; + m_tmrLastCursorMovement.reset(); if (e.state == WL_POINTER_BUTTON_STATE_PRESSED) { @@ -793,8 +797,7 @@ void CInputManager::onMouseWheel(IPointer::SAxisEvent e) { else if (e.delta == 0) PROTO::inputCapture->sendAxisStop((hyprlandInputCaptureManagerV1Axis)e.axis); - //TODO: Inhibit inputs - bool passEvent = g_pKeybindManager->onAxisEvent(e); + bool passEvent = !PROTO::inputCapture->isCaptured() && g_pKeybindManager->onAxisEvent(e); if (!passEvent) return; @@ -877,7 +880,9 @@ void CInputManager::onMouseWheel(IPointer::SAxisEvent e) { void CInputManager::onMouseFrame() { PROTO::inputCapture->sendFrame(); - //TODO: Inhibit inputs + if (PROTO::inputCapture->isCaptured()) + return; + g_pSeatManager->sendPointerFrame(); } @@ -1340,11 +1345,10 @@ void CInputManager::onKeyboardKey(std::any event, SP pKeyboard) { const auto EMAP = std::unordered_map{{"keyboard", pKeyboard}, {"event", event}}; EMIT_HOOK_EVENT_CANCELLABLE("keyPress", EMAP); - bool passEvent = DISALLOWACTION || g_pKeybindManager->onKeyEvent(event, pKeyboard); + bool passEvent = !PROTO::inputCapture->isCaptured() && (DISALLOWACTION || g_pKeybindManager->onKeyEvent(event, pKeyboard)); auto e = std::any_cast(event); - //TODO: Inhibit inputs PROTO::inputCapture->sendKey(e.keycode, (hyprlandInputCaptureManagerV1KeyState)e.state); if (passEvent) { diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 1d9bdecffa8..075c551ede6 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -5,7 +5,7 @@ #include CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name) : IWaylandProtocol(iface, ver, name) { - ; + active = false; } void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id) { @@ -21,9 +21,19 @@ void CInputCaptureProtocol::onManagerResourceDestroy(wl_resource* res) { std::erase_if(m_vManagers, [&](const auto& other) { return other->resource() == res; }); } -void CInputCaptureProtocol::onCapture(CHyprlandInputCaptureManagerV1* pMgr) {} +void CInputCaptureProtocol::onCapture(CHyprlandInputCaptureManagerV1* pMgr) { + Debug::log(LOG, "[input-capture] Input captured"); + active = true; +} + +void CInputCaptureProtocol::onRelease(CHyprlandInputCaptureManagerV1* pMgr) { + Debug::log(LOG, "[input-capture] Input released"); + active = false; +} -void CInputCaptureProtocol::onRelease(CHyprlandInputCaptureManagerV1* pMgr) {} +bool CInputCaptureProtocol::isCaptured() { + return active; +} void CInputCaptureProtocol::sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta) { for (const UP& manager : m_vManagers) { diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 639a21f157e..07af655359c 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -8,16 +8,22 @@ class CInputCaptureProtocol : public IWaylandProtocol { public: CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name); virtual void bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id); - void sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta); - void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); - void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); - void sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value); - void sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120); - void sendAxisStop(hyprlandInputCaptureManagerV1Axis axis); - void sendFrame(); + bool isCaptured(); + + // + void sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta); + void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); + void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); + void sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value); + void sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120); + void sendAxisStop(hyprlandInputCaptureManagerV1Axis axis); + + void sendFrame(); private: + bool active; + void onManagerResourceDestroy(wl_resource* res); void onCapture(CHyprlandInputCaptureManagerV1* pMgr); void onRelease(CHyprlandInputCaptureManagerV1* pMgr); From c92e592a553ddc7633784380802a071158993f29 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Thu, 26 Sep 2024 12:06:17 +0200 Subject: [PATCH 04/16] input-capture: code cleanup --- src/protocols/InputCapture.cpp | 4 ---- src/protocols/InputCapture.hpp | 1 - 2 files changed, 5 deletions(-) diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 075c551ede6..e8ac910ad7e 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -1,8 +1,4 @@ #include "InputCapture.hpp" -#include "hyprland-input-capture-v1.hpp" -#include -#include -#include CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name) : IWaylandProtocol(iface, ver, name) { active = false; diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 07af655359c..1c6685f5028 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -1,6 +1,5 @@ #pragma once -#include "WaylandProtocol.hpp" #include "hyprland-input-capture-v1.hpp" #include From bc6b47234d35279da52e0fba20d7d92cd7190f69 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Thu, 26 Sep 2024 14:09:35 +0200 Subject: [PATCH 05/16] input-capture: fix build --- src/protocols/InputCapture.hpp | 1 + 1 file changed, 1 insertion(+) diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 1c6685f5028..fafe81bbc87 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -1,6 +1,7 @@ #pragma once #include "hyprland-input-capture-v1.hpp" +#include "../protocols/WaylandProtocol.hpp" #include class CInputCaptureProtocol : public IWaylandProtocol { From 0c2c8f2d883592b8b028a14f5c20ccb8df1b4b98 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Mon, 30 Sep 2024 12:31:31 +0200 Subject: [PATCH 06/16] input-capture: Init active property & upstream protocol changes --- src/managers/PointerManager.cpp | 3 +-- src/protocols/InputCapture.cpp | 6 +++--- src/protocols/InputCapture.hpp | 4 ++-- subprojects/hyprland-protocols | 2 +- 4 files changed, 7 insertions(+), 8 deletions(-) diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index 4c981143fd5..f4701dfc27a 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -700,8 +700,7 @@ void CPointerManager::move(const Vector2D& deltaLogical) { const auto oldPos = pointerPos; auto newPos = oldPos + Vector2D{std::isnan(deltaLogical.x) ? 0.0 : deltaLogical.x, std::isnan(deltaLogical.y) ? 0.0 : deltaLogical.y}; - - PROTO::inputCapture->sendAbsoluteMotion(newPos, deltaLogical); + PROTO::inputCapture->sendMotion(newPos, deltaLogical); if (PROTO::inputCapture->isCaptured()) return; diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index e8ac910ad7e..2fa99ac2a9f 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -1,7 +1,7 @@ #include "InputCapture.hpp" CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name) : IWaylandProtocol(iface, ver, name) { - active = false; + ; } void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id) { @@ -31,9 +31,9 @@ bool CInputCaptureProtocol::isCaptured() { return active; } -void CInputCaptureProtocol::sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta) { +void CInputCaptureProtocol::sendMotion(const Vector2D& absolutePosition, const Vector2D& delta) { for (const UP& manager : m_vManagers) { - manager->sendAbsoluteMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), + manager->sendMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), wl_fixed_from_double(delta.y)); } } diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index fafe81bbc87..efe69463fad 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -12,7 +12,7 @@ class CInputCaptureProtocol : public IWaylandProtocol { bool isCaptured(); // - void sendAbsoluteMotion(const Vector2D& absolutePosition, const Vector2D& delta); + void sendMotion(const Vector2D& absolutePosition, const Vector2D& delta); void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); void sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value); @@ -22,7 +22,7 @@ class CInputCaptureProtocol : public IWaylandProtocol { void sendFrame(); private: - bool active; + bool active = false; void onManagerResourceDestroy(wl_resource* res); void onCapture(CHyprlandInputCaptureManagerV1* pMgr); diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index 53a994b2efb..479cc226451 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit 53a994b2efbcc19862125fc9a8d5a752a24a0f20 +Subproject commit 479cc226451c264396a4c442710d6b56dce2fa46 From 38bc5f553e1d003132e1db96720a780625257a7e Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Mon, 30 Sep 2024 17:13:37 +0200 Subject: [PATCH 07/16] input-capture: fixes --- src/protocols/InputCapture.cpp | 43 ++++++++++++++-------------------- src/protocols/InputCapture.hpp | 7 +----- 2 files changed, 18 insertions(+), 32 deletions(-) diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 2fa99ac2a9f..c2cfd6f2d1c 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -7,24 +7,16 @@ CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const in void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id) { const auto RESOURCE = m_vManagers.emplace_back(std::make_unique(client, ver, id)).get(); - RESOURCE->setOnDestroy([this](CHyprlandInputCaptureManagerV1* p) { this->onManagerResourceDestroy(p->resource()); }); + RESOURCE->setOnDestroy([this](CHyprlandInputCaptureManagerV1* p) { std::erase_if(m_vManagers, [&](const auto& other) { return other->resource() == p->resource(); }); }); - RESOURCE->setCapture([this](CHyprlandInputCaptureManagerV1* p) { this->onCapture(p); }); - RESOURCE->setRelease([this](CHyprlandInputCaptureManagerV1* p) { this->onRelease(p); }); -} - -void CInputCaptureProtocol::onManagerResourceDestroy(wl_resource* res) { - std::erase_if(m_vManagers, [&](const auto& other) { return other->resource() == res; }); -} - -void CInputCaptureProtocol::onCapture(CHyprlandInputCaptureManagerV1* pMgr) { - Debug::log(LOG, "[input-capture] Input captured"); - active = true; -} - -void CInputCaptureProtocol::onRelease(CHyprlandInputCaptureManagerV1* pMgr) { - Debug::log(LOG, "[input-capture] Input released"); - active = false; + RESOURCE->setCapture([this](CHyprlandInputCaptureManagerV1* p) { + Debug::log(LOG, "[input-capture] Input captured"); + active = true; + }); + RESOURCE->setRelease([this](CHyprlandInputCaptureManagerV1* p) { + Debug::log(LOG, "[input-capture] Input released"); + active = false; + }); } bool CInputCaptureProtocol::isCaptured() { @@ -32,44 +24,43 @@ bool CInputCaptureProtocol::isCaptured() { } void CInputCaptureProtocol::sendMotion(const Vector2D& absolutePosition, const Vector2D& delta) { - for (const UP& manager : m_vManagers) { - manager->sendMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), - wl_fixed_from_double(delta.y)); + for (const auto& manager : m_vManagers) { + manager->sendMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), wl_fixed_from_double(delta.y)); } } void CInputCaptureProtocol::sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state) { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendKey(keyCode, state); } } void CInputCaptureProtocol::sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state) { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendButton(button, state); } } void CInputCaptureProtocol::sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value) { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendAxis(axis, value); } } void CInputCaptureProtocol::sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120) { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendAxisValue120(axis, value120); } } void CInputCaptureProtocol::sendAxisStop(hyprlandInputCaptureManagerV1Axis axis) { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendAxisStop(axis); } } void CInputCaptureProtocol::sendFrame() { - for (const UP& manager : m_vManagers) { + for (const auto& manager : m_vManagers) { manager->sendFrame(); } } diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index efe69463fad..645c25ecca4 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -9,7 +9,7 @@ class CInputCaptureProtocol : public IWaylandProtocol { CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name); virtual void bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id); - bool isCaptured(); + bool isCaptured(); // void sendMotion(const Vector2D& absolutePosition, const Vector2D& delta); @@ -23,11 +23,6 @@ class CInputCaptureProtocol : public IWaylandProtocol { private: bool active = false; - - void onManagerResourceDestroy(wl_resource* res); - void onCapture(CHyprlandInputCaptureManagerV1* pMgr); - void onRelease(CHyprlandInputCaptureManagerV1* pMgr); - // std::vector> m_vManagers; }; From bc5f6cdf19bd085141ed69d3963af8edb831c65e Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Mon, 7 Oct 2024 14:48:52 +0200 Subject: [PATCH 08/16] input-capture: impl keymap --- src/managers/SeatManager.cpp | 2 ++ src/protocols/InputCapture.cpp | 58 ++++++++++++++++++++++++++-------- src/protocols/InputCapture.hpp | 7 ++-- subprojects/hyprland-protocols | 2 +- 4 files changed, 53 insertions(+), 16 deletions(-) diff --git a/src/managers/SeatManager.cpp b/src/managers/SeatManager.cpp index b169faeb6ab..71c5e41a9d1 100644 --- a/src/managers/SeatManager.cpp +++ b/src/managers/SeatManager.cpp @@ -4,6 +4,7 @@ #include "../protocols/DataDeviceWlr.hpp" #include "../protocols/PrimarySelection.hpp" #include "../protocols/core/Compositor.hpp" +#include "../protocols/InputCapture.hpp" #include "../Compositor.hpp" #include "../devices/IKeyboard.hpp" #include "wlr-layer-shell-unstable-v1.hpp" @@ -97,6 +98,7 @@ void CSeatManager::updateActiveKeyboardData() { if (keyboard) PROTO::seat->updateRepeatInfo(keyboard->repeatRate, keyboard->repeatDelay); PROTO::seat->updateKeymap(); + PROTO::inputCapture->updateKeymap(); } void CSeatManager::setKeyboardFocus(SP surf) { diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index c2cfd6f2d1c..0381d63c60f 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -1,11 +1,15 @@ #include "InputCapture.hpp" +#include "../devices/IKeyboard.hpp" +#include "managers/SeatManager.hpp" +#include + CInputCaptureProtocol::CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name) : IWaylandProtocol(iface, ver, name) { ; } void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id) { - const auto RESOURCE = m_vManagers.emplace_back(std::make_unique(client, ver, id)).get(); + const auto& RESOURCE = m_vManagers.emplace_back(std::make_unique(client, ver, id)); RESOURCE->setOnDestroy([this](CHyprlandInputCaptureManagerV1* p) { std::erase_if(m_vManagers, [&](const auto& other) { return other->resource() == p->resource(); }); }); @@ -17,50 +21,78 @@ void CInputCaptureProtocol::bindManager(wl_client* client, void* data, uint32_t Debug::log(LOG, "[input-capture] Input released"); active = false; }); + + sendKeymap(g_pSeatManager->keyboard.lock(), RESOURCE); } bool CInputCaptureProtocol::isCaptured() { return active; } +void CInputCaptureProtocol::updateKeymap() { + for (const auto& manager : m_vManagers) + sendKeymap(g_pSeatManager->keyboard.lock(), manager); +} + void CInputCaptureProtocol::sendMotion(const Vector2D& absolutePosition, const Vector2D& delta) { for (const auto& manager : m_vManagers) { manager->sendMotion(wl_fixed_from_double(absolutePosition.x), wl_fixed_from_double(absolutePosition.y), wl_fixed_from_double(delta.x), wl_fixed_from_double(delta.y)); } } +void CInputCaptureProtocol::sendKeymap(SP keyboard, const std::unique_ptr& manager) { + if (!keyboard) + return; + + hyprlandInputCaptureManagerV1KeymapFormat format; + int fd; + uint32_t size; + if (keyboard) { + format = HYPRLAND_INPUT_CAPTURE_MANAGER_V1_KEYMAP_FORMAT_XKB_V1; + fd = keyboard->xkbKeymapFD; + size = keyboard->xkbKeymapString.length() + 1; + } else { + format = HYPRLAND_INPUT_CAPTURE_MANAGER_V1_KEYMAP_FORMAT_NO_KEYMAP; + fd = open("/dev/null", O_RDONLY | O_CLOEXEC); + if (fd < 0) { + LOGM(ERR, "Failed to open /dev/null"); + return; + } + size = 0; + } + + manager->sendKeymap(format, fd, size); + + if (!keyboard) + close(fd); +} + void CInputCaptureProtocol::sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state) { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendKey(keyCode, state); - } } void CInputCaptureProtocol::sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state) { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendButton(button, state); - } } void CInputCaptureProtocol::sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value) { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendAxis(axis, value); - } } void CInputCaptureProtocol::sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120) { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendAxisValue120(axis, value120); - } } void CInputCaptureProtocol::sendAxisStop(hyprlandInputCaptureManagerV1Axis axis) { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendAxisStop(axis); - } } void CInputCaptureProtocol::sendFrame() { - for (const auto& manager : m_vManagers) { + for (const auto& manager : m_vManagers) manager->sendFrame(); - } } diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 645c25ecca4..3f174448c56 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -9,9 +9,10 @@ class CInputCaptureProtocol : public IWaylandProtocol { CInputCaptureProtocol(const wl_interface* iface, const int& ver, const std::string& name); virtual void bindManager(wl_client* client, void* data, uint32_t ver, uint32_t id); - bool isCaptured(); - // + bool isCaptured(); + + void updateKeymap(); void sendMotion(const Vector2D& absolutePosition, const Vector2D& delta); void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); @@ -22,6 +23,8 @@ class CInputCaptureProtocol : public IWaylandProtocol { void sendFrame(); private: + void sendKeymap(SP keyboard, const std::unique_ptr& manager); + bool active = false; // std::vector> m_vManagers; diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index 479cc226451..d3674e1f4ea 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit 479cc226451c264396a4c442710d6b56dce2fa46 +Subproject commit d3674e1f4eac730efc01c08e794a988be31ec73e From c240846f88d7100183d410ec21efb6d13f25395d Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Mon, 21 Oct 2024 10:27:03 +0200 Subject: [PATCH 09/16] input-capture: impl force release --- src/managers/KeybindManager.cpp | 15 +++++++++++++++ src/managers/KeybindManager.hpp | 1 + src/managers/input/InputManager.cpp | 2 +- src/protocols/InputCapture.cpp | 8 ++++++++ src/protocols/InputCapture.hpp | 2 ++ subprojects/hyprland-protocols | 2 +- 6 files changed, 28 insertions(+), 2 deletions(-) diff --git a/src/managers/KeybindManager.cpp b/src/managers/KeybindManager.cpp index e1ad7d609ca..a2debd8124a 100644 --- a/src/managers/KeybindManager.cpp +++ b/src/managers/KeybindManager.cpp @@ -5,6 +5,7 @@ #include "../protocols/ShortcutsInhibit.hpp" #include "../protocols/GlobalShortcuts.hpp" #include "../render/decorations/CHyprGroupBarDecoration.hpp" +#include "../devices/IKeyboard.hpp" #include "KeybindManager.hpp" #include "PointerManager.hpp" #include "Compositor.hpp" @@ -12,6 +13,7 @@ #include "eventLoop/EventLoopManager.hpp" #include "debug/Log.hpp" #include "helpers/varlist/VarList.hpp" +#include "protocols/InputCapture.hpp" #include #include @@ -126,6 +128,7 @@ CKeybindManager::CKeybindManager() { m_mDispatchers["event"] = event; m_mDispatchers["global"] = global; m_mDispatchers["setprop"] = setProp; + m_mDispatchers["releaseinputcapture"] = releaseInputCapture; m_tScrollTimer.reset(); @@ -743,6 +746,14 @@ SDispatchResult CKeybindManager::handleKeybinds(const uint32_t modmask, const SP m_iPassPressed = (int)pressed; + // We only process the releaseinputcapture dispatcher when input capture is active + if (PROTO::inputCapture->isCaptured()) { + if (k->handler == "releaseinputcapture") + res = DISPATCHER->second(k->arg); + else + break; + } + // if the dispatchers says to pass event then we will if (k->handler == "mouse") res = DISPATCHER->second((pressed ? "1" : "0") + k->arg); @@ -3053,3 +3064,7 @@ SDispatchResult CKeybindManager::setProp(std::string args) { return {}; } +SDispatchResult CKeybindManager::releaseInputCapture(std::string args) { + PROTO::inputCapture->forceRelease(); + return {}; +} diff --git a/src/managers/KeybindManager.hpp b/src/managers/KeybindManager.hpp index c8c6266cbdf..d268d3a1b86 100644 --- a/src/managers/KeybindManager.hpp +++ b/src/managers/KeybindManager.hpp @@ -215,6 +215,7 @@ class CKeybindManager { static SDispatchResult global(std::string); static SDispatchResult event(std::string); static SDispatchResult setProp(std::string); + static SDispatchResult releaseInputCapture(std::string); friend class CCompositor; friend class CInputManager; diff --git a/src/managers/input/InputManager.cpp b/src/managers/input/InputManager.cpp index 8ac0e505a30..ee60a1d6316 100644 --- a/src/managers/input/InputManager.cpp +++ b/src/managers/input/InputManager.cpp @@ -1345,7 +1345,7 @@ void CInputManager::onKeyboardKey(std::any event, SP pKeyboard) { const auto EMAP = std::unordered_map{{"keyboard", pKeyboard}, {"event", event}}; EMIT_HOOK_EVENT_CANCELLABLE("keyPress", EMAP); - bool passEvent = !PROTO::inputCapture->isCaptured() && (DISALLOWACTION || g_pKeybindManager->onKeyEvent(event, pKeyboard)); + bool passEvent = (DISALLOWACTION || g_pKeybindManager->onKeyEvent(event, pKeyboard)) && !PROTO::inputCapture->isCaptured(); auto e = std::any_cast(event); diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 0381d63c60f..b305827e0ea 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -67,6 +67,14 @@ void CInputCaptureProtocol::sendKeymap(SP keyboard, const std::unique close(fd); } +void CInputCaptureProtocol::forceRelease() { + Debug::log(LOG, "[input-capture] Force Input released"); + active = false; + + for (const auto& manager : m_vManagers) + manager->sendForceRelease(); +} + void CInputCaptureProtocol::sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state) { for (const auto& manager : m_vManagers) manager->sendKey(keyCode, state); diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 3f174448c56..90dd74e5c38 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -13,6 +13,8 @@ class CInputCaptureProtocol : public IWaylandProtocol { bool isCaptured(); void updateKeymap(); + void forceRelease(); + void sendMotion(const Vector2D& absolutePosition, const Vector2D& delta); void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index d3674e1f4ea..0c7cf263fae 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit d3674e1f4eac730efc01c08e794a988be31ec73e +Subproject commit 0c7cf263faeae9429942c6ffbc1e9b5dfd709bf4 From e2fe005ad5126ebba093454749cb7f938d59ae60 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Tue, 22 Oct 2024 11:54:16 +0200 Subject: [PATCH 10/16] input-capture: fix rebase issues --- src/managers/PointerManager.cpp | 9 +-------- src/managers/input/InputManager.cpp | 9 ++++++++- src/protocols/core/Output.cpp | 3 --- 3 files changed, 9 insertions(+), 12 deletions(-) diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index f4701dfc27a..e330ddf5787 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -876,14 +876,7 @@ void CPointerManager::attachPointer(SP pointer) { }); listener->frame = pointer->pointerEvents.frame.registerListener([] (std::any e) { - bool shouldSkip = false; - if (!g_pSeatManager->mouse.expired() && g_pInputManager->isLocked()) { - auto PMONITOR = g_pCompositor->m_pLastMonitor.get(); - shouldSkip = PMONITOR && PMONITOR->shouldSkipScheduleFrameOnMouseEvent(); - } - g_pSeatManager->isPointerFrameSkipped = shouldSkip; - if (!g_pSeatManager->isPointerFrameSkipped) - g_pSeatManager->sendPointerFrame(); + g_pInputManager->onMouseFrame(); }); listener->swipeBegin = pointer->pointerEvents.swipeBegin.registerListener([] (std::any e) { diff --git a/src/managers/input/InputManager.cpp b/src/managers/input/InputManager.cpp index ee60a1d6316..4484b57822a 100644 --- a/src/managers/input/InputManager.cpp +++ b/src/managers/input/InputManager.cpp @@ -883,7 +883,14 @@ void CInputManager::onMouseFrame() { if (PROTO::inputCapture->isCaptured()) return; - g_pSeatManager->sendPointerFrame(); + bool shouldSkip = false; + if (!g_pSeatManager->mouse.expired() && g_pInputManager->isLocked()) { + auto PMONITOR = g_pCompositor->m_pLastMonitor.get(); + shouldSkip = PMONITOR && PMONITOR->shouldSkipScheduleFrameOnMouseEvent(); + } + g_pSeatManager->isPointerFrameSkipped = shouldSkip; + if (!g_pSeatManager->isPointerFrameSkipped) + g_pSeatManager->sendPointerFrame(); } Vector2D CInputManager::getMouseCoordsInternal() { diff --git a/src/protocols/core/Output.cpp b/src/protocols/core/Output.cpp index ea4183d7171..c1b984d8e99 100644 --- a/src/protocols/core/Output.cpp +++ b/src/protocols/core/Output.cpp @@ -78,9 +78,6 @@ void CWLOutputResource::updateState() { monitor->output->model.c_str(), monitor->transform); resource->sendMode((wl_output_mode)(WL_OUTPUT_MODE_CURRENT), monitor->vecPixelSize.x, monitor->vecPixelSize.y, monitor->refreshRate * 1000.0); - resource->sendGeometry(0, 0, monitor->output->physicalSize.x, monitor->output->physicalSize.y, (wl_output_subpixel)monitor->output->subpixel, monitor->output->make.c_str(), - monitor->output->model.c_str(), monitor->transform); - if (resource->version() >= 2) resource->sendDone(); } From c8b8ec020c6c094aca18a41cf1a72cceecb791ab Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Tue, 17 Dec 2024 18:41:55 +0100 Subject: [PATCH 11/16] input-capture: add protocol to meson.build --- protocols/meson.build | 1 + 1 file changed, 1 insertion(+) diff --git a/protocols/meson.build b/protocols/meson.build index 2c0d06d1221..cbc79d1f777 100644 --- a/protocols/meson.build +++ b/protocols/meson.build @@ -37,6 +37,7 @@ protocols = [ hyprland_protocol_dir / 'protocols/hyprland-toplevel-export-v1.xml', hyprland_protocol_dir / 'protocols/hyprland-focus-grab-v1.xml', hyprland_protocol_dir / 'protocols/hyprland-ctm-control-v1.xml', + hyprland_protocol_dir / 'protocols/hyprland-input-capture-v1.xml', wayland_protocol_dir / 'staging/tearing-control/tearing-control-v1.xml', wayland_protocol_dir / 'staging/fractional-scale/fractional-scale-v1.xml', wayland_protocol_dir / 'unstable/xdg-output/xdg-output-unstable-v1.xml', From 70a8353bfaf6ac95a1c24a411bf0a92854ba3ede Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Tue, 17 Dec 2024 19:03:25 +0100 Subject: [PATCH 12/16] input-capture: fix build --- src/protocols/InputCapture.cpp | 1 - src/protocols/InputCapture.hpp | 1 + 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index b305827e0ea..5263911309e 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -1,6 +1,5 @@ #include "InputCapture.hpp" -#include "../devices/IKeyboard.hpp" #include "managers/SeatManager.hpp" #include diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index 90dd74e5c38..f22f31b331a 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -2,6 +2,7 @@ #include "hyprland-input-capture-v1.hpp" #include "../protocols/WaylandProtocol.hpp" +#include "../devices/IKeyboard.hpp" #include class CInputCaptureProtocol : public IWaylandProtocol { From 56a9676af4647bb535882292e8930d1c1ab18197 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Tue, 17 Dec 2024 19:20:52 +0100 Subject: [PATCH 13/16] input-capture: send modifiers (highly experimental) --- src/managers/input/InputManager.cpp | 5 +++++ src/protocols/InputCapture.cpp | 6 ++++++ src/protocols/InputCapture.hpp | 1 + subprojects/hyprland-protocols | 2 +- 4 files changed, 13 insertions(+), 1 deletion(-) diff --git a/src/managers/input/InputManager.cpp b/src/managers/input/InputManager.cpp index 4484b57822a..52feee892eb 100644 --- a/src/managers/input/InputManager.cpp +++ b/src/managers/input/InputManager.cpp @@ -1384,6 +1384,11 @@ void CInputManager::onKeyboardMod(SP pKeyboard) { auto MODS = pKeyboard->modifiersState; MODS.depressed = ALLMODS; + PROTO::inputCapture->sendModifiers(MODS.depressed,MODS.latched, MODS.locked, MODS.group); + + if (PROTO::inputCapture->isCaptured()) + return; + const auto IME = m_sIMERelay.m_pIME.lock(); if (IME && IME->hasGrab() && !DISALLOWACTION) { diff --git a/src/protocols/InputCapture.cpp b/src/protocols/InputCapture.cpp index 5263911309e..c91d13f7009 100644 --- a/src/protocols/InputCapture.cpp +++ b/src/protocols/InputCapture.cpp @@ -79,6 +79,12 @@ void CInputCaptureProtocol::sendKey(uint32_t keyCode, hyprlandInputCaptureManage manager->sendKey(keyCode, state); } +void CInputCaptureProtocol::sendModifiers(uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked, uint32_t group) { + for (const auto& manager : m_vManagers) + manager->sendModifiers(mods_depressed, mods_locked, mods_locked, group); +} + + void CInputCaptureProtocol::sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state) { for (const auto& manager : m_vManagers) manager->sendButton(button, state); diff --git a/src/protocols/InputCapture.hpp b/src/protocols/InputCapture.hpp index f22f31b331a..eb63bdd5e56 100644 --- a/src/protocols/InputCapture.hpp +++ b/src/protocols/InputCapture.hpp @@ -18,6 +18,7 @@ class CInputCaptureProtocol : public IWaylandProtocol { void sendMotion(const Vector2D& absolutePosition, const Vector2D& delta); void sendKey(uint32_t keyCode, hyprlandInputCaptureManagerV1KeyState state); + void sendModifiers(uint32_t mods_depressed, uint32_t mods_latched, uint32_t mods_locked, uint32_t group); void sendButton(uint32_t button, hyprlandInputCaptureManagerV1ButtonState state); void sendAxis(hyprlandInputCaptureManagerV1Axis axis, double value); void sendAxisValue120(hyprlandInputCaptureManagerV1Axis axis, int32_t value120); diff --git a/subprojects/hyprland-protocols b/subprojects/hyprland-protocols index 0c7cf263fae..cb1b99ca037 160000 --- a/subprojects/hyprland-protocols +++ b/subprojects/hyprland-protocols @@ -1 +1 @@ -Subproject commit 0c7cf263faeae9429942c6ffbc1e9b5dfd709bf4 +Subproject commit cb1b99ca037311417f9fde600732f21dc9ed983e From 981e19c74a51a1449f69fda7e62d67a96c790727 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Tue, 17 Dec 2024 20:41:27 +0100 Subject: [PATCH 14/16] input-capture: don't send motion when the input is locked (like inside a 3d game) --- src/managers/PointerManager.cpp | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/managers/PointerManager.cpp b/src/managers/PointerManager.cpp index e330ddf5787..d44e883d91e 100644 --- a/src/managers/PointerManager.cpp +++ b/src/managers/PointerManager.cpp @@ -700,7 +700,9 @@ void CPointerManager::move(const Vector2D& deltaLogical) { const auto oldPos = pointerPos; auto newPos = oldPos + Vector2D{std::isnan(deltaLogical.x) ? 0.0 : deltaLogical.x, std::isnan(deltaLogical.y) ? 0.0 : deltaLogical.y}; - PROTO::inputCapture->sendMotion(newPos, deltaLogical); + + if (!g_pInputManager->isLocked()) + PROTO::inputCapture->sendMotion(newPos, deltaLogical); if (PROTO::inputCapture->isCaptured()) return; From 54f356dd8d3deb99b8e2115d9a3e3268a767d4f5 Mon Sep 17 00:00:00 2001 From: Gwilherm Folliot Date: Wed, 18 Dec 2024 02:05:41 +0100 Subject: [PATCH 15/16] input-capture: hide cursor when the input is captured --- src/render/Renderer.cpp | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/src/render/Renderer.cpp b/src/render/Renderer.cpp index d8b8a847db6..e8553b5b1ea 100644 --- a/src/render/Renderer.cpp +++ b/src/render/Renderer.cpp @@ -19,6 +19,7 @@ #include "../protocols/core/Compositor.hpp" #include "../protocols/DRMSyncobj.hpp" #include "../protocols/LinuxDMABUF.hpp" +#include "../protocols/InputCapture.hpp" #include "../helpers/sync/SyncTimeline.hpp" #include "pass/TexPassElement.hpp" #include "pass/ClearPassElement.hpp" @@ -2290,7 +2291,9 @@ void CHyprRenderer::ensureCursorRenderingMode() { if (*PCURSORTIMEOUT > 0) m_sCursorHiddenConditions.hiddenOnTimeout = *PCURSORTIMEOUT < g_pInputManager->m_tmrLastCursorMovement.getSeconds(); - const bool HIDE = m_sCursorHiddenConditions.hiddenOnTimeout || m_sCursorHiddenConditions.hiddenOnTouch || m_sCursorHiddenConditions.hiddenOnKeyboard; + + const bool HIDE = m_sCursorHiddenConditions.hiddenOnTimeout || m_sCursorHiddenConditions.hiddenOnTouch || m_sCursorHiddenConditions.hiddenOnKeyboard || + PROTO::inputCapture->isCaptured(); if (HIDE == m_bCursorHidden) return; From 50fb4947a4a8a184720beca7363e1dd58bd4bde6 Mon Sep 17 00:00:00 2001 From: Alex Moon Date: Fri, 20 Dec 2024 14:40:45 +0000 Subject: [PATCH 16/16] change repo URL in plugin manager (#2) --- hyprpm/src/core/PluginManager.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/hyprpm/src/core/PluginManager.cpp b/hyprpm/src/core/PluginManager.cpp index 48b108c8493..8f458f22329 100644 --- a/hyprpm/src/core/PluginManager.cpp +++ b/hyprpm/src/core/PluginManager.cpp @@ -464,7 +464,7 @@ bool CPluginManager::updateHeaders(bool force) { return false; } - progress.printMessageAbove(statusString("!", Colors::YELLOW, "Cloning https://github.com/hyprwm/Hyprland, this might take a moment.")); + progress.printMessageAbove(statusString("!", Colors::YELLOW, "Cloning https://github.com/3l0w/Hyprland, this might take a moment.")); const bool bShallow = (HLVER.branch == "main") && !m_bNoShallow; @@ -475,12 +475,12 @@ bool CPluginManager::updateHeaders(bool force) { if (m_bVerbose && bShallow) progress.printMessageAbove(verboseString("will shallow since: {}", SHALLOW_DATE)); - std::string ret = execAndGet(std::format("cd {} && git clone --recursive https://github.com/hyprwm/Hyprland hyprland-{}{}", getTempRoot(), USERNAME, + std::string ret = execAndGet(std::format("cd {} && git clone --recursive https://github.com/3l0w/Hyprland hyprland-{}{}", getTempRoot(), USERNAME, (bShallow ? " --shallow-since='" + SHALLOW_DATE + "'" : ""))); if (!std::filesystem::exists(WORKINGDIR)) { progress.printMessageAbove(failureString("Clone failed. Retrying without shallow.")); - ret = execAndGet(std::format("cd {} && git clone --recursive https://github.com/hyprwm/hyprland hyprland-{}", getTempRoot(), USERNAME)); + ret = execAndGet(std::format("cd {} && git clone --recursive https://github.com/3l0w/hyprland hyprland-{}", getTempRoot(), USERNAME)); } if (!std::filesystem::exists(WORKINGDIR + "/.git")) {