cursor: refactor override handling (#12166)

much cleaner and more reliable. Should fix https://github.com/hyprwm/Hyprland/issues/12088
This commit is contained in:
Vaxry 2025-10-31 00:14:08 +00:00 committed by GitHub
parent 6ade4d58ca
commit 5e6cec962c
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 152 additions and 102 deletions

View file

@ -12,6 +12,7 @@
#include "../managers/LayoutManager.hpp" #include "../managers/LayoutManager.hpp"
#include "../managers/EventManager.hpp" #include "../managers/EventManager.hpp"
#include "../managers/HookSystemManager.hpp" #include "../managers/HookSystemManager.hpp"
#include "../managers/cursor/CursorShapeOverrideController.hpp"
void IHyprLayout::onWindowCreated(PHLWINDOW pWindow, eDirection direction) { void IHyprLayout::onWindowCreated(PHLWINDOW pWindow, eDirection direction) {
CBox desiredGeometry = g_pXWaylandManager->getGeometryForWindow(pWindow); CBox desiredGeometry = g_pXWaylandManager->getGeometryForWindow(pWindow);
@ -243,7 +244,7 @@ void IHyprLayout::onBeginDragWindow() {
// Window will be floating. Let's check if it's valid. It should be, but I don't like crashing. // Window will be floating. Let's check if it's valid. It should be, but I don't like crashing.
if (!validMapped(DRAGGINGWINDOW)) { if (!validMapped(DRAGGINGWINDOW)) {
Debug::log(ERR, "Dragging attempted on an invalid window!"); Debug::log(ERR, "Dragging attempted on an invalid window!");
g_pKeybindManager->changeMouseBindMode(MBIND_INVALID); CKeybindManager::changeMouseBindMode(MBIND_INVALID);
return; return;
} }
@ -259,41 +260,41 @@ void IHyprLayout::onBeginDragWindow() {
switch (*RESIZECORNER) { switch (*RESIZECORNER) {
case 1: case 1:
m_grabbedCorner = CORNER_TOPLEFT; m_grabbedCorner = CORNER_TOPLEFT;
g_pInputManager->setCursorImageUntilUnset("nw-resize"); Cursor::overrideController->setOverride("nw-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
break; break;
case 2: case 2:
m_grabbedCorner = CORNER_TOPRIGHT; m_grabbedCorner = CORNER_TOPRIGHT;
g_pInputManager->setCursorImageUntilUnset("ne-resize"); Cursor::overrideController->setOverride("ne-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
break; break;
case 3: case 3:
m_grabbedCorner = CORNER_BOTTOMRIGHT; m_grabbedCorner = CORNER_BOTTOMRIGHT;
g_pInputManager->setCursorImageUntilUnset("se-resize"); Cursor::overrideController->setOverride("se-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
break; break;
case 4: case 4:
m_grabbedCorner = CORNER_BOTTOMLEFT; m_grabbedCorner = CORNER_BOTTOMLEFT;
g_pInputManager->setCursorImageUntilUnset("sw-resize"); Cursor::overrideController->setOverride("sw-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
break; break;
} }
} else if (m_beginDragXY.x < m_beginDragPositionXY.x + m_beginDragSizeXY.x / 2.0) { } else if (m_beginDragXY.x < m_beginDragPositionXY.x + m_beginDragSizeXY.x / 2.0) {
if (m_beginDragXY.y < m_beginDragPositionXY.y + m_beginDragSizeXY.y / 2.0) { if (m_beginDragXY.y < m_beginDragPositionXY.y + m_beginDragSizeXY.y / 2.0) {
m_grabbedCorner = CORNER_TOPLEFT; m_grabbedCorner = CORNER_TOPLEFT;
g_pInputManager->setCursorImageUntilUnset("nw-resize"); Cursor::overrideController->setOverride("nw-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
} else { } else {
m_grabbedCorner = CORNER_BOTTOMLEFT; m_grabbedCorner = CORNER_BOTTOMLEFT;
g_pInputManager->setCursorImageUntilUnset("sw-resize"); Cursor::overrideController->setOverride("sw-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
} }
} else { } else {
if (m_beginDragXY.y < m_beginDragPositionXY.y + m_beginDragSizeXY.y / 2.0) { if (m_beginDragXY.y < m_beginDragPositionXY.y + m_beginDragSizeXY.y / 2.0) {
m_grabbedCorner = CORNER_TOPRIGHT; m_grabbedCorner = CORNER_TOPRIGHT;
g_pInputManager->setCursorImageUntilUnset("ne-resize"); Cursor::overrideController->setOverride("ne-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
} else { } else {
m_grabbedCorner = CORNER_BOTTOMRIGHT; m_grabbedCorner = CORNER_BOTTOMRIGHT;
g_pInputManager->setCursorImageUntilUnset("se-resize"); Cursor::overrideController->setOverride("se-resize", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
} }
} }
if (g_pInputManager->m_dragMode != MBIND_RESIZE && g_pInputManager->m_dragMode != MBIND_RESIZE_FORCE_RATIO && g_pInputManager->m_dragMode != MBIND_RESIZE_BLOCK_RATIO) if (g_pInputManager->m_dragMode != MBIND_RESIZE && g_pInputManager->m_dragMode != MBIND_RESIZE_FORCE_RATIO && g_pInputManager->m_dragMode != MBIND_RESIZE_BLOCK_RATIO)
g_pInputManager->setCursorImageUntilUnset("grabbing"); Cursor::overrideController->setOverride("grabbing", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
g_pHyprRenderer->damageWindow(DRAGGINGWINDOW); g_pHyprRenderer->damageWindow(DRAGGINGWINDOW);
@ -310,13 +311,13 @@ void IHyprLayout::onEndDragWindow() {
if (!validMapped(DRAGGINGWINDOW)) { if (!validMapped(DRAGGINGWINDOW)) {
if (DRAGGINGWINDOW) { if (DRAGGINGWINDOW) {
g_pInputManager->unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
g_pInputManager->m_currentlyDraggedWindow.reset(); g_pInputManager->m_currentlyDraggedWindow.reset();
} }
return; return;
} }
g_pInputManager->unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
g_pInputManager->m_currentlyDraggedWindow.reset(); g_pInputManager->m_currentlyDraggedWindow.reset();
g_pInputManager->m_wasDraggingWindow = true; g_pInputManager->m_wasDraggingWindow = true;

View file

@ -0,0 +1,43 @@
#include "CursorShapeOverrideController.hpp"
#include <ranges>
using namespace Cursor;
void CShapeOverrideController::setOverride(const std::string& name, eCursorShapeOverrideGroup group) {
if (m_overrides[group] == name)
return;
m_overrides[group] = name;
recheckOverridesResendIfChanged();
}
void CShapeOverrideController::unsetOverride(eCursorShapeOverrideGroup group) {
if (m_overrides[group].empty())
return;
m_overrides[group] = "";
recheckOverridesResendIfChanged();
}
void CShapeOverrideController::recheckOverridesResendIfChanged() {
for (const auto& s : m_overrides | std::views::reverse) {
if (s.empty())
continue;
if (s == m_overrideShape)
return;
m_overrideShape = s;
m_events.overrideChanged.emit(s);
return;
}
if (m_overrideShape.empty())
return;
m_overrideShape = "";
m_events.overrideChanged.emit("");
}

View file

@ -0,0 +1,49 @@
#pragma once
#include "../../helpers/memory/Memory.hpp"
#include "../../helpers/signal/Signal.hpp"
#include <array>
#include <string>
namespace Cursor {
enum eCursorShapeOverrideGroup : uint8_t {
// unknown group - lowest priority
CURSOR_OVERRIDE_UNKNOWN = 0,
// window edges for resizing from edge
CURSOR_OVERRIDE_WINDOW_EDGE,
// Drag and drop
CURSOR_OVERRIDE_DND,
// special action: Interactive::CDrag, kill, etc.
CURSOR_OVERRIDE_SPECIAL_ACTION,
//
CURSOR_OVERRIDE_END,
};
class CShapeOverrideController {
public:
CShapeOverrideController() = default;
~CShapeOverrideController() = default;
CShapeOverrideController(const CShapeOverrideController&) = delete;
CShapeOverrideController(CShapeOverrideController&) = delete;
CShapeOverrideController(CShapeOverrideController&&) = delete;
void setOverride(const std::string& name, eCursorShapeOverrideGroup group);
void unsetOverride(eCursorShapeOverrideGroup group);
struct {
// if string is empty, override was cleared
CSignalT<const std::string&> overrideChanged;
} m_events;
private:
void recheckOverridesResendIfChanged();
std::array<std::string, CURSOR_OVERRIDE_END> m_overrides;
std::string m_overrideShape;
};
inline UP<CShapeOverrideController> overrideController = makeUnique<CShapeOverrideController>();
};

View file

@ -44,6 +44,7 @@
#include "../../helpers/MiscFunctions.hpp" #include "../../helpers/MiscFunctions.hpp"
#include "trackpad/TrackpadGestures.hpp" #include "trackpad/TrackpadGestures.hpp"
#include "../cursor/CursorShapeOverrideController.hpp"
#include <aquamarine/input/Input.hpp> #include <aquamarine/input/Input.hpp>
@ -65,20 +66,33 @@ CInputManager::CInputManager() {
m_cursorSurfaceInfo.name = event.shapeName; m_cursorSurfaceInfo.name = event.shapeName;
m_cursorSurfaceInfo.hidden = false; m_cursorSurfaceInfo.hidden = false;
m_cursorSurfaceInfo.inUse = true;
g_pHyprRenderer->setCursorFromName(m_cursorSurfaceInfo.name); g_pHyprRenderer->setCursorFromName(m_cursorSurfaceInfo.name);
}); });
m_listeners.newIdleInhibitor = PROTO::idleInhibit->m_events.newIdleInhibitor.listen([this](const auto& data) { this->newIdleInhibitor(data); }); m_listeners.newIdleInhibitor = PROTO::idleInhibit->m_events.newIdleInhibitor.listen([this](const auto& data) { newIdleInhibitor(data); });
m_listeners.newVirtualKeyboard = PROTO::virtualKeyboard->m_events.newKeyboard.listen([this](const auto& keyboard) { m_listeners.newVirtualKeyboard = PROTO::virtualKeyboard->m_events.newKeyboard.listen([this](const auto& keyboard) {
this->newVirtualKeyboard(keyboard); newVirtualKeyboard(keyboard);
updateCapabilities(); updateCapabilities();
}); });
m_listeners.newVirtualMouse = PROTO::virtualPointer->m_events.newPointer.listen([this](const auto& mouse) {
this->newVirtualMouse(mouse); m_listeners.newVirtualMouse = PROTO::virtualPointer->m_events.newPointer.listen([this](const auto& mouse) {
newVirtualMouse(mouse);
updateCapabilities(); updateCapabilities();
}); });
m_listeners.setCursor = g_pSeatManager->m_events.setCursor.listen([this](const auto& event) { this->processMouseRequest(event); });
m_listeners.setCursor = g_pSeatManager->m_events.setCursor.listen([this](const auto& event) { processMouseRequest(event); });
m_listeners.overrideChanged = Cursor::overrideController->m_events.overrideChanged.listen([this](const std::string& shape) {
if (shape.empty()) {
m_cursorImageOverridden = false;
restoreCursorIconToApp();
return;
}
m_cursorImageOverridden = true;
g_pHyprRenderer->setCursorFromName(shape);
});
m_cursorSurfaceInfo.wlSurface = CWLSurface::create(); m_cursorSurfaceInfo.wlSurface = CWLSurface::create();
} }
@ -472,17 +486,7 @@ void CInputManager::mouseMoveUnified(uint32_t time, bool refocus, bool mouse, st
if (!foundSurface) { if (!foundSurface) {
if (!m_emptyFocusCursorSet) { if (!m_emptyFocusCursorSet) {
if (*PRESIZEONBORDER && *PRESIZECURSORICON && m_borderIconDirection != BORDERICON_NONE) { g_pHyprRenderer->setCursorFromName("left_ptr");
m_borderIconDirection = BORDERICON_NONE;
unsetCursorImage();
}
// TODO: maybe wrap?
if (m_clickBehavior == CLICKMODE_KILL)
setCursorImageOverride("crosshair");
else
setCursorImageOverride("left_ptr");
m_emptyFocusCursorSet = true; m_emptyFocusCursorSet = true;
} }
@ -532,7 +536,7 @@ void CInputManager::mouseMoveUnified(uint32_t time, bool refocus, bool mouse, st
if (pFoundWindow && foundSurface == pFoundWindow->m_wlSurface->resource() && !m_cursorImageOverridden) { if (pFoundWindow && foundSurface == pFoundWindow->m_wlSurface->resource() && !m_cursorImageOverridden) {
const auto BOX = pFoundWindow->getWindowMainSurfaceBox(); const auto BOX = pFoundWindow->getWindowMainSurfaceBox();
if (VECNOTINRECT(mouseCoords, BOX.x, BOX.y, BOX.x + BOX.width, BOX.y + BOX.height)) if (VECNOTINRECT(mouseCoords, BOX.x, BOX.y, BOX.x + BOX.width, BOX.y + BOX.height))
setCursorImageOverride("left_ptr"); g_pHyprRenderer->setCursorFromName("left_ptr");
else else
restoreCursorIconToApp(); restoreCursorIconToApp();
} }
@ -540,11 +544,8 @@ void CInputManager::mouseMoveUnified(uint32_t time, bool refocus, bool mouse, st
if (pFoundWindow) { if (pFoundWindow) {
// change cursor icon if hovering over border // change cursor icon if hovering over border
if (*PRESIZEONBORDER && *PRESIZECURSORICON) { if (*PRESIZEONBORDER && *PRESIZECURSORICON) {
if (!pFoundWindow->isFullscreen() && !pFoundWindow->hasPopupAt(mouseCoords)) { if (!pFoundWindow->isFullscreen() && !pFoundWindow->hasPopupAt(mouseCoords))
setCursorIconOnBorder(pFoundWindow); setCursorIconOnBorder(pFoundWindow);
} else if (m_borderIconDirection != BORDERICON_NONE) {
unsetCursorImage();
}
} }
if (FOLLOWMOUSE != 1 && !refocus) { if (FOLLOWMOUSE != 1 && !refocus) {
@ -595,7 +596,7 @@ void CInputManager::mouseMoveUnified(uint32_t time, bool refocus, bool mouse, st
} else { } else {
if (*PRESIZEONBORDER && *PRESIZECURSORICON && m_borderIconDirection != BORDERICON_NONE) { if (*PRESIZEONBORDER && *PRESIZECURSORICON && m_borderIconDirection != BORDERICON_NONE) {
m_borderIconDirection = BORDERICON_NONE; m_borderIconDirection = BORDERICON_NONE;
unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_WINDOW_EDGE);
} }
if (pFoundLayerSurface && (pFoundLayerSurface->m_layerSurface->m_current.interactivity != ZWLR_LAYER_SURFACE_V1_KEYBOARD_INTERACTIVITY_NONE) && FOLLOWMOUSE != 3 && if (pFoundLayerSurface && (pFoundLayerSurface->m_layerSurface->m_current.interactivity != ZWLR_LAYER_SURFACE_V1_KEYBOARD_INTERACTIVITY_NONE) && FOLLOWMOUSE != 3 &&
@ -667,14 +668,10 @@ void CInputManager::processMouseRequest(const CSeatManager::SSetCursorEvent& eve
m_cursorSurfaceInfo.name = ""; m_cursorSurfaceInfo.name = "";
m_cursorSurfaceInfo.inUse = true;
g_pHyprRenderer->setCursorSurface(m_cursorSurfaceInfo.wlSurface, event.hotspot.x, event.hotspot.y); g_pHyprRenderer->setCursorSurface(m_cursorSurfaceInfo.wlSurface, event.hotspot.x, event.hotspot.y);
} }
void CInputManager::restoreCursorIconToApp() { void CInputManager::restoreCursorIconToApp() {
if (m_cursorSurfaceInfo.inUse)
return;
if (m_cursorSurfaceInfo.hidden) { if (m_cursorSurfaceInfo.hidden) {
g_pHyprRenderer->setCursorSurface(nullptr, 0, 0); g_pHyprRenderer->setCursorSurface(nullptr, 0, 0);
return; return;
@ -683,29 +680,12 @@ void CInputManager::restoreCursorIconToApp() {
if (m_cursorSurfaceInfo.name.empty()) { if (m_cursorSurfaceInfo.name.empty()) {
if (m_cursorSurfaceInfo.wlSurface->exists()) if (m_cursorSurfaceInfo.wlSurface->exists())
g_pHyprRenderer->setCursorSurface(m_cursorSurfaceInfo.wlSurface, m_cursorSurfaceInfo.vHotspot.x, m_cursorSurfaceInfo.vHotspot.y); g_pHyprRenderer->setCursorSurface(m_cursorSurfaceInfo.wlSurface, m_cursorSurfaceInfo.vHotspot.x, m_cursorSurfaceInfo.vHotspot.y);
} else { } else
g_pHyprRenderer->setCursorFromName(m_cursorSurfaceInfo.name); g_pHyprRenderer->setCursorFromName(m_cursorSurfaceInfo.name);
}
m_cursorSurfaceInfo.inUse = true;
}
void CInputManager::setCursorImageOverride(const std::string& name) {
if (m_cursorImageOverridden)
return;
m_cursorSurfaceInfo.inUse = false;
g_pHyprRenderer->setCursorFromName(name);
} }
bool CInputManager::cursorImageUnlocked() { bool CInputManager::cursorImageUnlocked() {
if (m_clickBehavior == CLICKMODE_KILL) return !m_cursorImageOverridden;
return false;
if (m_cursorImageOverridden)
return false;
return true;
} }
eClickBehaviorMode CInputManager::getClickMode() { eClickBehaviorMode CInputManager::getClickMode() {
@ -729,7 +709,7 @@ void CInputManager::setClickMode(eClickBehaviorMode mode) {
refocus(); refocus();
// set cursor // set cursor
g_pHyprRenderer->setCursorFromName("crosshair", true); Cursor::overrideController->setOverride("crosshair", Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
break; break;
default: break; default: break;
} }
@ -834,6 +814,7 @@ void CInputManager::processMouseDownKill(const IPointer::SButtonEvent& e) {
// reset click behavior mode // reset click behavior mode
m_clickBehavior = CLICKMODE_DEFAULT; m_clickBehavior = CLICKMODE_DEFAULT;
Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_SPECIAL_ACTION);
} }
void CInputManager::onMouseWheel(IPointer::SAxisEvent e, SP<IPointer> pointer) { void CInputManager::onMouseWheel(IPointer::SAxisEvent e, SP<IPointer> pointer) {
@ -1867,20 +1848,6 @@ void CInputManager::destroySwitch(SSwitchDevice* pDevice) {
m_switches.remove(*pDevice); m_switches.remove(*pDevice);
} }
void CInputManager::setCursorImageUntilUnset(std::string name) {
g_pHyprRenderer->setCursorFromName(name);
m_cursorImageOverridden = true;
m_cursorSurfaceInfo.inUse = false;
}
void CInputManager::unsetCursorImage() {
if (!m_cursorImageOverridden)
return;
m_cursorImageOverridden = false;
restoreCursorIconToApp();
}
std::string CInputManager::getNameForNewDevice(std::string internalName) { std::string CInputManager::getNameForNewDevice(std::string internalName) {
auto proposedNewName = deviceNameToInternalString(internalName); auto proposedNewName = deviceNameToInternalString(internalName);
@ -1908,16 +1875,12 @@ void CInputManager::releaseAllMouseButtons() {
} }
void CInputManager::setCursorIconOnBorder(PHLWINDOW w) { void CInputManager::setCursorIconOnBorder(PHLWINDOW w) {
// do not override cursor icons set by mouse binds // ignore X11 OR windows, they shouldn't be touched
if (g_pInputManager->m_currentlyDraggedWindow.expired()) { if (w->m_isX11 && w->isX11OverrideRedirect()) {
m_borderIconDirection = BORDERICON_NONE; Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_WINDOW_EDGE);
return; return;
} }
// ignore X11 OR windows, they shouldn't be touched
if (w->m_isX11 && w->isX11OverrideRedirect())
return;
static auto PEXTENDBORDERGRAB = CConfigValue<Hyprlang::INT>("general:extend_border_grab_area"); static auto PEXTENDBORDERGRAB = CConfigValue<Hyprlang::INT>("general:extend_border_grab_area");
const int BORDERSIZE = w->getRealBorderSize(); const int BORDERSIZE = w->getRealBorderSize();
const int ROUNDING = w->rounding(); const int ROUNDING = w->rounding();
@ -1998,15 +1961,15 @@ void CInputManager::setCursorIconOnBorder(PHLWINDOW w) {
m_borderIconDirection = direction; m_borderIconDirection = direction;
switch (direction) { switch (direction) {
case BORDERICON_NONE: unsetCursorImage(); break; case BORDERICON_NONE: Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_UP: setCursorImageUntilUnset("top_side"); break; case BORDERICON_UP: Cursor::overrideController->setOverride("top_side", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_DOWN: setCursorImageUntilUnset("bottom_side"); break; case BORDERICON_DOWN: Cursor::overrideController->setOverride("bottom_side", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_LEFT: setCursorImageUntilUnset("left_side"); break; case BORDERICON_LEFT: Cursor::overrideController->setOverride("left_side", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_RIGHT: setCursorImageUntilUnset("right_side"); break; case BORDERICON_RIGHT: Cursor::overrideController->setOverride("right_side", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_UP_LEFT: setCursorImageUntilUnset("top_left_corner"); break; case BORDERICON_UP_LEFT: Cursor::overrideController->setOverride("top_left_corner", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_DOWN_LEFT: setCursorImageUntilUnset("bottom_left_corner"); break; case BORDERICON_DOWN_LEFT: Cursor::overrideController->setOverride("bottom_left_corner", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_UP_RIGHT: setCursorImageUntilUnset("top_right_corner"); break; case BORDERICON_UP_RIGHT: Cursor::overrideController->setOverride("top_right_corner", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
case BORDERICON_DOWN_RIGHT: setCursorImageUntilUnset("bottom_right_corner"); break; case BORDERICON_DOWN_RIGHT: Cursor::overrideController->setOverride("bottom_right_corner", Cursor::CURSOR_OVERRIDE_WINDOW_EDGE); break;
} }
} }

View file

@ -193,11 +193,7 @@ class CInputManager {
uint32_t getModsFromAllKBs(); uint32_t getModsFromAllKBs();
// for virtual keyboards: whether we should respect them as normal ones // for virtual keyboards: whether we should respect them as normal ones
bool shouldIgnoreVirtualKeyboard(SP<IKeyboard>); bool shouldIgnoreVirtualKeyboard(SP<IKeyboard>);
// for special cursors that we choose
void setCursorImageUntilUnset(std::string);
void unsetCursorImage();
std::string getNameForNewDevice(std::string); std::string getNameForNewDevice(std::string);
@ -226,6 +222,7 @@ class CInputManager {
CHyprSignalListener newVirtualKeyboard; CHyprSignalListener newVirtualKeyboard;
CHyprSignalListener newVirtualMouse; CHyprSignalListener newVirtualMouse;
CHyprSignalListener setCursor; CHyprSignalListener setCursor;
CHyprSignalListener overrideChanged;
} m_listeners; } m_listeners;
bool m_cursorImageOverridden = false; bool m_cursorImageOverridden = false;
@ -282,16 +279,12 @@ class CInputManager {
void setBorderCursorIcon(eBorderIconDirection); void setBorderCursorIcon(eBorderIconDirection);
void setCursorIconOnBorder(PHLWINDOW w); void setCursorIconOnBorder(PHLWINDOW w);
// temporary. Obeys setUntilUnset.
void setCursorImageOverride(const std::string& name);
// cursor surface // cursor surface
struct { struct {
bool hidden = false; // null surface = hidden bool hidden = false; // null surface = hidden
SP<CWLSurface> wlSurface; SP<CWLSurface> wlSurface;
Vector2D vHotspot; Vector2D vHotspot;
std::string name; // if not empty, means set by name. std::string name; // if not empty, means set by name.
bool inUse = false;
} m_cursorSurfaceInfo; } m_cursorSurfaceInfo;
void restoreCursorIconToApp(); // no-op if restored void restoreCursorIconToApp(); // no-op if restored

View file

@ -11,6 +11,7 @@
#include "../../xwayland/Server.hpp" #include "../../xwayland/Server.hpp"
#include "../../managers/input/InputManager.hpp" #include "../../managers/input/InputManager.hpp"
#include "../../managers/HookSystemManager.hpp" #include "../../managers/HookSystemManager.hpp"
#include "../../managers/cursor/CursorShapeOverrideController.hpp"
#include "../../helpers/Monitor.hpp" #include "../../helpers/Monitor.hpp"
#include "../../render/Renderer.hpp" #include "../../render/Renderer.hpp"
#include "../../xwayland/Dnd.hpp" #include "../../xwayland/Dnd.hpp"
@ -553,7 +554,7 @@ void CWLDataDeviceProtocol::initiateDrag(WP<CWLDataSourceResource> currentSource
abortDrag(); abortDrag();
} }
g_pInputManager->setCursorImageUntilUnset("grabbing"); Cursor::overrideController->setOverride("grabbing", Cursor::CURSOR_OVERRIDE_DND);
m_dnd.overriddenCursor = true; m_dnd.overriddenCursor = true;
LOGM(LOG, "initiateDrag: source {:x}, surface: {:x}, origin: {:x}", (uintptr_t)currentSource.get(), (uintptr_t)dragSurface, (uintptr_t)origin); LOGM(LOG, "initiateDrag: source {:x}, surface: {:x}, origin: {:x}", (uintptr_t)currentSource.get(), (uintptr_t)dragSurface, (uintptr_t)origin);
@ -734,7 +735,7 @@ void CWLDataDeviceProtocol::dropDrag() {
if (m_dnd.focusedDevice->getX11()) { if (m_dnd.focusedDevice->getX11()) {
m_dnd.focusedDevice->sendLeave(); m_dnd.focusedDevice->sendLeave();
if (m_dnd.overriddenCursor) if (m_dnd.overriddenCursor)
g_pInputManager->unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_DND);
m_dnd.overriddenCursor = false; m_dnd.overriddenCursor = false;
cleanupDndState(true, true, true); cleanupDndState(true, true, true);
return; return;
@ -743,7 +744,7 @@ void CWLDataDeviceProtocol::dropDrag() {
m_dnd.focusedDevice->sendLeave(); m_dnd.focusedDevice->sendLeave();
if (m_dnd.overriddenCursor) if (m_dnd.overriddenCursor)
g_pInputManager->unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_DND);
m_dnd.overriddenCursor = false; m_dnd.overriddenCursor = false;
cleanupDndState(false, false, false); cleanupDndState(false, false, false);
} }
@ -784,7 +785,7 @@ void CWLDataDeviceProtocol::abortDrag() {
cleanupDndState(false, false, false); cleanupDndState(false, false, false);
if (m_dnd.overriddenCursor) if (m_dnd.overriddenCursor)
g_pInputManager->unsetCursorImage(); Cursor::overrideController->unsetOverride(Cursor::CURSOR_OVERRIDE_DND);
m_dnd.overriddenCursor = false; m_dnd.overriddenCursor = false;
if (!m_dnd.focusedDevice && !m_dnd.currentSource) if (!m_dnd.focusedDevice && !m_dnd.currentSource)