2022-03-17 15:53:45 +01:00
# include "InputManager.hpp"
2022-06-09 12:46:55 +02:00
# include "../../Compositor.hpp"
2023-01-08 16:35:24 +01:00
# include "wlr/types/wlr_switch.h"
2023-03-01 15:06:52 +01:00
# include <ranges>
2022-03-17 15:53:45 +01:00
2022-03-24 15:57:46 +01:00
void CInputManager : : onMouseMoved ( wlr_pointer_motion_event * e ) {
2022-12-16 18:17:31 +01:00
static auto * const PSENS = & g_pConfigManager - > getConfigValuePtr ( " general:sensitivity " ) - > floatValue ;
static auto * const PNOACCEL = & g_pConfigManager - > getConfigValuePtr ( " input:force_no_accel " ) - > intValue ;
2022-09-03 19:35:17 +02:00
static auto * const PSENSTORAW = & g_pConfigManager - > getConfigValuePtr ( " general:apply_sens_to_raw " ) - > intValue ;
2022-03-17 15:53:45 +01:00
2022-12-16 18:17:31 +01:00
const auto DELTA = * PNOACCEL = = 1 ? Vector2D ( e - > unaccel_dx , e - > unaccel_dy ) : Vector2D ( e - > delta_x , e - > delta_y ) ;
2022-06-06 12:08:33 +02:00
2022-09-03 19:35:17 +02:00
if ( * PSENSTORAW = = 1 )
2022-12-16 18:17:31 +01:00
wlr_relative_pointer_manager_v1_send_relative_motion ( g_pCompositor - > m_sWLRRelPointerMgr , g_pCompositor - > m_sSeat . seat , ( uint64_t ) e - > time_msec * 1000 , DELTA . x * * PSENS ,
DELTA . y * * PSENS , e - > unaccel_dx * * PSENS , e - > unaccel_dy * * PSENS ) ;
2022-04-18 13:13:41 +02:00
else
2022-12-16 18:17:31 +01:00
wlr_relative_pointer_manager_v1_send_relative_motion ( g_pCompositor - > m_sWLRRelPointerMgr , g_pCompositor - > m_sSeat . seat , ( uint64_t ) e - > time_msec * 1000 , DELTA . x , DELTA . y ,
e - > unaccel_dx , e - > unaccel_dy ) ;
2022-04-17 21:40:04 +02:00
2022-09-03 19:35:17 +02:00
wlr_cursor_move ( g_pCompositor - > m_sWLRCursor , & e - > pointer - > base , DELTA . x * * PSENS , DELTA . y * * PSENS ) ;
2022-03-18 20:42:49 +01:00
2022-03-18 23:52:36 +01:00
mouseMoveUnified ( e - > time_msec ) ;
2022-06-24 23:27:02 +02:00
m_tmrLastCursorMovement . reset ( ) ;
2023-01-17 11:47:39 +01:00
m_bLastInputTouch = false ;
2022-03-17 16:56:33 +01:00
}
2022-03-24 15:57:46 +01:00
void CInputManager : : onMouseWarp ( wlr_pointer_motion_absolute_event * e ) {
wlr_cursor_warp_absolute ( g_pCompositor - > m_sWLRCursor , & e - > pointer - > base , e - > x , e - > y ) ;
2022-03-18 20:42:49 +01:00
2022-03-18 23:52:36 +01:00
mouseMoveUnified ( e - > time_msec ) ;
2022-06-24 23:27:02 +02:00
m_tmrLastCursorMovement . reset ( ) ;
2023-01-17 11:47:39 +01:00
m_bLastInputTouch = false ;
2022-03-18 23:52:36 +01:00
}
2023-01-31 01:26:15 +01:00
void CInputManager : : simulateMouseMovement ( ) {
timespec now ;
clock_gettime ( CLOCK_MONOTONIC , & now ) ;
m_vLastCursorPosFloored = m_vLastCursorPosFloored - Vector2D ( 1 , 1 ) ; // hack: force the mouseMoveUnified to report without making this a refocus.
mouseMoveUnified ( now . tv_sec * 1000 + now . tv_nsec / 10000000 ) ;
m_tmrLastCursorMovement . reset ( ) ;
}
2022-04-13 20:19:40 +02:00
void CInputManager : : mouseMoveUnified ( uint32_t time , bool refocus ) {
2023-02-18 23:35:31 +01:00
static auto * const PFOLLOWMOUSE = & g_pConfigManager - > getConfigValuePtr ( " input:follow_mouse " ) - > intValue ;
static auto * const PMOUSEDPMS = & g_pConfigManager - > getConfigValuePtr ( " misc:mouse_move_enables_dpms " ) - > intValue ;
static auto * const PFOLLOWONDND = & g_pConfigManager - > getConfigValuePtr ( " misc:always_follow_on_dnd " ) - > intValue ;
static auto * const PHOGFOCUS = & g_pConfigManager - > getConfigValuePtr ( " misc:layers_hog_keyboard_focus " ) - > intValue ;
static auto * const PFLOATBEHAVIOR = & g_pConfigManager - > getConfigValuePtr ( " input:float_switch_override_focus " ) - > intValue ;
static auto * const PMOUSEFOCUSMON = & g_pConfigManager - > getConfigValuePtr ( " misc:mouse_move_focuses_monitor " ) - > intValue ;
static auto * const PRESIZEONBORDER = & g_pConfigManager - > getConfigValuePtr ( " general:resize_on_border " ) - > intValue ;
static auto * const PBORDERSIZE = & g_pConfigManager - > getConfigValuePtr ( " general:border_size " ) - > intValue ;
static auto * const PBORDERGRABEXTEND = & g_pConfigManager - > getConfigValuePtr ( " general:extend_border_grab_area " ) - > intValue ;
static auto * const PRESIZECURSORICON = & g_pConfigManager - > getConfigValuePtr ( " general:hover_icon_on_border " ) - > intValue ;
const auto BORDER_GRAB_AREA = * PRESIZEONBORDER ? * PBORDERSIZE + * PBORDERGRABEXTEND : 0 ;
2022-12-16 18:17:31 +01:00
2023-03-30 01:34:24 +02:00
const auto FOLLOWMOUSE = * PFOLLOWONDND & & m_sDrag . drag ? 1 : * PFOLLOWMOUSE ;
2022-12-16 18:17:31 +01:00
m_pFoundSurfaceToFocus = nullptr ;
m_pFoundLSToFocus = nullptr ;
m_pFoundWindowToFocus = nullptr ;
wlr_surface * foundSurface = nullptr ;
Vector2D surfaceCoords ;
Vector2D surfacePos = Vector2D ( - 1337 , - 1337 ) ;
CWindow * pFoundWindow = nullptr ;
2022-11-04 12:09:34 +01:00
SLayerSurface * pFoundLayerSurface = nullptr ;
2022-10-14 21:46:32 +02:00
2022-10-24 01:14:42 +02:00
if ( ! g_pCompositor - > m_bReadyToProcess | | g_pCompositor - > m_bIsShuttingDown )
2022-04-28 17:57:24 +02:00
return ;
2022-07-31 15:46:42 +02:00
if ( ! g_pCompositor - > m_bDPMSStateON & & * PMOUSEDPMS ) {
// enable dpms
g_pKeybindManager - > dpms ( " on " ) ;
}
2022-12-16 18:17:31 +01:00
Vector2D mouseCoords = getMouseCoordsInternal ( ) ;
2022-07-01 17:59:11 +02:00
const auto MOUSECOORDSFLOORED = mouseCoords . floor ( ) ;
2022-07-03 14:51:32 +02:00
if ( MOUSECOORDSFLOORED = = m_vLastCursorPosFloored & & ! refocus )
2022-07-01 17:59:11 +02:00
return ;
2023-02-28 22:47:00 +01:00
EMIT_HOOK_EVENT ( " mouseMove " , MOUSECOORDSFLOORED ) ;
2022-07-01 17:59:11 +02:00
m_vLastCursorPosFloored = MOUSECOORDSFLOORED ;
2022-05-18 14:57:08 +02:00
const auto PMONITOR = g_pCompositor - > getMonitorFromCursor ( ) ;
// constraints
// All constraints TODO: multiple mice?
2023-02-28 20:02:30 +01:00
if ( g_pCompositor - > m_sSeat . mouse & & g_pCompositor - > m_sSeat . mouse - > currentConstraint & & ! g_pCompositor - > m_sSeat . exclusiveClient & & ! g_pSessionLockManager - > isSessionLocked ( ) ) {
2022-05-18 14:57:08 +02:00
// XWayland windows sometimes issue constraints weirdly.
// TODO: We probably should search their parent. wlr_xwayland_surface->parent
const auto CONSTRAINTWINDOW = g_pCompositor - > getConstraintWindow ( g_pCompositor - > m_sSeat . mouse ) ;
2022-12-16 18:17:31 +01:00
const auto PCONSTRAINT = constraintFromWlr ( g_pCompositor - > m_sSeat . mouse - > currentConstraint ) ;
2022-05-18 14:57:08 +02:00
2022-12-10 15:43:46 +01:00
if ( ! CONSTRAINTWINDOW | | ! PCONSTRAINT ) {
2022-08-09 20:36:21 +02:00
unconstrainMouse ( ) ;
2022-05-18 14:57:08 +02:00
} else {
// Native Wayland apps know how 2 constrain themselves.
// XWayland, we just have to accept them. Might cause issues, but thats XWayland for ya.
2022-12-16 18:17:31 +01:00
const auto CONSTRAINTPOS =
CONSTRAINTWINDOW - > m_bIsX11 ? Vector2D ( CONSTRAINTWINDOW - > m_uSurface . xwayland - > x , CONSTRAINTWINDOW - > m_uSurface . xwayland - > y ) : CONSTRAINTWINDOW - > m_vRealPosition . vec ( ) ;
const auto CONSTRAINTSIZE = CONSTRAINTWINDOW - > m_bIsX11 ? Vector2D ( CONSTRAINTWINDOW - > m_uSurface . xwayland - > width , CONSTRAINTWINDOW - > m_uSurface . xwayland - > height ) :
CONSTRAINTWINDOW - > m_vRealSize . vec ( ) ;
2022-05-18 14:57:08 +02:00
2022-11-20 18:35:07 +01:00
if ( g_pCompositor - > m_sSeat . mouse - > currentConstraint - > type = = WLR_POINTER_CONSTRAINT_V1_LOCKED ) {
// we just snap the cursor to where it should be.
2022-05-18 14:57:08 +02:00
2022-12-16 18:17:31 +01:00
Vector2D hint = { PCONSTRAINT - > positionHint . x , PCONSTRAINT - > positionHint . y } ;
2022-05-18 14:57:08 +02:00
2023-02-25 23:19:51 +01:00
if ( hint ! = Vector2D { - 1 , - 1 } )
wlr_cursor_warp_closest ( g_pCompositor - > m_sWLRCursor , g_pCompositor - > m_sSeat . mouse - > mouse , CONSTRAINTPOS . x + hint . x , CONSTRAINTPOS . y + hint . y ) ;
2022-11-20 19:38:53 +01:00
return ; // don't process anything else, the cursor is locked. The surface should not receive any further events.
// these are usually FPS games. They will use the relative motion.
2022-05-18 14:57:08 +02:00
} else {
2022-11-20 18:35:07 +01:00
// we restrict the cursor to the confined region
2022-12-10 15:43:46 +01:00
if ( ! pixman_region32_contains_point ( & PCONSTRAINT - > constraint - > region , mouseCoords . x - CONSTRAINTPOS . x , mouseCoords . y - CONSTRAINTPOS . y , nullptr ) ) {
2022-11-20 18:35:07 +01:00
if ( g_pCompositor - > m_sSeat . mouse - > constraintActive ) {
wlr_cursor_warp_closest ( g_pCompositor - > m_sWLRCursor , NULL , mouseCoords . x , mouseCoords . y ) ;
mouseCoords = getMouseCoordsInternal ( ) ;
}
} else {
if ( ( ! CONSTRAINTWINDOW - > m_bIsX11 & & PMONITOR & & CONSTRAINTWINDOW - > m_iWorkspaceID = = PMONITOR - > activeWorkspace ) | | ( CONSTRAINTWINDOW - > m_bIsX11 ) ) {
g_pCompositor - > m_sSeat . mouse - > constraintActive = true ;
}
2022-05-18 14:57:08 +02:00
}
}
2022-11-04 19:09:40 +01:00
if ( CONSTRAINTWINDOW - > m_bIsX11 ) {
2023-03-20 16:00:58 +01:00
foundSurface = CONSTRAINTWINDOW - > m_pWLSurface . wlr ( ) ;
2022-12-16 18:17:31 +01:00
surfacePos = CONSTRAINTWINDOW - > m_vRealPosition . vec ( ) ;
2022-11-04 19:09:40 +01:00
} else {
g_pCompositor - > vectorWindowToSurface ( mouseCoords , CONSTRAINTWINDOW , surfaceCoords ) ;
}
pFoundWindow = CONSTRAINTWINDOW ;
2022-05-18 14:57:08 +02:00
}
}
2022-03-31 17:25:23 +02:00
// update stuff
updateDragIcon ( ) ;
2023-04-02 14:30:45 +02:00
if ( ! m_sDrag . drag & & ! m_lCurrentlyHeldButtons . empty ( ) & & g_pCompositor - > m_pLastFocus ) {
if ( m_bLastFocusOnLS ) {
foundSurface = g_pCompositor - > m_pLastFocus ;
pFoundLayerSurface = g_pCompositor - > getLayerSurfaceFromSurface ( foundSurface ) ;
if ( pFoundLayerSurface ) {
surfacePos = g_pCompositor - > getLayerSurfaceFromSurface ( foundSurface ) - > position ;
m_bFocusHeldByButtons = true ;
m_bRefocusHeldByButtons = refocus ;
} else {
// ?
foundSurface = nullptr ;
pFoundLayerSurface = nullptr ;
}
} else if ( g_pCompositor - > m_pLastWindow ) {
foundSurface = g_pCompositor - > m_pLastFocus ;
2023-04-02 14:42:57 +02:00
pFoundWindow = g_pCompositor - > m_pLastWindow ;
2023-04-02 14:30:45 +02:00
if ( ! g_pCompositor - > m_pLastWindow - > m_bIsX11 )
2023-04-02 14:42:57 +02:00
foundSurface = g_pCompositor - > vectorWindowToSurface ( mouseCoords , g_pCompositor - > m_pLastWindow , surfaceCoords ) ;
2023-04-02 14:30:45 +02:00
else
surfacePos = g_pCompositor - > m_pLastWindow - > m_vRealPosition . vec ( ) ;
m_bFocusHeldByButtons = true ;
m_bRefocusHeldByButtons = refocus ;
}
}
2022-04-13 20:19:40 +02:00
g_pLayoutManager - > getCurrentLayout ( ) - > onMouseMove ( getMouseCoordsInternal ( ) ) ;
2022-04-01 23:31:12 +02:00
2023-02-12 11:38:37 +01:00
if ( PMONITOR & & PMONITOR ! = g_pCompositor - > m_pLastMonitor & & ( * PMOUSEFOCUSMON | | refocus ) ) {
2022-11-19 17:41:36 +01:00
g_pCompositor - > setActiveMonitor ( PMONITOR ) ;
2022-03-31 17:25:23 +02:00
2022-04-11 19:51:37 +02:00
// set active workspace and deactivate all other in wlr
2022-05-25 10:25:36 +02:00
const auto ACTIVEWORKSPACE = g_pCompositor - > getWorkspaceByID ( PMONITOR - > activeWorkspace ) ;
g_pCompositor - > deactivateAllWLRWorkspaces ( ACTIVEWORKSPACE - > m_pWlrHandle ) ;
ACTIVEWORKSPACE - > setActive ( true ) ;
2022-04-11 19:51:37 +02:00
}
2023-02-03 12:58:55 +01:00
if ( g_pSessionLockManager - > isSessionLocked ( ) ) {
const auto PSLS = PMONITOR ? g_pSessionLockManager - > getSessionLockSurfaceForMonitor ( PMONITOR - > ID ) : nullptr ;
if ( ! PSLS )
return ;
foundSurface = PSLS - > pWlrLockSurface - > surface ;
surfacePos = PMONITOR - > vecPosition ;
}
2022-04-25 21:49:45 +02:00
// overlay is above fullscreen
if ( ! foundSurface )
2023-01-22 17:03:25 +01:00
foundSurface = g_pCompositor - > vectorToLayerSurface ( mouseCoords , & PMONITOR - > m_aLayerSurfaceLayers [ ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY ] , & surfaceCoords , & pFoundLayerSurface ) ;
2022-04-25 21:49:45 +02:00
2022-09-19 11:23:13 +02:00
if ( ! foundSurface )
2023-01-22 17:03:25 +01:00
foundSurface = g_pCompositor - > vectorToLayerSurface ( mouseCoords , & PMONITOR - > m_aLayerSurfaceLayers [ ZWLR_LAYER_SHELL_V1_LAYER_TOP ] , & surfaceCoords , & pFoundLayerSurface ) ;
2022-09-19 11:23:13 +02:00
2022-04-25 21:49:45 +02:00
// then, we check if the workspace doesnt have a fullscreen window
2022-03-21 19:18:33 +01:00
const auto PWORKSPACE = g_pCompositor - > getWorkspaceByID ( PMONITOR - > activeWorkspace ) ;
2022-05-31 14:01:00 +02:00
if ( PWORKSPACE - > m_bHasFullscreenWindow & & ! foundSurface & & PWORKSPACE - > m_efFullscreenMode = = FULLSCREEN_FULL ) {
2022-04-11 19:51:37 +02:00
pFoundWindow = g_pCompositor - > getFullscreenWindowOnWorkspace ( PWORKSPACE - > m_iID ) ;
2022-09-05 00:59:13 +02:00
if ( ! pFoundWindow ) {
// what the fuck, somehow happens occasionally??
PWORKSPACE - > m_bHasFullscreenWindow = false ;
return ;
}
2022-05-23 22:07:29 +02:00
// only check floating because tiled cant be over fullscreen
2023-03-01 15:06:52 +01:00
for ( auto & w : g_pCompositor - > m_vWindows | std : : views : : reverse ) {
wlr_box box = { w - > m_vRealPosition . vec ( ) . x - BORDER_GRAB_AREA , w - > m_vRealPosition . vec ( ) . y - BORDER_GRAB_AREA , w - > m_vRealSize . vec ( ) . x + 2 * BORDER_GRAB_AREA ,
w - > m_vRealSize . vec ( ) . y + 2 * BORDER_GRAB_AREA } ;
if ( ( ( w - > m_bIsFloating & & w - > m_bIsMapped & & ( w - > m_bCreatedOverFullscreen | | w - > m_bPinned ) ) | |
( g_pCompositor - > isWorkspaceSpecial ( w - > m_iWorkspaceID ) & & PMONITOR - > specialWorkspaceID ) ) & &
wlr_box_contains_point ( & box , mouseCoords . x , mouseCoords . y ) & & g_pCompositor - > isWorkspaceVisible ( w - > m_iWorkspaceID ) & & ! w - > isHidden ( ) ) {
pFoundWindow = w . get ( ) ;
2022-05-23 22:07:29 +02:00
break ;
2022-04-02 18:57:09 +02:00
}
}
2022-10-30 13:28:37 +01:00
if ( ! pFoundWindow - > m_bIsX11 ) {
foundSurface = g_pCompositor - > vectorWindowToSurface ( mouseCoords , pFoundWindow , surfaceCoords ) ;
2022-12-16 18:17:31 +01:00
surfacePos = Vector2D ( - 1337 , - 1337 ) ;
2022-10-30 13:28:37 +01:00
} else {
2023-03-20 16:00:58 +01:00
foundSurface = pFoundWindow - > m_pWLSurface . wlr ( ) ;
2022-12-16 18:17:31 +01:00
surfacePos = pFoundWindow - > m_vRealPosition . vec ( ) ;
2022-10-30 13:28:37 +01:00
}
2022-03-21 19:18:33 +01:00
}
2022-03-20 14:36:55 +01:00
// then windows
2022-04-24 11:41:52 +02:00
if ( ! foundSurface ) {
2022-07-06 14:58:46 +02:00
if ( PWORKSPACE - > m_bHasFullscreenWindow & & PWORKSPACE - > m_efFullscreenMode = = FULLSCREEN_MAXIMIZED ) {
2022-11-27 23:42:22 +01:00
if ( PMONITOR - > specialWorkspaceID ) {
2022-07-06 14:58:46 +02:00
pFoundWindow = g_pCompositor - > vectorToWindowIdeal ( mouseCoords ) ;
2022-11-27 23:42:22 +01:00
if ( pFoundWindow & & ! g_pCompositor - > isWorkspaceSpecial ( pFoundWindow - > m_iWorkspaceID ) ) {
2022-07-06 14:58:46 +02:00
pFoundWindow = g_pCompositor - > getFullscreenWindowOnWorkspace ( PWORKSPACE - > m_iID ) ;
}
} else {
2022-07-13 18:33:36 +02:00
pFoundWindow = g_pCompositor - > vectorToWindowIdeal ( mouseCoords ) ;
if ( ! ( pFoundWindow & & pFoundWindow - > m_bIsFloating & & pFoundWindow - > m_bCreatedOverFullscreen ) )
pFoundWindow = g_pCompositor - > getFullscreenWindowOnWorkspace ( PWORKSPACE - > m_iID ) ;
2022-07-06 14:58:46 +02:00
}
2022-08-19 17:25:07 +02:00
} else {
2022-06-29 14:44:24 +02:00
pFoundWindow = g_pCompositor - > vectorToWindowIdeal ( mouseCoords ) ;
2022-08-19 23:22:53 +02:00
// TODO: this causes crashes, sometimes. ???
2022-12-16 18:17:31 +01:00
// if (refocus && !pFoundWindow) {
// pFoundWindow = g_pCompositor->getFirstWindowOnWorkspace(PMONITOR->activeWorkspace);
// }
2022-08-19 17:25:07 +02:00
}
2022-04-24 11:41:52 +02:00
if ( pFoundWindow ) {
if ( ! pFoundWindow - > m_bIsX11 ) {
foundSurface = g_pCompositor - > vectorWindowToSurface ( mouseCoords , pFoundWindow , surfaceCoords ) ;
} else {
2023-03-20 16:00:58 +01:00
foundSurface = pFoundWindow - > m_pWLSurface . wlr ( ) ;
2022-12-16 18:17:31 +01:00
surfacePos = pFoundWindow - > m_vRealPosition . vec ( ) ;
2022-04-24 11:41:52 +02:00
}
2022-04-02 13:02:16 +02:00
}
2022-03-20 14:36:55 +01:00
}
2022-04-02 18:57:09 +02:00
2022-03-20 14:36:55 +01:00
// then surfaces below
if ( ! foundSurface )
2023-01-22 17:03:25 +01:00
foundSurface = g_pCompositor - > vectorToLayerSurface ( mouseCoords , & PMONITOR - > m_aLayerSurfaceLayers [ ZWLR_LAYER_SHELL_V1_LAYER_BOTTOM ] , & surfaceCoords , & pFoundLayerSurface ) ;
2022-03-20 14:36:55 +01:00
if ( ! foundSurface )
2023-01-22 17:03:25 +01:00
foundSurface =
g_pCompositor - > vectorToLayerSurface ( mouseCoords , & PMONITOR - > m_aLayerSurfaceLayers [ ZWLR_LAYER_SHELL_V1_LAYER_BACKGROUND ] , & surfaceCoords , & pFoundLayerSurface ) ;
2022-03-20 14:36:55 +01:00
2022-08-28 22:45:05 +02:00
g_pCompositor - > scheduleFrameForMonitor ( g_pCompositor - > m_pLastMonitor ) ;
2022-03-20 14:36:55 +01:00
if ( ! foundSurface ) {
2022-08-28 22:45:05 +02:00
if ( ! m_bEmptyFocusCursorSet ) {
2023-02-22 00:57:38 +01:00
m_eBorderIconDirection = BORDERICON_NONE ;
2023-01-26 11:39:06 +01:00
if ( g_pHyprRenderer - > m_bHasARenderedCursor ) {
// TODO: maybe wrap?
if ( m_ecbClickBehavior = = CLICKMODE_KILL )
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " crosshair " , g_pCompositor - > m_sWLRCursor ) ;
else
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " left_ptr " , g_pCompositor - > m_sWLRCursor ) ;
}
2022-12-16 18:17:31 +01:00
2022-08-28 22:45:05 +02:00
m_bEmptyFocusCursorSet = true ;
}
2022-03-18 23:52:36 +01:00
2022-03-22 18:29:13 +01:00
wlr_seat_pointer_clear_focus ( g_pCompositor - > m_sSeat . seat ) ;
2022-03-18 23:52:36 +01:00
2022-07-07 21:47:59 +02:00
if ( refocus ) { // if we are forcing a refocus, and we don't find a surface, clear the kb focus too!
2022-08-08 20:21:11 +02:00
g_pCompositor - > focusWindow ( nullptr ) ;
2022-07-07 21:47:59 +02:00
}
2022-03-18 23:52:36 +01:00
return ;
}
2022-08-28 22:45:05 +02:00
m_bEmptyFocusCursorSet = false ;
2022-03-18 23:52:36 +01:00
if ( time )
2022-03-22 18:29:13 +01:00
wlr_idle_notify_activity ( g_pCompositor - > m_sWLRIdle , g_pCompositor - > m_sSeat . seat ) ;
2022-03-18 23:52:36 +01:00
2022-04-24 17:42:59 +02:00
Vector2D surfaceLocal = surfacePos = = Vector2D ( - 1337 , - 1337 ) ? surfaceCoords : mouseCoords - surfacePos ;
2022-03-18 23:52:36 +01:00
2022-06-22 15:45:56 +02:00
if ( pFoundWindow & & ! pFoundWindow - > m_bIsX11 & & surfacePos ! = Vector2D ( - 1337 , - 1337 ) ) {
// calc for oversized windows... fucking bullshit.
wlr_box geom ;
wlr_xdg_surface_get_geometry ( pFoundWindow - > m_uSurface . xdg , & geom ) ;
surfaceLocal = mouseCoords - surfacePos + Vector2D ( geom . x , geom . y ) ;
}
2022-08-10 17:46:01 +02:00
bool allowKeyboardRefocus = true ;
if ( * PHOGFOCUS & & ! refocus & & g_pCompositor - > m_pLastFocus ) {
const auto PLS = g_pCompositor - > getLayerSurfaceFromSurface ( g_pCompositor - > m_pLastFocus ) ;
if ( PLS & & PLS - > layerSurface - > current . keyboard_interactive ) {
allowKeyboardRefocus = false ;
}
}
2022-10-14 21:46:32 +02:00
// set the values for use
if ( refocus ) {
2022-12-16 18:17:31 +01:00
m_pFoundLSToFocus = pFoundLayerSurface ;
m_pFoundWindowToFocus = pFoundWindow ;
2022-10-14 21:46:32 +02:00
m_pFoundSurfaceToFocus = foundSurface ;
}
2022-04-13 20:19:40 +02:00
if ( pFoundWindow ) {
2023-02-22 00:57:38 +01:00
// change cursor icon if hovering over border
2023-02-18 23:35:31 +01:00
if ( * PRESIZEONBORDER & & * PRESIZECURSORICON & & ! pFoundWindow - > m_bIsFullscreen & & ! pFoundWindow - > hasPopupAt ( mouseCoords ) ) {
setCursorIconOnBorder ( pFoundWindow ) ;
}
2023-03-30 00:44:25 +02:00
// if we're on an input deco, reset cursor. Don't on overridden
// if (!m_bCursorImageOverridden) {
2023-03-01 23:12:26 +01:00
// if (!VECINRECT(m_vLastCursorPosFloored, pFoundWindow->m_vRealPosition.vec().x, pFoundWindow->m_vRealPosition.vec().y,
// pFoundWindow->m_vRealPosition.vec().x + pFoundWindow->m_vRealSize.vec().x, pFoundWindow->m_vRealPosition.vec().y + pFoundWindow->m_vRealSize.vec().y)) {
// wlr_xcursor_manager_set_cursor_image(g_pCompositor->m_sWLRXCursorMgr, "left_ptr", g_pCompositor->m_sWLRCursor);
// cursorSurfaceInfo.bUsed = false;
// } else if (!cursorSurfaceInfo.bUsed) {
// cursorSurfaceInfo.bUsed = true;
// wlr_cursor_set_surface(g_pCompositor->m_sWLRCursor, cursorSurfaceInfo.pSurface, cursorSurfaceInfo.vHotspot.x, cursorSurfaceInfo.vHotspot.y);
// }
// }
2023-03-01 00:06:46 +01:00
2023-03-30 01:34:24 +02:00
if ( FOLLOWMOUSE ! = 1 & & ! refocus ) {
2022-12-16 18:17:31 +01:00
if ( pFoundWindow ! = g_pCompositor - > m_pLastWindow & & g_pCompositor - > m_pLastWindow & &
( ( pFoundWindow - > m_bIsFloating & & * PFLOATBEHAVIOR = = 2 ) | | ( g_pCompositor - > m_pLastWindow - > m_bIsFloating ! = pFoundWindow - > m_bIsFloating & & * PFLOATBEHAVIOR ! = 0 ) ) ) {
2022-04-13 20:19:40 +02:00
// enter if change floating style
2023-03-30 01:34:24 +02:00
if ( FOLLOWMOUSE ! = 3 & & allowKeyboardRefocus )
2022-08-01 18:42:11 +02:00
g_pCompositor - > focusWindow ( pFoundWindow , foundSurface ) ;
2022-04-13 20:19:40 +02:00
wlr_seat_pointer_notify_enter ( g_pCompositor - > m_sSeat . seat , foundSurface , surfaceLocal . x , surfaceLocal . y ) ;
2023-03-30 01:34:24 +02:00
} else if ( FOLLOWMOUSE = = 2 | | FOLLOWMOUSE = = 3 ) {
2022-06-12 02:14:22 +02:00
wlr_seat_pointer_notify_enter ( g_pCompositor - > m_sSeat . seat , foundSurface , surfaceLocal . x , surfaceLocal . y ) ;
}
2022-07-19 14:05:12 +02:00
2022-09-04 18:42:11 +02:00
if ( pFoundWindow = = g_pCompositor - > m_pLastWindow ) {
if ( foundSurface ! = g_pCompositor - > m_pLastFocus | | m_bLastFocusOnLS ) {
// ^^^ changed the subsurface ^^^ came back from a LS
wlr_seat_pointer_notify_enter ( g_pCompositor - > m_sSeat . seat , foundSurface , surfaceLocal . x , surfaceLocal . y ) ;
}
2022-07-19 14:05:12 +02:00
}
2023-03-30 01:34:24 +02:00
if ( FOLLOWMOUSE ! = 0 | | pFoundWindow = = g_pCompositor - > m_pLastWindow )
2022-09-18 13:35:05 +02:00
wlr_seat_pointer_notify_motion ( g_pCompositor - > m_sSeat . seat , time , surfaceLocal . x , surfaceLocal . y ) ;
2022-09-25 20:07:48 +02:00
2022-09-04 18:42:11 +02:00
m_bLastFocusOnLS = false ;
2022-12-16 18:17:31 +01:00
return ; // don't enter any new surfaces
2022-04-13 20:19:40 +02:00
} else {
2023-03-30 01:34:24 +02:00
if ( ( FOLLOWMOUSE ! = 3 & & allowKeyboardRefocus ) | | refocus )
2022-08-01 18:42:11 +02:00
g_pCompositor - > focusWindow ( pFoundWindow , foundSurface ) ;
2022-04-13 20:19:40 +02:00
}
2022-09-04 18:42:11 +02:00
m_bLastFocusOnLS = false ;
2022-08-10 17:46:01 +02:00
} else {
2023-02-22 00:57:38 +01:00
if ( * PRESIZEONBORDER & & * PRESIZECURSORICON & & m_eBorderIconDirection ! = BORDERICON_NONE ) {
m_eBorderIconDirection = BORDERICON_NONE ;
unsetCursorImage ( ) ;
}
2023-04-09 20:53:31 +02:00
if ( pFoundLayerSurface & &
( pFoundLayerSurface - > layerSurface - > current . keyboard_interactive | | ( pFoundLayerSurface - > layer > = ZWLR_LAYER_SHELL_V1_LAYER_TOP & & ! g_pCompositor - > m_pLastWindow ) ) & &
2023-03-30 01:34:24 +02:00
FOLLOWMOUSE ! = 3 & & allowKeyboardRefocus ) {
2022-08-10 17:46:01 +02:00
g_pCompositor - > focusSurface ( foundSurface ) ;
2022-08-19 17:29:16 +02:00
}
2022-12-10 16:15:40 +01:00
if ( pFoundLayerSurface )
m_bLastFocusOnLS = true ;
2022-08-10 17:46:01 +02:00
}
2022-03-22 21:59:14 +01:00
2022-04-05 18:29:58 +02:00
wlr_seat_pointer_notify_enter ( g_pCompositor - > m_sSeat . seat , foundSurface , surfaceLocal . x , surfaceLocal . y ) ;
wlr_seat_pointer_notify_motion ( g_pCompositor - > m_sSeat . seat , time , surfaceLocal . x , surfaceLocal . y ) ;
2022-03-18 20:42:49 +01:00
}
2022-03-24 15:57:46 +01:00
void CInputManager : : onMouseButton ( wlr_pointer_button_event * e ) {
2022-03-22 18:29:13 +01:00
wlr_idle_notify_activity ( g_pCompositor - > m_sWLRIdle , g_pCompositor - > m_sSeat . seat ) ;
2022-03-18 20:42:49 +01:00
2023-02-28 22:45:57 +01:00
EMIT_HOOK_EVENT ( " mouseButton " , e ) ;
2022-06-24 23:27:02 +02:00
m_tmrLastCursorMovement . reset ( ) ;
2023-01-11 18:38:54 +01:00
if ( e - > state = = WLR_BUTTON_PRESSED ) {
m_lCurrentlyHeldButtons . push_back ( e - > button ) ;
} else {
if ( std : : find_if ( m_lCurrentlyHeldButtons . begin ( ) , m_lCurrentlyHeldButtons . end ( ) , [ & ] ( const auto & other ) { return other = = e - > button ; } ) = = m_lCurrentlyHeldButtons . end ( ) )
return ;
std : : erase_if ( m_lCurrentlyHeldButtons , [ & ] ( const auto & other ) { return other = = e - > button ; } ) ;
}
2022-06-27 13:42:20 +02:00
switch ( m_ecbClickBehavior ) {
2022-12-16 18:17:31 +01:00
case CLICKMODE_DEFAULT : processMouseDownNormal ( e ) ; break ;
case CLICKMODE_KILL : processMouseDownKill ( e ) ; break ;
default : break ;
2022-06-27 13:42:20 +02:00
}
2023-04-02 14:30:45 +02:00
if ( m_bFocusHeldByButtons & & m_lCurrentlyHeldButtons . empty ( ) & & e - > state = = WLR_BUTTON_RELEASED ) {
if ( m_bRefocusHeldByButtons )
refocus ( ) ;
else
simulateMouseMovement ( ) ;
m_bFocusHeldByButtons = false ;
m_bRefocusHeldByButtons = false ;
}
2022-06-27 13:42:20 +02:00
}
void CInputManager : : processMouseRequest ( wlr_seat_pointer_request_set_cursor_event * e ) {
if ( ! g_pHyprRenderer - > shouldRenderCursor ( ) )
return ;
if ( ! e - > surface ) {
g_pHyprRenderer - > m_bWindowRequestedCursorHide = true ;
} else {
g_pHyprRenderer - > m_bWindowRequestedCursorHide = false ;
}
2023-03-30 00:44:25 +02:00
if ( m_bCursorImageOverridden ) {
2022-11-15 11:39:05 +01:00
return ;
}
2022-06-27 13:42:20 +02:00
if ( m_ecbClickBehavior = = CLICKMODE_KILL ) {
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " crosshair " , g_pCompositor - > m_sWLRCursor ) ;
return ;
}
2023-03-01 23:12:26 +01:00
// cursorSurfaceInfo.pSurface = e->surface;
2023-03-01 00:06:46 +01:00
2023-03-01 23:12:26 +01:00
// if (e->surface) {
// hyprListener_CursorSurfaceDestroy.removeCallback();
// hyprListener_CursorSurfaceDestroy.initCallback(
// &e->surface->events.destroy, [&](void* owner, void* data) { cursorSurfaceInfo.pSurface = nullptr; }, this, "InputManager");
// cursorSurfaceInfo.vHotspot = {e->hotspot_x, e->hotspot_y};
// }
2023-03-01 00:06:46 +01:00
2022-06-27 13:42:20 +02:00
if ( e - > seat_client = = g_pCompositor - > m_sSeat . seat - > pointer_state . focused_client )
wlr_cursor_set_surface ( g_pCompositor - > m_sWLRCursor , e - > surface , e - > hotspot_x , e - > hotspot_y ) ;
}
eClickBehaviorMode CInputManager : : getClickMode ( ) {
return m_ecbClickBehavior ;
}
void CInputManager : : setClickMode ( eClickBehaviorMode mode ) {
switch ( mode ) {
case CLICKMODE_DEFAULT :
Debug : : log ( LOG , " SetClickMode: DEFAULT " ) ;
m_ecbClickBehavior = CLICKMODE_DEFAULT ;
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " left_ptr " , g_pCompositor - > m_sWLRCursor ) ;
break ;
case CLICKMODE_KILL :
Debug : : log ( LOG , " SetClickMode: KILL " ) ;
m_ecbClickBehavior = CLICKMODE_KILL ;
// remove constraints
2022-08-09 20:36:21 +02:00
g_pInputManager - > unconstrainMouse ( ) ;
2022-06-27 13:42:20 +02:00
refocus ( ) ;
// set cursor
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " crosshair " , g_pCompositor - > m_sWLRCursor ) ;
break ;
2022-12-16 18:17:31 +01:00
default : break ;
2022-09-25 20:07:48 +02:00
}
2022-06-27 13:42:20 +02:00
}
void CInputManager : : processMouseDownNormal ( wlr_pointer_button_event * e ) {
2022-12-16 18:17:31 +01:00
2022-07-26 14:50:21 +02:00
// notify the keybind manager
2023-02-18 23:35:31 +01:00
static auto * const PPASSMOUSE = & g_pConfigManager - > getConfigValuePtr ( " binds:pass_mouse_when_bound " ) - > intValue ;
const auto PASS = g_pKeybindManager - > onMouseEvent ( e ) ;
static auto * const PFOLLOWMOUSE = & g_pConfigManager - > getConfigValuePtr ( " input:follow_mouse " ) - > intValue ;
static auto * const PRESIZEONBORDER = & g_pConfigManager - > getConfigValuePtr ( " general:resize_on_border " ) - > intValue ;
2022-07-26 14:50:21 +02:00
if ( ! PASS & & ! * PPASSMOUSE )
return ;
2023-02-18 23:35:31 +01:00
// clicking on border triggers resize
// TODO detect click on LS properly
if ( * PRESIZEONBORDER & & ! m_bLastFocusOnLS ) {
const auto mouseCoords = g_pInputManager - > getMouseCoordsInternal ( ) ;
const auto w = g_pCompositor - > vectorToWindowIdeal ( mouseCoords ) ;
if ( w & & ! w - > m_bIsFullscreen ) {
const wlr_box real = { w - > m_vRealPosition . vec ( ) . x , w - > m_vRealPosition . vec ( ) . y , w - > m_vRealSize . vec ( ) . x , w - > m_vRealSize . vec ( ) . y } ;
if ( ( ! wlr_box_contains_point ( & real , mouseCoords . x , mouseCoords . y ) | | w - > isInCurvedCorner ( mouseCoords . x , mouseCoords . y ) ) & & ! w - > hasPopupAt ( mouseCoords ) ) {
g_pKeybindManager - > resizeWithBorder ( e ) ;
return ;
}
}
}
2022-03-18 20:42:49 +01:00
switch ( e - > state ) {
case WLR_BUTTON_PRESSED :
2022-11-08 21:28:41 +01:00
if ( * PFOLLOWMOUSE = = 3 ) // don't refocus on full loose
break ;
2023-04-04 00:09:44 +02:00
if ( ! g_pCompositor - > m_sSeat . mouse | | ! g_pCompositor - > m_sSeat . mouse - > currentConstraint ) {
// a bit hacky
// if we only pressed one button, allow us to refocus. m_lCurrentlyHeldButtons.size() > 0 will stick the focus
if ( m_lCurrentlyHeldButtons . size ( ) = = 1 ) {
const auto COPY = m_lCurrentlyHeldButtons ;
m_lCurrentlyHeldButtons . clear ( ) ;
refocus ( ) ;
m_lCurrentlyHeldButtons = COPY ;
} else
refocus ( ) ;
}
2022-04-02 13:02:16 +02:00
2022-04-04 16:28:43 +02:00
// if clicked on a floating window make it top
2022-10-14 21:46:32 +02:00
if ( g_pCompositor - > m_pLastWindow & & g_pCompositor - > m_pLastWindow - > m_bIsFloating )
2022-04-04 16:28:43 +02:00
g_pCompositor - > moveWindowToTop ( g_pCompositor - > m_pLastWindow ) ;
2022-03-18 20:42:49 +01:00
break ;
2022-12-16 18:17:31 +01:00
case WLR_BUTTON_RELEASED : break ;
2022-03-18 20:42:49 +01:00
}
// notify app if we didnt handle it
2022-03-22 21:59:14 +01:00
if ( g_pCompositor - > doesSeatAcceptInput ( g_pCompositor - > m_pLastFocus ) ) {
2022-03-22 18:29:13 +01:00
wlr_seat_pointer_notify_button ( g_pCompositor - > m_sSeat . seat , e - > time_msec , e - > button , e - > state ) ;
2022-03-22 21:59:14 +01:00
}
2022-06-27 13:42:20 +02:00
}
void CInputManager : : processMouseDownKill ( wlr_pointer_button_event * e ) {
switch ( e - > state ) {
case WLR_BUTTON_PRESSED : {
2022-12-29 11:52:46 +01:00
const auto PWINDOW = g_pCompositor - > vectorToWindowIdeal ( getMouseCoordsInternal ( ) ) ;
2022-06-27 13:42:20 +02:00
2022-10-14 21:46:32 +02:00
if ( ! PWINDOW ) {
2022-06-27 13:42:20 +02:00
Debug : : log ( ERR , " Cannot kill invalid window! " ) ;
break ;
}
// kill the mf
kill ( PWINDOW - > getPID ( ) , SIGKILL ) ;
break ;
}
2022-12-16 18:17:31 +01:00
case WLR_BUTTON_RELEASED : break ;
default : break ;
2022-06-27 13:42:20 +02:00
}
// reset click behavior mode
m_ecbClickBehavior = CLICKMODE_DEFAULT ;
2022-03-17 16:56:33 +01:00
}
2022-06-20 21:47:28 +02:00
void CInputManager : : onMouseWheel ( wlr_pointer_axis_event * e ) {
2022-12-16 18:17:31 +01:00
static auto * const PSCROLLFACTOR = & g_pConfigManager - > getConfigValuePtr ( " input:touchpad:scroll_factor " ) - > floatValue ;
2022-10-06 21:20:10 +02:00
2022-12-16 18:17:31 +01:00
auto factor = ( * PSCROLLFACTOR < = 0.f | | e - > source ! = WLR_AXIS_SOURCE_FINGER ? 1.f : * PSCROLLFACTOR ) ;
2022-10-06 21:20:10 +02:00
2022-12-16 18:17:31 +01:00
bool passEvent = g_pKeybindManager - > onAxisEvent ( e ) ;
2022-06-20 21:47:28 +02:00
2022-07-21 18:18:03 +02:00
wlr_idle_notify_activity ( g_pCompositor - > m_sWLRIdle , g_pCompositor - > m_sSeat . seat ) ;
2022-06-20 21:47:28 +02:00
2022-07-21 18:18:03 +02:00
if ( passEvent ) {
2022-12-16 18:17:31 +01:00
wlr_seat_pointer_notify_axis ( g_pCompositor - > m_sSeat . seat , e - > time_msec , e - > orientation , factor * e - > delta , std : : round ( factor * e - > delta_discrete ) , e - > source ) ;
2022-07-15 20:54:05 +02:00
}
2022-06-20 21:47:28 +02:00
}
2022-03-17 19:03:15 +01:00
Vector2D CInputManager : : getMouseCoordsInternal ( ) {
2022-03-19 20:30:21 +01:00
return Vector2D ( g_pCompositor - > m_sWLRCursor - > x , g_pCompositor - > m_sWLRCursor - > y ) ;
2022-03-17 20:55:04 +01:00
}
void CInputManager : : newKeyboard ( wlr_input_device * keyboard ) {
2022-06-28 15:40:14 +02:00
const auto PNEWKEYBOARD = & m_lKeyboards . emplace_back ( ) ;
2022-03-17 20:55:04 +01:00
PNEWKEYBOARD - > keyboard = keyboard ;
2022-06-30 21:26:00 +02:00
try {
2022-12-03 21:36:52 +01:00
PNEWKEYBOARD - > name = getNameForNewDevice ( keyboard - > name ) ;
2022-06-30 21:26:00 +02:00
} catch ( std : : exception & e ) {
2022-12-16 18:17:31 +01:00
Debug : : log ( ERR , " Keyboard had no name??? " ) ; // logic error
2022-06-30 21:26:00 +02:00
}
2022-03-17 20:55:04 +01:00
2022-07-11 23:09:35 +02:00
PNEWKEYBOARD - > hyprListener_keyboardMod . initCallback ( & wlr_keyboard_from_input_device ( keyboard ) - > events . modifiers , & Events : : listener_keyboardMod , PNEWKEYBOARD , " Keyboard " ) ;
PNEWKEYBOARD - > hyprListener_keyboardKey . initCallback ( & wlr_keyboard_from_input_device ( keyboard ) - > events . key , & Events : : listener_keyboardKey , PNEWKEYBOARD , " Keyboard " ) ;
2022-03-28 22:31:39 +02:00
PNEWKEYBOARD - > hyprListener_keyboardDestroy . initCallback ( & keyboard - > events . destroy , & Events : : listener_keyboardDestroy , PNEWKEYBOARD , " Keyboard " ) ;
2022-03-17 20:55:04 +01:00
2022-12-16 18:17:31 +01:00
PNEWKEYBOARD - > hyprListener_keyboardKeymap . initCallback (
& wlr_keyboard_from_input_device ( keyboard ) - > events . keymap ,
[ & ] ( void * owner , void * data ) {
const auto PKEYBOARD = ( SKeyboard * ) owner ;
2023-02-19 21:54:53 +01:00
const auto LAYOUT = getActiveLayoutForKeyboard ( PKEYBOARD ) ;
2022-09-25 20:07:48 +02:00
2023-02-19 21:54:53 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activelayout " , PKEYBOARD - > name + " , " + LAYOUT } , true ) ; // force as this should ALWAYS be sent
EMIT_HOOK_EVENT ( " activeLayout " , ( std : : vector < void * > { PKEYBOARD , ( void * ) & LAYOUT } ) ) ;
2022-12-16 18:17:31 +01:00
} ,
PNEWKEYBOARD , " Keyboard " ) ;
2022-08-18 17:17:33 +02:00
2022-08-18 17:50:32 +02:00
disableAllKeyboards ( false ) ;
2022-09-25 20:07:48 +02:00
2022-06-03 18:59:39 +02:00
m_pActiveKeyboard = PNEWKEYBOARD ;
2022-08-18 17:17:33 +02:00
PNEWKEYBOARD - > active = true ;
2022-06-25 11:50:09 +02:00
applyConfigToKeyboard ( PNEWKEYBOARD ) ;
2022-06-03 19:15:39 +02:00
2022-07-11 23:09:35 +02:00
wlr_seat_set_keyboard ( g_pCompositor - > m_sSeat . seat , wlr_keyboard_from_input_device ( keyboard ) ) ;
2022-03-19 11:27:19 +01:00
Debug : : log ( LOG , " New keyboard created, pointers Hypr: %x and WLR: %x " , PNEWKEYBOARD , keyboard ) ;
2022-03-24 21:05:34 +01:00
}
2022-08-05 16:21:08 +02:00
void CInputManager : : newVirtualKeyboard ( wlr_input_device * keyboard ) {
const auto PNEWKEYBOARD = & m_lKeyboards . emplace_back ( ) ;
2022-12-16 18:17:31 +01:00
PNEWKEYBOARD - > keyboard = keyboard ;
2022-08-05 16:21:08 +02:00
PNEWKEYBOARD - > isVirtual = true ;
try {
2022-12-03 21:36:52 +01:00
PNEWKEYBOARD - > name = getNameForNewDevice ( keyboard - > name ) ;
2022-08-05 16:21:08 +02:00
} catch ( std : : exception & e ) {
2022-12-16 18:17:31 +01:00
Debug : : log ( ERR , " Keyboard had no name??? " ) ; // logic error
2022-08-05 16:21:08 +02:00
}
PNEWKEYBOARD - > hyprListener_keyboardMod . initCallback ( & wlr_keyboard_from_input_device ( keyboard ) - > events . modifiers , & Events : : listener_keyboardMod , PNEWKEYBOARD , " Keyboard " ) ;
PNEWKEYBOARD - > hyprListener_keyboardKey . initCallback ( & wlr_keyboard_from_input_device ( keyboard ) - > events . key , & Events : : listener_keyboardKey , PNEWKEYBOARD , " Keyboard " ) ;
PNEWKEYBOARD - > hyprListener_keyboardDestroy . initCallback ( & keyboard - > events . destroy , & Events : : listener_keyboardDestroy , PNEWKEYBOARD , " Keyboard " ) ;
2022-12-16 18:17:31 +01:00
PNEWKEYBOARD - > hyprListener_keyboardKeymap . initCallback (
& wlr_keyboard_from_input_device ( keyboard ) - > events . keymap ,
[ & ] ( void * owner , void * data ) {
const auto PKEYBOARD = ( SKeyboard * ) owner ;
2023-02-19 21:54:53 +01:00
const auto LAYOUT = getActiveLayoutForKeyboard ( PKEYBOARD ) ;
2022-09-25 20:07:48 +02:00
2023-02-19 21:54:53 +01:00
g_pEventManager - > postEvent ( SHyprIPCEvent { " activelayout " , PKEYBOARD - > name + " , " + LAYOUT } , true ) ; // force as this should ALWAYS be sent
EMIT_HOOK_EVENT ( " activeLayout " , ( std : : vector < void * > { PKEYBOARD , ( void * ) & LAYOUT } ) ) ;
2022-12-16 18:17:31 +01:00
} ,
PNEWKEYBOARD , " Keyboard " ) ;
2022-08-18 17:50:32 +02:00
disableAllKeyboards ( true ) ;
2022-08-05 16:21:08 +02:00
m_pActiveKeyboard = PNEWKEYBOARD ;
2022-08-18 17:17:33 +02:00
PNEWKEYBOARD - > active = true ;
2022-08-05 16:21:08 +02:00
applyConfigToKeyboard ( PNEWKEYBOARD ) ;
wlr_seat_set_keyboard ( g_pCompositor - > m_sSeat . seat , wlr_keyboard_from_input_device ( keyboard ) ) ;
Debug : : log ( LOG , " New virtual keyboard created, pointers Hypr: %x and WLR: %x " , PNEWKEYBOARD , keyboard ) ;
}
2022-03-24 21:05:34 +01:00
void CInputManager : : setKeyboardLayout ( ) {
2022-06-23 15:48:31 +02:00
for ( auto & k : m_lKeyboards )
applyConfigToKeyboard ( & k ) ;
2022-08-21 16:43:18 +02:00
g_pKeybindManager - > updateXKBTranslationState ( ) ;
2022-06-23 15:48:31 +02:00
}
void CInputManager : : applyConfigToKeyboard ( SKeyboard * pKeyboard ) {
2022-12-16 18:17:31 +01:00
auto devname = pKeyboard - > name ;
2022-07-28 21:38:30 +02:00
const auto HASCONFIG = g_pConfigManager - > deviceConfigExists ( devname ) ;
Debug : : log ( LOG , " ApplyConfigToKeyboard for \" %s \" , hasconfig: %i " , pKeyboard - > name . c_str ( ) , ( int ) HASCONFIG ) ;
2022-03-24 21:05:34 +01:00
2022-06-23 15:48:31 +02:00
ASSERT ( pKeyboard ) ;
2022-03-24 21:05:34 +01:00
2022-07-11 23:09:35 +02:00
if ( ! wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) )
2022-07-03 22:54:47 +02:00
return ;
2022-12-16 18:17:31 +01:00
const auto REPEATRATE = HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " repeat_rate " ) : g_pConfigManager - > getInt ( " input:repeat_rate " ) ;
2022-07-28 21:38:30 +02:00
const auto REPEATDELAY = HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " repeat_delay " ) : g_pConfigManager - > getInt ( " input:repeat_delay " ) ;
2022-03-24 21:05:34 +01:00
2022-07-28 21:38:30 +02:00
const auto NUMLOCKON = HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " numlock_by_default " ) : g_pConfigManager - > getInt ( " input:numlock_by_default " ) ;
2022-06-30 21:26:00 +02:00
2022-08-19 20:01:51 +02:00
const auto FILEPATH = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_file " ) : g_pConfigManager - > getString ( " input:kb_file " ) ;
2022-12-16 18:17:31 +01:00
const auto RULES = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_rules " ) : g_pConfigManager - > getString ( " input:kb_rules " ) ;
const auto MODEL = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_model " ) : g_pConfigManager - > getString ( " input:kb_model " ) ;
const auto LAYOUT = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_layout " ) : g_pConfigManager - > getString ( " input:kb_layout " ) ;
const auto VARIANT = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_variant " ) : g_pConfigManager - > getString ( " input:kb_variant " ) ;
const auto OPTIONS = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " kb_options " ) : g_pConfigManager - > getString ( " input:kb_options " ) ;
2022-03-24 21:05:34 +01:00
2022-12-20 03:18:47 +01:00
const auto ENABLED = HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " enabled " ) : true ;
2022-12-16 18:20:51 +01:00
pKeyboard - > enabled = ENABLED ;
2022-06-28 15:40:14 +02:00
try {
2022-12-16 18:17:31 +01:00
if ( NUMLOCKON = = pKeyboard - > numlockOn & & REPEATDELAY = = pKeyboard - > repeatDelay & & REPEATRATE = = pKeyboard - > repeatRate & & RULES ! = " " & &
RULES = = pKeyboard - > currentRules . rules & & MODEL = = pKeyboard - > currentRules . model & & LAYOUT = = pKeyboard - > currentRules . layout & &
VARIANT = = pKeyboard - > currentRules . variant & & OPTIONS = = pKeyboard - > currentRules . options & & FILEPATH = = pKeyboard - > xkbFilePath ) {
2022-06-28 15:40:14 +02:00
Debug : : log ( LOG , " Not applying config to keyboard, it did not change. " ) ;
return ;
}
} catch ( std : : exception & e ) {
// can be libc errors for null std::string
// we can ignore those and just apply
2022-06-25 11:50:09 +02:00
}
2022-07-11 23:09:35 +02:00
wlr_keyboard_set_repeat_info ( wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) , std : : max ( 0 , REPEATRATE ) , std : : max ( 0 , REPEATDELAY ) ) ;
2022-06-30 21:26:00 +02:00
pKeyboard - > repeatDelay = REPEATDELAY ;
2022-12-16 18:17:31 +01:00
pKeyboard - > repeatRate = REPEATRATE ;
pKeyboard - > numlockOn = NUMLOCKON ;
2022-08-19 20:01:51 +02:00
pKeyboard - > xkbFilePath = FILEPATH . c_str ( ) ;
2022-06-30 21:26:00 +02:00
2022-12-16 18:17:31 +01:00
xkb_rule_names rules = { . rules = RULES . c_str ( ) , . model = MODEL . c_str ( ) , . layout = LAYOUT . c_str ( ) , . variant = VARIANT . c_str ( ) , . options = OPTIONS . c_str ( ) } ;
2022-03-24 21:05:34 +01:00
2022-12-16 18:17:31 +01:00
pKeyboard - > currentRules . rules = RULES ;
pKeyboard - > currentRules . model = MODEL ;
2022-07-08 12:27:05 +02:00
pKeyboard - > currentRules . variant = VARIANT ;
pKeyboard - > currentRules . options = OPTIONS ;
2022-12-16 18:17:31 +01:00
pKeyboard - > currentRules . layout = LAYOUT ;
2022-06-25 11:50:09 +02:00
2022-03-24 21:05:34 +01:00
const auto CONTEXT = xkb_context_new ( XKB_CONTEXT_NO_FLAGS ) ;
2022-06-23 18:52:05 +02:00
if ( ! CONTEXT ) {
Debug : : log ( ERR , " applyConfigToKeyboard: CONTEXT null?? " ) ;
2022-03-24 21:05:34 +01:00
return ;
}
2022-12-16 18:17:31 +01:00
Debug : : log ( LOG , " Attempting to create a keymap for layout %s with variant %s (rules: %s, model: %s, options: %s) " , rules . layout , rules . variant , rules . rules , rules . model ,
rules . options ) ;
2022-06-25 11:50:09 +02:00
2022-12-16 18:17:31 +01:00
xkb_keymap * KEYMAP = NULL ;
2022-08-19 20:01:51 +02:00
if ( ! FILEPATH . empty ( ) ) {
2022-12-16 18:17:31 +01:00
auto path = absolutePath ( FILEPATH , g_pConfigManager - > configCurrentPath ) ;
2022-08-19 20:01:51 +02:00
2022-12-16 18:17:31 +01:00
if ( ! std : : filesystem : : exists ( path ) ) {
Debug : : log ( ERR , " input:kb_file= file doesnt exist " ) ;
} else {
KEYMAP = xkb_keymap_new_from_file ( CONTEXT , fopen ( path . c_str ( ) , " r " ) , XKB_KEYMAP_FORMAT_TEXT_V1 , XKB_KEYMAP_COMPILE_NO_FLAGS ) ;
}
2022-09-25 20:07:48 +02:00
}
2022-08-19 20:01:51 +02:00
if ( ! KEYMAP ) {
2022-12-16 18:17:31 +01:00
KEYMAP = xkb_keymap_new_from_names ( CONTEXT , & rules , XKB_KEYMAP_COMPILE_NO_FLAGS ) ;
2022-08-19 20:01:51 +02:00
}
2022-06-02 20:31:47 +02:00
2022-03-24 21:05:34 +01:00
if ( ! KEYMAP ) {
2022-12-16 18:17:31 +01:00
g_pConfigManager - > addParseError ( " Invalid keyboard layout passed. ( rules: " + RULES + " , model: " + MODEL + " , variant: " + VARIANT + " , options: " + OPTIONS +
" , layout: " + LAYOUT + " ) " ) ;
2022-08-09 18:13:13 +02:00
2022-12-16 18:17:31 +01:00
Debug : : log ( ERR , " Keyboard layout %s with variant %s (rules: %s, model: %s, options: %s) couldn't have been loaded. " , rules . layout , rules . variant , rules . rules , rules . model ,
rules . options ) ;
2022-06-25 11:50:09 +02:00
memset ( & rules , 0 , sizeof ( rules ) ) ;
2022-12-16 18:17:31 +01:00
pKeyboard - > currentRules . rules = " " ;
pKeyboard - > currentRules . model = " " ;
2022-07-08 12:27:05 +02:00
pKeyboard - > currentRules . variant = " " ;
pKeyboard - > currentRules . options = " " ;
2022-12-16 18:17:31 +01:00
pKeyboard - > currentRules . layout = " " ;
2022-06-25 11:50:09 +02:00
KEYMAP = xkb_keymap_new_from_names ( CONTEXT , & rules , XKB_KEYMAP_COMPILE_NO_FLAGS ) ;
2022-06-03 18:59:39 +02:00
}
2022-06-02 19:47:11 +02:00
2022-07-11 23:09:35 +02:00
wlr_keyboard_set_keymap ( wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) , KEYMAP ) ;
2022-06-02 19:47:11 +02:00
2022-06-03 18:59:39 +02:00
wlr_keyboard_modifiers wlrMods = { 0 } ;
2022-06-02 19:47:11 +02:00
2022-06-30 21:26:00 +02:00
if ( NUMLOCKON = = 1 ) {
2022-06-03 18:59:39 +02:00
// lock numlock
const auto IDX = xkb_map_mod_get_index ( KEYMAP , XKB_MOD_NAME_NUM ) ;
if ( IDX ! = XKB_MOD_INVALID )
wlrMods . locked | = ( uint32_t ) 1 < < IDX ;
}
if ( wlrMods . locked ! = 0 ) {
2022-07-11 23:09:35 +02:00
wlr_keyboard_notify_modifiers ( wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) , 0 , 0 , wlrMods . locked , 0 ) ;
2022-06-02 19:47:11 +02:00
}
2022-03-24 21:05:34 +01:00
xkb_keymap_unref ( KEYMAP ) ;
xkb_context_unref ( CONTEXT ) ;
2023-02-19 21:54:53 +01:00
const auto LAYOUTSTR = getActiveLayoutForKeyboard ( pKeyboard ) ;
g_pEventManager - > postEvent ( SHyprIPCEvent { " activelayout " , pKeyboard - > name + " , " + LAYOUTSTR } , true ) ; // force as this should ALWAYS be sent
EMIT_HOOK_EVENT ( " activeLayout " , ( std : : vector < void * > { pKeyboard , ( void * ) & LAYOUTSTR } ) ) ;
2022-08-18 17:17:33 +02:00
2022-06-23 15:48:31 +02:00
Debug : : log ( LOG , " Set the keyboard layout to %s and variant to %s for keyboard \" %s \" " , rules . layout , rules . variant , pKeyboard - > keyboard - > name ) ;
2022-09-25 20:07:48 +02:00
}
2022-03-17 20:55:04 +01:00
2022-07-11 20:23:16 +02:00
void CInputManager : : newMouse ( wlr_input_device * mouse , bool virt ) {
2022-04-17 21:40:04 +02:00
m_lMice . emplace_back ( ) ;
const auto PMOUSE = & m_lMice . back ( ) ;
PMOUSE - > mouse = mouse ;
2022-12-16 18:17:31 +01:00
PMOUSE - > virt = virt ;
2022-06-30 21:26:00 +02:00
try {
2022-12-03 21:36:52 +01:00
PMOUSE - > name = getNameForNewDevice ( mouse - > name ) ;
2022-12-16 18:17:31 +01:00
} catch ( std : : exception & e ) {
2022-06-30 21:26:00 +02:00
Debug : : log ( ERR , " Mouse had no name??? " ) ; // logic error
}
2022-03-18 22:53:27 +01:00
if ( wlr_input_device_is_libinput ( mouse ) ) {
const auto LIBINPUTDEV = ( libinput_device * ) wlr_libinput_get_device_handle ( mouse ) ;
2022-12-16 18:17:31 +01:00
Debug : : log ( LOG , " New mouse has libinput sens %.2f (%.2f) with accel profile %i (%i) " , libinput_device_config_accel_get_speed ( LIBINPUTDEV ) ,
libinput_device_config_accel_get_default_speed ( LIBINPUTDEV ) , libinput_device_config_accel_get_profile ( LIBINPUTDEV ) ,
libinput_device_config_accel_get_default_profile ( LIBINPUTDEV ) ) ;
2022-03-18 22:53:27 +01:00
}
wlr_cursor_attach_input_device ( g_pCompositor - > m_sWLRCursor , mouse ) ;
2022-03-19 11:27:19 +01:00
2022-10-27 13:58:10 +02:00
PMOUSE - > connected = true ;
2022-11-04 11:48:42 +01:00
setPointerConfigs ( ) ;
PMOUSE - > hyprListener_destroyMouse . initCallback ( & mouse - > events . destroy , & Events : : listener_destroyMouse , PMOUSE , " Mouse " ) ;
2022-04-17 21:40:04 +02:00
g_pCompositor - > m_sSeat . mouse = PMOUSE ;
2022-06-24 23:27:02 +02:00
m_tmrLastCursorMovement . reset ( ) ;
2022-03-19 11:27:19 +01:00
Debug : : log ( LOG , " New mouse created, pointer WLR: %x " , mouse ) ;
2022-03-18 22:53:27 +01:00
}
2022-10-05 22:21:22 +02:00
void CInputManager : : setPointerConfigs ( ) {
2022-07-19 19:26:53 +02:00
for ( auto & m : m_lMice ) {
2022-10-05 22:21:22 +02:00
const auto PPOINTER = & m ;
2022-07-19 19:26:53 +02:00
2022-12-16 18:17:31 +01:00
auto devname = PPOINTER - > name ;
2022-07-28 21:38:30 +02:00
const auto HASCONFIG = g_pConfigManager - > deviceConfigExists ( devname ) ;
2022-07-19 19:26:53 +02:00
2022-10-27 13:58:10 +02:00
if ( HASCONFIG ) {
const auto ENABLED = g_pConfigManager - > getDeviceInt ( devname , " enabled " ) ;
if ( ENABLED & & ! m . connected ) {
wlr_cursor_attach_input_device ( g_pCompositor - > m_sWLRCursor , m . mouse ) ;
m . connected = true ;
} else if ( ! ENABLED & & m . connected ) {
wlr_cursor_detach_input_device ( g_pCompositor - > m_sWLRCursor , m . mouse ) ;
m . connected = false ;
}
}
2022-07-19 19:26:53 +02:00
if ( wlr_input_device_is_libinput ( m . mouse ) ) {
const auto LIBINPUTDEV = ( libinput_device * ) wlr_libinput_get_device_handle ( m . mouse ) ;
2022-12-16 18:17:31 +01:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " clickfinger_behavior " ) : g_pConfigManager - > getInt ( " input:touchpad:clickfinger_behavior " ) ) = =
0 ) // toggle software buttons or clickfinger
2022-07-19 19:26:53 +02:00
libinput_device_config_click_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_CLICK_METHOD_BUTTON_AREAS ) ;
else
libinput_device_config_click_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_CLICK_METHOD_CLICKFINGER ) ;
2022-10-05 04:25:26 +02:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " left_handed " ) : g_pConfigManager - > getInt ( " input:left_handed " ) ) = = 0 )
2022-10-04 22:46:41 +02:00
libinput_device_config_left_handed_set ( LIBINPUTDEV , 0 ) ;
else
libinput_device_config_left_handed_set ( LIBINPUTDEV , 1 ) ;
2022-12-16 18:17:31 +01:00
if ( libinput_device_config_middle_emulation_is_available ( LIBINPUTDEV ) ) { // middleclick on r+l mouse button pressed
2022-07-28 21:38:30 +02:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " middle_button_emulation " ) : g_pConfigManager - > getInt ( " input:touchpad:middle_button_emulation " ) ) = = 1 )
2022-07-19 19:26:53 +02:00
libinput_device_config_middle_emulation_set_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_MIDDLE_EMULATION_ENABLED ) ;
else
libinput_device_config_middle_emulation_set_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_MIDDLE_EMULATION_DISABLED ) ;
2023-02-05 15:17:23 +01:00
const auto TAP_MAP = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " tap_button_map " ) : g_pConfigManager - > getString ( " input:touchpad:tap_button_map " ) ;
if ( TAP_MAP = = " " | | TAP_MAP = = " lrm " )
libinput_device_config_tap_set_button_map ( LIBINPUTDEV , LIBINPUT_CONFIG_TAP_MAP_LRM ) ;
else if ( TAP_MAP = = " lmr " )
libinput_device_config_tap_set_button_map ( LIBINPUTDEV , LIBINPUT_CONFIG_TAP_MAP_LMR ) ;
else
Debug : : log ( WARN , " Tap button mapping unknown " ) ;
2022-07-19 19:26:53 +02:00
}
2022-10-05 22:21:22 +02:00
const auto SCROLLMETHOD = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " scroll_method " ) : g_pConfigManager - > getString ( " input:scroll_method " ) ;
2022-11-24 18:11:21 +01:00
if ( SCROLLMETHOD = = " " ) {
2022-10-05 22:21:22 +02:00
libinput_device_config_scroll_set_method ( LIBINPUTDEV , libinput_device_config_scroll_get_default_method ( LIBINPUTDEV ) ) ;
} else if ( SCROLLMETHOD = = " no_scroll " ) {
libinput_device_config_scroll_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_SCROLL_NO_SCROLL ) ;
} else if ( SCROLLMETHOD = = " 2fg " ) {
libinput_device_config_scroll_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_SCROLL_2FG ) ;
} else if ( SCROLLMETHOD = = " edge " ) {
libinput_device_config_scroll_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_SCROLL_EDGE ) ;
} else if ( SCROLLMETHOD = = " on_button_down " ) {
libinput_device_config_scroll_set_method ( LIBINPUTDEV , LIBINPUT_CONFIG_SCROLL_ON_BUTTON_DOWN ) ;
} else {
Debug : : log ( WARN , " Scroll method unknown " ) ;
}
2022-12-22 13:05:26 +01:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " tap-and-drag " ) : g_pConfigManager - > getInt ( " input:touchpad:tap-and-drag " ) ) = = 0 )
libinput_device_config_tap_set_drag_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_DRAG_DISABLED ) ;
else
libinput_device_config_tap_set_drag_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_DRAG_ENABLED ) ;
2022-07-28 21:38:30 +02:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " drag_lock " ) : g_pConfigManager - > getInt ( " input:touchpad:drag_lock " ) ) = = 0 )
2022-07-19 19:26:53 +02:00
libinput_device_config_tap_set_drag_lock_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_DRAG_LOCK_DISABLED ) ;
else
libinput_device_config_tap_set_drag_lock_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_DRAG_LOCK_ENABLED ) ;
2022-12-16 18:17:31 +01:00
if ( libinput_device_config_tap_get_finger_count ( LIBINPUTDEV ) ) // this is for tapping (like on a laptop)
2022-07-28 21:38:30 +02:00
if ( ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " tap-to-click " ) : g_pConfigManager - > getInt ( " input:touchpad:tap-to-click " ) ) = = 1 )
2022-07-19 19:26:53 +02:00
libinput_device_config_tap_set_enabled ( LIBINPUTDEV , LIBINPUT_CONFIG_TAP_ENABLED ) ;
if ( libinput_device_config_scroll_has_natural_scroll ( LIBINPUTDEV ) ) {
double w = 0 , h = 0 ;
2022-12-16 18:17:31 +01:00
if ( libinput_device_has_capability ( LIBINPUTDEV , LIBINPUT_DEVICE_CAP_POINTER ) & &
libinput_device_get_size ( LIBINPUTDEV , & w , & h ) = = 0 ) // pointer with size is a touchpad
libinput_device_config_scroll_set_natural_scroll_enabled (
LIBINPUTDEV , ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " natural_scroll " ) : g_pConfigManager - > getInt ( " input:touchpad:natural_scroll " ) ) ) ;
2022-07-19 19:26:53 +02:00
else
2022-12-16 18:17:31 +01:00
libinput_device_config_scroll_set_natural_scroll_enabled (
LIBINPUTDEV , ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " natural_scroll " ) : g_pConfigManager - > getInt ( " input:natural_scroll " ) ) ) ;
2022-07-19 19:26:53 +02:00
}
if ( libinput_device_config_dwt_is_available ( LIBINPUTDEV ) ) {
2022-12-16 18:17:31 +01:00
const auto DWT = static_cast < enum libinput_config_dwt_state > (
( HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " disable_while_typing " ) : g_pConfigManager - > getInt ( " input:touchpad:disable_while_typing " ) ) ! = 0 ) ;
2022-07-19 19:26:53 +02:00
libinput_device_config_dwt_set_enabled ( LIBINPUTDEV , DWT ) ;
}
2022-12-16 18:17:31 +01:00
const auto LIBINPUTSENS =
std : : clamp ( ( HASCONFIG ? g_pConfigManager - > getDeviceFloat ( devname , " sensitivity " ) : g_pConfigManager - > getFloat ( " input:sensitivity " ) ) , - 1.f , 1.f ) ;
2022-07-19 19:26:53 +02:00
libinput_device_config_accel_set_speed ( LIBINPUTDEV , LIBINPUTSENS ) ;
2022-10-05 22:21:22 +02:00
const auto ACCELPROFILE = HASCONFIG ? g_pConfigManager - > getDeviceString ( devname , " accel_profile " ) : g_pConfigManager - > getString ( " input:accel_profile " ) ;
2022-11-24 18:11:21 +01:00
if ( ACCELPROFILE = = " " ) {
2022-10-05 22:21:22 +02:00
libinput_device_config_accel_set_profile ( LIBINPUTDEV , libinput_device_config_accel_get_default_profile ( LIBINPUTDEV ) ) ;
} else if ( ACCELPROFILE = = " adaptive " ) {
libinput_device_config_accel_set_profile ( LIBINPUTDEV , LIBINPUT_CONFIG_ACCEL_PROFILE_ADAPTIVE ) ;
} else if ( ACCELPROFILE = = " flat " ) {
libinput_device_config_accel_set_profile ( LIBINPUTDEV , LIBINPUT_CONFIG_ACCEL_PROFILE_FLAT ) ;
} else {
Debug : : log ( WARN , " Unknown acceleration profile, falling back to default " ) ;
2022-11-22 00:26:18 +01:00
}
const auto SCROLLBUTTON = HASCONFIG ? g_pConfigManager - > getDeviceInt ( devname , " scroll_button " ) : g_pConfigManager - > getInt ( " input:scroll_button " ) ;
libinput_device_config_scroll_set_button ( LIBINPUTDEV , SCROLLBUTTON = = 0 ? libinput_device_config_scroll_get_default_button ( LIBINPUTDEV ) : SCROLLBUTTON ) ;
2022-10-05 22:21:22 +02:00
2022-07-19 19:26:53 +02:00
Debug : : log ( LOG , " Applied config to mouse %s, sens %.2f " , m . name . c_str ( ) , LIBINPUTSENS ) ;
}
}
}
2022-03-18 23:25:26 +01:00
void CInputManager : : destroyKeyboard ( SKeyboard * pKeyboard ) {
2022-03-28 22:31:39 +02:00
pKeyboard - > hyprListener_keyboardDestroy . removeCallback ( ) ;
pKeyboard - > hyprListener_keyboardMod . removeCallback ( ) ;
pKeyboard - > hyprListener_keyboardKey . removeCallback ( ) ;
2022-03-18 23:25:26 +01:00
2022-06-03 18:59:39 +02:00
if ( pKeyboard - > active ) {
m_lKeyboards . remove ( * pKeyboard ) ;
if ( m_lKeyboards . size ( ) > 0 ) {
2022-12-16 18:17:31 +01:00
m_pActiveKeyboard = & m_lKeyboards . back ( ) ;
2022-06-03 18:59:39 +02:00
m_pActiveKeyboard - > active = true ;
} else {
m_pActiveKeyboard = nullptr ;
}
2023-01-02 12:08:00 +01:00
} else
m_lKeyboards . remove ( * pKeyboard ) ;
2022-03-18 23:25:26 +01:00
}
void CInputManager : : destroyMouse ( wlr_input_device * mouse ) {
2022-04-17 21:40:04 +02:00
for ( auto & m : m_lMice ) {
if ( m . mouse = = mouse ) {
m_lMice . remove ( m ) ;
2022-05-12 12:13:02 +02:00
break ;
2022-04-17 21:40:04 +02:00
}
}
g_pCompositor - > m_sSeat . mouse = m_lMice . size ( ) > 0 ? & m_lMice . front ( ) : nullptr ;
2022-05-12 12:13:02 +02:00
2022-05-18 14:57:08 +02:00
if ( g_pCompositor - > m_sSeat . mouse )
2022-08-09 20:36:21 +02:00
unconstrainMouse ( ) ;
2022-03-18 23:25:26 +01:00
}
2022-12-16 18:17:31 +01:00
void CInputManager : : updateKeyboardsLeds ( wlr_input_device * pKeyboard ) {
2022-11-12 12:12:37 +01:00
auto keyboard = wlr_keyboard_from_input_device ( pKeyboard ) ;
2022-12-16 18:17:31 +01:00
2022-11-12 12:12:37 +01:00
if ( keyboard - > xkb_state = = NULL ) {
2022-12-16 18:17:31 +01:00
return ;
}
uint32_t leds = 0 ;
for ( uint32_t i = 0 ; i < WLR_LED_COUNT ; + + i ) {
if ( xkb_state_led_index_is_active ( keyboard - > xkb_state , keyboard - > led_indexes [ i ] ) ) {
leds | = ( 1 < < i ) ;
}
}
2022-11-12 12:12:37 +01:00
for ( auto & kb : m_lKeyboards ) {
if ( ( kb . isVirtual & & shouldIgnoreVirtualKeyboard ( & kb ) ) | | kb . keyboard = = pKeyboard )
continue ;
wlr_keyboard_led_update ( wlr_keyboard_from_input_device ( kb . keyboard ) , leds ) ;
}
}
2022-03-24 15:57:46 +01:00
void CInputManager : : onKeyboardKey ( wlr_keyboard_key_event * e , SKeyboard * pKeyboard ) {
2022-12-16 18:20:51 +01:00
if ( ! pKeyboard - > enabled )
return ;
2023-03-16 01:30:07 +01:00
static auto * const PDPMS = & g_pConfigManager - > getConfigValuePtr ( " misc:key_press_enables_dpms " ) - > intValue ;
2023-03-16 16:29:48 +01:00
if ( * PDPMS & & ! g_pCompositor - > m_bDPMSStateON ) {
2023-03-12 14:46:38 +01:00
// enable dpms
g_pKeybindManager - > dpms ( " on " ) ;
}
2022-07-20 22:45:06 +02:00
bool passEvent = g_pKeybindManager - > onKeyEvent ( e , pKeyboard ) ;
2022-03-18 23:06:45 +01:00
2022-03-22 18:29:13 +01:00
wlr_idle_notify_activity ( g_pCompositor - > m_sWLRIdle , g_pCompositor - > m_sSeat . seat ) ;
2022-03-17 20:55:04 +01:00
2022-07-20 22:45:06 +02:00
if ( passEvent ) {
2022-08-05 16:21:08 +02:00
const auto PIMEGRAB = m_sIMERelay . getIMEKeyboardGrab ( pKeyboard ) ;
if ( PIMEGRAB & & PIMEGRAB - > pWlrKbGrab & & PIMEGRAB - > pWlrKbGrab - > input_method ) {
wlr_input_method_keyboard_grab_v2_set_keyboard ( PIMEGRAB - > pWlrKbGrab , wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ) ;
wlr_input_method_keyboard_grab_v2_send_key ( PIMEGRAB - > pWlrKbGrab , e - > time_msec , e - > keycode , e - > state ) ;
} else {
wlr_seat_set_keyboard ( g_pCompositor - > m_sSeat . seat , wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ) ;
wlr_seat_keyboard_notify_key ( g_pCompositor - > m_sSeat . seat , e - > time_msec , e - > keycode , e - > state ) ;
}
2022-11-12 12:12:37 +01:00
updateKeyboardsLeds ( pKeyboard - > keyboard ) ;
2022-03-19 22:03:40 +01:00
}
2022-03-18 23:06:45 +01:00
}
2022-03-17 20:55:04 +01:00
2022-03-18 23:06:45 +01:00
void CInputManager : : onKeyboardMod ( void * data , SKeyboard * pKeyboard ) {
2022-12-16 18:20:51 +01:00
if ( ! pKeyboard - > enabled )
return ;
2022-08-05 16:21:08 +02:00
const auto PIMEGRAB = m_sIMERelay . getIMEKeyboardGrab ( pKeyboard ) ;
2022-09-04 18:46:28 +02:00
const auto ALLMODS = accumulateModsFromAllKBs ( ) ;
2022-12-16 18:17:31 +01:00
auto MODS = wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) - > modifiers ;
MODS . depressed = ALLMODS ;
2022-09-04 18:46:28 +02:00
2022-08-05 16:21:08 +02:00
if ( PIMEGRAB & & PIMEGRAB - > pWlrKbGrab & & PIMEGRAB - > pWlrKbGrab - > input_method ) {
wlr_input_method_keyboard_grab_v2_set_keyboard ( PIMEGRAB - > pWlrKbGrab , wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ) ;
2022-09-04 18:46:28 +02:00
wlr_input_method_keyboard_grab_v2_send_modifiers ( PIMEGRAB - > pWlrKbGrab , & MODS ) ;
2022-08-05 16:21:08 +02:00
} else {
wlr_seat_set_keyboard ( g_pCompositor - > m_sSeat . seat , wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ) ;
2022-09-04 18:46:28 +02:00
wlr_seat_keyboard_notify_modifiers ( g_pCompositor - > m_sSeat . seat , & MODS ) ;
2022-08-05 16:21:08 +02:00
}
2022-08-18 17:17:33 +02:00
2022-11-12 12:12:37 +01:00
updateKeyboardsLeds ( pKeyboard - > keyboard ) ;
2022-08-18 17:17:33 +02:00
const auto PWLRKB = wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ;
if ( PWLRKB - > modifiers . group ! = pKeyboard - > activeLayout ) {
2022-12-16 18:17:31 +01:00
pKeyboard - > activeLayout = PWLRKB - > modifiers . group ;
2022-08-18 17:17:33 +02:00
2023-02-19 21:54:53 +01:00
const auto LAYOUT = getActiveLayoutForKeyboard ( pKeyboard ) ;
g_pEventManager - > postEvent ( SHyprIPCEvent { " activelayout " , pKeyboard - > name + " , " + LAYOUT } , true ) ; // force as this should ALWAYS be sent
EMIT_HOOK_EVENT ( " activeLayout " , ( std : : vector < void * > { pKeyboard , ( void * ) & LAYOUT } ) ) ;
2022-12-16 18:17:31 +01:00
}
2022-03-21 19:28:43 +01:00
}
2022-11-07 23:22:13 +01:00
bool CInputManager : : shouldIgnoreVirtualKeyboard ( SKeyboard * pKeyboard ) {
2022-12-16 18:17:31 +01:00
return ! pKeyboard | |
( m_sIMERelay . m_pKeyboardGrab & &
wl_resource_get_client ( m_sIMERelay . m_pKeyboardGrab - > pWlrKbGrab - > resource ) = = wl_resource_get_client ( wlr_input_device_get_virtual_keyboard ( pKeyboard - > keyboard ) - > resource ) ) ;
2022-11-07 23:22:13 +01:00
}
2022-03-21 19:28:43 +01:00
void CInputManager : : refocus ( ) {
2022-04-13 20:19:40 +02:00
mouseMoveUnified ( 0 , true ) ;
2022-03-31 17:25:23 +02:00
}
void CInputManager : : updateDragIcon ( ) {
2022-08-06 22:26:32 +02:00
if ( ! m_sDrag . dragIcon )
2022-03-31 17:25:23 +02:00
return ;
2022-08-06 22:26:32 +02:00
switch ( m_sDrag . dragIcon - > drag - > grab_type ) {
2022-12-16 18:17:31 +01:00
case WLR_DRAG_GRAB_KEYBOARD : break ;
2022-06-29 14:15:08 +02:00
case WLR_DRAG_GRAB_KEYBOARD_POINTER : {
2022-08-06 22:26:32 +02:00
wlr_box box = { m_sDrag . pos . x - 2 , m_sDrag . pos . y - 2 , m_sDrag . dragIcon - > surface - > current . width + 4 , m_sDrag . dragIcon - > surface - > current . height + 4 } ;
2022-06-29 14:15:08 +02:00
g_pHyprRenderer - > damageBox ( & box ) ;
2022-08-06 22:26:32 +02:00
m_sDrag . pos = getMouseCoordsInternal ( ) ;
2022-03-31 17:25:23 +02:00
break ;
2022-06-29 14:15:08 +02:00
}
2022-12-16 18:17:31 +01:00
default : break ;
2022-03-31 17:25:23 +02:00
}
2022-04-17 21:40:04 +02:00
}
void CInputManager : : recheckConstraint ( SMouse * pMouse ) {
if ( ! pMouse - > currentConstraint )
return ;
const auto PREGION = & pMouse - > currentConstraint - > region ;
if ( pMouse - > currentConstraint - > type = = WLR_POINTER_CONSTRAINT_V1_CONFINED ) {
pixman_region32_copy ( & pMouse - > confinedTo , PREGION ) ;
} else {
pixman_region32_clear ( & pMouse - > confinedTo ) ;
}
}
void CInputManager : : constrainMouse ( SMouse * pMouse , wlr_pointer_constraint_v1 * constraint ) {
if ( pMouse - > currentConstraint = = constraint )
return ;
2022-12-16 18:17:31 +01:00
const auto PWINDOW = g_pCompositor - > getWindowFromSurface ( constraint - > surface ) ;
2022-04-17 21:40:04 +02:00
const auto MOUSECOORDS = getMouseCoordsInternal ( ) ;
pMouse - > hyprListener_commitConstraint . removeCallback ( ) ;
if ( pMouse - > currentConstraint ) {
if ( ! constraint ) {
// warpe to hint
if ( constraint - > current . committed & WLR_POINTER_CONSTRAINT_V1_STATE_CURSOR_HINT ) {
if ( PWINDOW ) {
2022-04-25 13:40:46 +02:00
if ( PWINDOW - > m_bIsX11 ) {
2022-12-16 18:17:31 +01:00
wlr_cursor_warp ( g_pCompositor - > m_sWLRCursor , nullptr , constraint - > current . cursor_hint . x + PWINDOW - > m_uSurface . xwayland - > x ,
PWINDOW - > m_uSurface . xwayland - > y + PWINDOW - > m_vRealPosition . vec ( ) . y ) ;
2022-04-25 13:40:46 +02:00
wlr_seat_pointer_warp ( constraint - > seat , constraint - > current . cursor_hint . x , constraint - > current . cursor_hint . y ) ;
} else {
2022-12-16 18:17:31 +01:00
wlr_cursor_warp ( g_pCompositor - > m_sWLRCursor , nullptr , constraint - > current . cursor_hint . x + PWINDOW - > m_vRealPosition . vec ( ) . x ,
constraint - > current . cursor_hint . y + PWINDOW - > m_vRealPosition . vec ( ) . y ) ;
2022-04-17 21:40:04 +02:00
2022-04-25 13:40:46 +02:00
wlr_seat_pointer_warp ( constraint - > seat , constraint - > current . cursor_hint . x , constraint - > current . cursor_hint . y ) ;
}
2022-04-17 21:40:04 +02:00
}
2022-12-10 15:43:46 +01:00
const auto PCONSTRAINT = constraintFromWlr ( constraint ) ;
if ( PCONSTRAINT ) { // should never be null but who knows
PCONSTRAINT - > positionHint = Vector2D ( constraint - > current . cursor_hint . x , constraint - > current . cursor_hint . y ) ;
2022-12-16 18:17:31 +01:00
PCONSTRAINT - > hintSet = true ;
2022-12-10 15:43:46 +01:00
}
2022-04-17 21:40:04 +02:00
}
}
2022-09-25 20:07:48 +02:00
2022-04-17 21:40:04 +02:00
wlr_pointer_constraint_v1_send_deactivated ( pMouse - > currentConstraint ) ;
}
pMouse - > currentConstraint = constraint ;
2022-12-16 18:17:31 +01:00
pMouse - > constraintActive = true ;
2022-04-17 21:40:04 +02:00
if ( pixman_region32_not_empty ( & constraint - > current . region ) ) {
pixman_region32_intersect ( & constraint - > region , & constraint - > surface - > input_region , & constraint - > current . region ) ;
} else {
pixman_region32_copy ( & constraint - > region , & constraint - > surface - > input_region ) ;
}
// warp to the constraint
recheckConstraint ( pMouse ) ;
wlr_pointer_constraint_v1_send_activated ( pMouse - > currentConstraint ) ;
pMouse - > hyprListener_commitConstraint . initCallback ( & pMouse - > currentConstraint - > surface - > events . commit , & Events : : listener_commitConstraint , pMouse , " Mouse constraint commit " ) ;
Debug : : log ( LOG , " Constrained mouse to %x " , pMouse - > currentConstraint ) ;
}
2022-08-09 20:36:21 +02:00
void CInputManager : : unconstrainMouse ( ) {
2023-02-28 20:02:30 +01:00
if ( ! g_pCompositor - > m_sSeat . mouse | | ! g_pCompositor - > m_sSeat . mouse - > currentConstraint )
2022-08-09 20:36:21 +02:00
return ;
const auto CONSTRAINTWINDOW = g_pCompositor - > getConstraintWindow ( g_pCompositor - > m_sSeat . mouse ) ;
if ( CONSTRAINTWINDOW ) {
2023-03-20 16:00:58 +01:00
g_pXWaylandManager - > activateSurface ( CONSTRAINTWINDOW - > m_pWLSurface . wlr ( ) , false ) ;
2022-08-09 20:36:21 +02:00
}
wlr_pointer_constraint_v1_send_deactivated ( g_pCompositor - > m_sSeat . mouse - > currentConstraint ) ;
g_pCompositor - > m_sSeat . mouse - > constraintActive = false ;
// TODO: its better to somehow detect the workspace...
g_pCompositor - > m_sSeat . mouse - > currentConstraint = nullptr ;
g_pCompositor - > m_sSeat . mouse - > hyprListener_commitConstraint . removeCallback ( ) ;
}
2022-04-17 21:40:04 +02:00
void Events : : listener_commitConstraint ( void * owner , void * data ) {
2022-12-10 15:43:46 +01:00
const auto PMOUSE = ( SMouse * ) owner ;
if ( PMOUSE - > currentConstraint - > current . committed & WLR_POINTER_CONSTRAINT_V1_STATE_CURSOR_HINT ) {
const auto PCONSTRAINT = g_pInputManager - > constraintFromWlr ( PMOUSE - > currentConstraint ) ;
2022-12-16 18:17:31 +01:00
if ( PCONSTRAINT ) { // should never be null but who knows
2022-12-10 15:43:46 +01:00
PCONSTRAINT - > positionHint = Vector2D ( PMOUSE - > currentConstraint - > current . cursor_hint . x , PMOUSE - > currentConstraint - > current . cursor_hint . y ) ;
2022-12-16 18:17:31 +01:00
PCONSTRAINT - > hintSet = true ;
2022-12-10 15:43:46 +01:00
}
}
2023-01-03 13:06:13 +01:00
if ( PMOUSE - > currentConstraint - > current . committed & WLR_POINTER_CONSTRAINT_V1_STATE_REGION ) {
if ( pixman_region32_not_empty ( & PMOUSE - > currentConstraint - > current . region ) ) {
pixman_region32_intersect ( & PMOUSE - > currentConstraint - > region , & PMOUSE - > currentConstraint - > surface - > input_region , & PMOUSE - > currentConstraint - > current . region ) ;
} else {
pixman_region32_copy ( & PMOUSE - > currentConstraint - > region , & PMOUSE - > currentConstraint - > surface - > input_region ) ;
}
g_pInputManager - > recheckConstraint ( PMOUSE ) ;
}
2022-04-17 21:40:04 +02:00
}
2022-06-09 12:46:55 +02:00
2022-12-05 15:28:27 +01:00
void CInputManager : : updateCapabilities ( ) {
uint32_t caps = 0 ;
if ( ! m_lKeyboards . empty ( ) )
caps | = WL_SEAT_CAPABILITY_KEYBOARD ;
if ( ! m_lMice . empty ( ) )
caps | = WL_SEAT_CAPABILITY_POINTER ;
if ( ! m_lTouchDevices . empty ( ) )
caps | = WL_SEAT_CAPABILITY_TOUCH ;
if ( ! m_lTabletTools . empty ( ) )
caps | = WL_SEAT_CAPABILITY_POINTER ;
wlr_seat_set_capabilities ( g_pCompositor - > m_sSeat . seat , caps ) ;
m_uiCapabilities = caps ;
2022-07-09 23:24:08 +02:00
}
uint32_t CInputManager : : accumulateModsFromAllKBs ( ) {
uint32_t finalMask = 0 ;
for ( auto & kb : m_lKeyboards ) {
2022-11-07 23:22:13 +01:00
if ( kb . isVirtual & & shouldIgnoreVirtualKeyboard ( & kb ) )
2022-09-05 11:19:40 +02:00
continue ;
2022-12-16 18:20:51 +01:00
if ( ! kb . enabled )
continue ;
2022-07-11 23:09:35 +02:00
finalMask | = wlr_keyboard_get_modifiers ( wlr_keyboard_from_input_device ( kb . keyboard ) ) ;
2022-07-09 23:24:08 +02:00
}
return finalMask ;
2022-07-12 13:11:54 +02:00
}
2022-08-16 16:10:20 +02:00
std : : string CInputManager : : getActiveLayoutForKeyboard ( SKeyboard * pKeyboard ) {
2022-12-16 18:17:31 +01:00
const auto WLRKB = wlr_keyboard_from_input_device ( pKeyboard - > keyboard ) ;
const auto KEYMAP = WLRKB - > keymap ;
const auto STATE = WLRKB - > xkb_state ;
2022-08-16 16:10:20 +02:00
const auto LAYOUTSNUM = xkb_keymap_num_layouts ( KEYMAP ) ;
2022-08-16 16:15:43 +02:00
for ( uint32_t i = 0 ; i < LAYOUTSNUM ; + + i ) {
2022-08-16 16:10:20 +02:00
if ( xkb_state_layout_index_is_active ( STATE , i , XKB_STATE_LAYOUT_EFFECTIVE ) ) {
const auto LAYOUTNAME = xkb_keymap_layout_get_name ( KEYMAP , i ) ;
2022-09-25 20:07:48 +02:00
if ( LAYOUTNAME )
2022-08-16 16:10:20 +02:00
return std : : string ( LAYOUTNAME ) ;
return " error " ;
}
}
return " none " ;
}
2022-08-18 17:50:32 +02:00
void CInputManager : : disableAllKeyboards ( bool virt ) {
for ( auto & k : m_lKeyboards ) {
if ( k . isVirtual ! = virt )
continue ;
k . active = false ;
}
2022-08-19 20:01:51 +02:00
}
2022-09-21 15:39:25 +02:00
void CInputManager : : newTouchDevice ( wlr_input_device * pDevice ) {
2022-12-16 18:17:31 +01:00
const auto PNEWDEV = & m_lTouchDevices . emplace_back ( ) ;
2022-09-21 15:39:25 +02:00
PNEWDEV - > pWlrDevice = pDevice ;
2022-10-07 16:03:52 +02:00
try {
2022-12-03 21:36:52 +01:00
PNEWDEV - > name = getNameForNewDevice ( pDevice - > name ) ;
2022-12-16 18:17:31 +01:00
} catch ( std : : exception & e ) {
2022-10-07 16:03:52 +02:00
Debug : : log ( ERR , " Touch Device had no name??? " ) ; // logic error
}
setTouchDeviceConfigs ( ) ;
2022-09-21 15:39:25 +02:00
wlr_cursor_attach_input_device ( g_pCompositor - > m_sWLRCursor , pDevice ) ;
Debug : : log ( LOG , " New touch device added at %x " , PNEWDEV ) ;
2022-12-16 18:17:31 +01:00
PNEWDEV - > hyprListener_destroy . initCallback (
& pDevice - > events . destroy , [ & ] ( void * owner , void * data ) { destroyTouchDevice ( ( STouchDevice * ) data ) ; } , PNEWDEV , " TouchDevice " ) ;
2022-09-21 15:39:25 +02:00
}
2022-10-07 16:03:52 +02:00
void CInputManager : : setTouchDeviceConfigs ( ) {
for ( auto & m : m_lTouchDevices ) {
const auto PTOUCHDEV = & m ;
2022-12-03 16:56:07 +01:00
const auto HASCONFIG = g_pConfigManager - > deviceConfigExists ( PTOUCHDEV - > name ) ;
2022-10-07 16:03:52 +02:00
if ( wlr_input_device_is_libinput ( m . pWlrDevice ) ) {
const auto LIBINPUTDEV = ( libinput_device * ) wlr_libinput_get_device_handle ( m . pWlrDevice ) ;
2022-12-16 18:17:31 +01:00
const int ROTATION =
2022-12-21 16:11:39 +01:00
std : : clamp ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( PTOUCHDEV - > name , " transform " ) : g_pConfigManager - > getInt ( " input:touchdevice:transform " ) , 0 , 7 ) ;
2022-10-07 16:03:52 +02:00
libinput_device_config_calibration_set_matrix ( LIBINPUTDEV , MATRICES [ ROTATION ] ) ;
2022-10-14 13:38:44 +02:00
2022-12-21 16:11:39 +01:00
const auto OUTPUT = HASCONFIG ? g_pConfigManager - > getDeviceString ( PTOUCHDEV - > name , " output " ) : g_pConfigManager - > getString ( " input:touchdevice:output " ) ;
2022-10-14 13:38:44 +02:00
if ( ! OUTPUT . empty ( ) & & OUTPUT ! = STRVAL_EMPTY )
PTOUCHDEV - > boundOutput = OUTPUT ;
else
PTOUCHDEV - > boundOutput = " " ;
2022-10-07 16:03:52 +02:00
}
}
}
2022-12-21 16:11:39 +01:00
void CInputManager : : setTabletConfigs ( ) {
for ( auto & t : m_lTablets ) {
const auto HASCONFIG = g_pConfigManager - > deviceConfigExists ( t . name ) ;
2023-01-15 20:38:58 +01:00
if ( wlr_input_device_is_libinput ( t . wlrDevice ) ) {
const auto LIBINPUTDEV = ( libinput_device * ) wlr_libinput_get_device_handle ( t . wlrDevice ) ;
2022-12-21 16:11:39 +01:00
2023-01-15 20:38:58 +01:00
const int ROTATION = std : : clamp ( HASCONFIG ? g_pConfigManager - > getDeviceInt ( t . name , " transform " ) : g_pConfigManager - > getInt ( " input:tablet:transform " ) , 0 , 7 ) ;
2023-01-28 19:10:57 +01:00
Debug : : log ( LOG , " Setting calibration matrix for device %s " , t . name . c_str ( ) ) ;
2023-01-15 20:38:58 +01:00
libinput_device_config_calibration_set_matrix ( LIBINPUTDEV , MATRICES [ ROTATION ] ) ;
const auto OUTPUT = HASCONFIG ? g_pConfigManager - > getDeviceString ( t . name , " output " ) : g_pConfigManager - > getString ( " input:tablet:output " ) ;
const auto PMONITOR = g_pCompositor - > getMonitorFromString ( OUTPUT ) ;
if ( ! OUTPUT . empty ( ) & & OUTPUT ! = STRVAL_EMPTY & & PMONITOR ) {
2022-12-21 16:11:39 +01:00
wlr_cursor_map_input_to_output ( g_pCompositor - > m_sWLRCursor , t . wlrDevice , PMONITOR - > output ) ;
wlr_cursor_map_input_to_region ( g_pCompositor - > m_sWLRCursor , t . wlrDevice , nullptr ) ;
}
}
}
}
2022-09-21 15:39:25 +02:00
void CInputManager : : destroyTouchDevice ( STouchDevice * pDevice ) {
Debug : : log ( LOG , " Touch device at %x removed " , pDevice ) ;
m_lTouchDevices . remove ( * pDevice ) ;
}
2022-10-04 21:07:21 +02:00
void CInputManager : : newSwitch ( wlr_input_device * pDevice ) {
2022-12-16 18:17:31 +01:00
const auto PNEWDEV = & m_lSwitches . emplace_back ( ) ;
2022-10-04 21:07:21 +02:00
PNEWDEV - > pWlrDevice = pDevice ;
Debug : : log ( LOG , " New switch with name \" %s \" added " , pDevice - > name ) ;
2022-12-16 18:17:31 +01:00
PNEWDEV - > hyprListener_destroy . initCallback (
& pDevice - > events . destroy , [ & ] ( void * owner , void * data ) { destroySwitch ( ( SSwitchDevice * ) owner ) ; } , PNEWDEV , " SwitchDevice " ) ;
2022-10-04 21:07:21 +02:00
const auto PSWITCH = wlr_switch_from_input_device ( pDevice ) ;
2022-12-16 18:17:31 +01:00
PNEWDEV - > hyprListener_toggle . initCallback (
& PSWITCH - > events . toggle ,
[ & ] ( void * owner , void * data ) {
const auto PDEVICE = ( SSwitchDevice * ) owner ;
const auto NAME = std : : string ( PDEVICE - > pWlrDevice - > name ) ;
2023-03-16 17:30:22 +01:00
const auto E = ( wlr_switch_toggle_event * ) data ;
if ( PDEVICE - > status ! = - 1 & & PDEVICE - > status = = E - > switch_state )
return ;
2022-10-04 21:07:21 +02:00
2022-12-16 18:17:31 +01:00
Debug : : log ( LOG , " Switch %s fired, triggering binds. " , NAME . c_str ( ) ) ;
2022-10-04 21:07:21 +02:00
2022-12-16 18:17:31 +01:00
g_pKeybindManager - > onSwitchEvent ( NAME ) ;
2023-01-08 16:35:24 +01:00
2023-03-16 17:30:22 +01:00
switch ( E - > switch_state ) {
2023-01-08 16:35:24 +01:00
case WLR_SWITCH_STATE_ON :
Debug : : log ( LOG , " Switch %s turn on, triggering binds. " , NAME . c_str ( ) ) ;
g_pKeybindManager - > onSwitchOnEvent ( NAME ) ;
break ;
case WLR_SWITCH_STATE_OFF :
Debug : : log ( LOG , " Switch %s turn off, triggering binds. " , NAME . c_str ( ) ) ;
g_pKeybindManager - > onSwitchOffEvent ( NAME ) ;
break ;
}
2023-03-16 17:30:22 +01:00
PDEVICE - > status = E - > switch_state ;
2022-12-16 18:17:31 +01:00
} ,
PNEWDEV , " SwitchDevice " ) ;
2022-10-04 21:07:21 +02:00
}
void CInputManager : : destroySwitch ( SSwitchDevice * pDevice ) {
m_lSwitches . remove ( * pDevice ) ;
2022-10-05 04:25:26 +02:00
}
2022-11-15 11:39:05 +01:00
void CInputManager : : setCursorImageUntilUnset ( std : : string name ) {
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , name . c_str ( ) , g_pCompositor - > m_sWLRCursor ) ;
2023-03-30 00:44:25 +02:00
m_bCursorImageOverridden = true ;
2022-11-15 11:39:05 +01:00
}
void CInputManager : : unsetCursorImage ( ) {
2023-03-30 00:44:25 +02:00
if ( ! m_bCursorImageOverridden )
2022-11-15 11:39:05 +01:00
return ;
2023-03-30 00:44:25 +02:00
m_bCursorImageOverridden = false ;
2022-11-15 11:39:05 +01:00
if ( ! g_pHyprRenderer - > m_bWindowRequestedCursorHide )
wlr_xcursor_manager_set_cursor_image ( g_pCompositor - > m_sWLRXCursorMgr , " left_ptr " , g_pCompositor - > m_sWLRCursor ) ;
}
2022-12-03 16:56:07 +01:00
std : : string CInputManager : : deviceNameToInternalString ( std : : string in ) {
std : : replace ( in . begin ( ) , in . end ( ) , ' ' , ' - ' ) ;
std : : transform ( in . begin ( ) , in . end ( ) , in . begin ( ) , : : tolower ) ;
return in ;
}
2022-12-03 21:36:52 +01:00
std : : string CInputManager : : getNameForNewDevice ( std : : string internalName ) {
auto proposedNewName = deviceNameToInternalString ( internalName ) ;
2022-12-16 18:17:31 +01:00
int dupeno = 0 ;
2022-12-03 21:36:52 +01:00
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lKeyboards . begin ( ) , m_lKeyboards . end ( ) ,
[ & ] ( const SKeyboard & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! = m_lKeyboards . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lMice . begin ( ) , m_lMice . end ( ) , [ & ] ( const SMouse & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! =
m_lMice . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lTouchDevices . begin ( ) , m_lTouchDevices . end ( ) ,
[ & ] ( const STouchDevice & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! = m_lTouchDevices . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lTabletPads . begin ( ) , m_lTabletPads . end ( ) ,
[ & ] ( const STabletPad & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! = m_lTabletPads . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lTablets . begin ( ) , m_lTablets . end ( ) ,
[ & ] ( const STablet & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! = m_lTablets . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
2022-12-16 18:17:31 +01:00
while ( std : : find_if ( m_lTabletTools . begin ( ) , m_lTabletTools . end ( ) ,
[ & ] ( const STabletTool & other ) { return other . name = = proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ; } ) ! = m_lTabletTools . end ( ) )
2022-12-03 21:36:52 +01:00
dupeno + + ;
return proposedNewName + ( dupeno = = 0 ? " " : ( " - " + std : : to_string ( dupeno ) ) ) ;
}
2022-12-10 15:43:46 +01:00
SConstraint * CInputManager : : constraintFromWlr ( wlr_pointer_constraint_v1 * constraint ) {
for ( auto & c : m_lConstraints ) {
if ( c . constraint = = constraint )
return & c ;
}
return nullptr ;
}
2023-01-11 18:38:54 +01:00
void CInputManager : : releaseAllMouseButtons ( ) {
const auto buttonsCopy = m_lCurrentlyHeldButtons ;
if ( g_pInputManager - > m_sDrag . drag )
return ;
for ( auto & mb : buttonsCopy ) {
wlr_seat_pointer_notify_button ( g_pCompositor - > m_sSeat . seat , 0 , mb , WLR_BUTTON_RELEASED ) ;
}
m_lCurrentlyHeldButtons . clear ( ) ;
}
2023-02-18 23:35:31 +01:00
void CInputManager : : setCursorIconOnBorder ( CWindow * w ) {
// do not override cursor icons set by mouse binds
if ( g_pKeybindManager - > m_bIsMouseBindActive ) {
m_eBorderIconDirection = BORDERICON_NONE ;
return ;
}
2023-04-04 00:15:33 +02:00
static auto * const PROUNDING = & g_pConfigManager - > getConfigValuePtr ( " decoration:rounding " ) - > intValue ;
static const auto * PBORDERSIZE = & g_pConfigManager - > getConfigValuePtr ( " general:border_size " ) - > intValue ;
static const auto * PEXTENDBORDERGRAB = & g_pConfigManager - > getConfigValuePtr ( " general:extend_border_grab_area " ) - > intValue ;
2023-02-18 23:35:31 +01:00
// give a small leeway (10 px) for corner icon
2023-04-04 00:15:33 +02:00
const auto CORNER = * PROUNDING + * PBORDERSIZE + 10 ;
const auto mouseCoords = getMouseCoordsInternal ( ) ;
wlr_box box = { w - > m_vRealPosition . vec ( ) . x , w - > m_vRealPosition . vec ( ) . y , w - > m_vRealSize . vec ( ) . x , w - > m_vRealSize . vec ( ) . y } ;
eBorderIconDirection direction = BORDERICON_NONE ;
wlr_box boxFullGrabInput = { box . x - * PEXTENDBORDERGRAB , box . y - * PEXTENDBORDERGRAB , box . width + 2 * * PEXTENDBORDERGRAB , box . height + 2 * * PEXTENDBORDERGRAB } ;
2023-04-04 00:17:06 +02:00
if ( ! wlr_box_contains_point ( & boxFullGrabInput , mouseCoords . x , mouseCoords . y ) | | ( ! m_lCurrentlyHeldButtons . empty ( ) & & ! currentlyDraggedWindow ) ) {
2023-04-04 00:15:33 +02:00
direction = BORDERICON_NONE ;
} else if ( wlr_box_contains_point ( & box , mouseCoords . x , mouseCoords . y ) ) {
2023-02-18 23:35:31 +01:00
if ( ! w - > isInCurvedCorner ( mouseCoords . x , mouseCoords . y ) ) {
direction = BORDERICON_NONE ;
} else {
if ( mouseCoords . y < box . y + CORNER ) {
if ( mouseCoords . x < box . x + CORNER )
direction = BORDERICON_UP_LEFT ;
else
direction = BORDERICON_UP_RIGHT ;
} else {
if ( mouseCoords . x < box . x + CORNER )
direction = BORDERICON_DOWN_LEFT ;
else
direction = BORDERICON_DOWN_RIGHT ;
}
}
} else {
if ( mouseCoords . y < box . y + CORNER ) {
if ( mouseCoords . x < box . x + CORNER )
direction = BORDERICON_UP_LEFT ;
else if ( mouseCoords . x > box . x + box . width - CORNER )
direction = BORDERICON_UP_RIGHT ;
else
direction = BORDERICON_UP ;
} else if ( mouseCoords . y > box . y + box . height - CORNER ) {
if ( mouseCoords . x < box . x + CORNER )
direction = BORDERICON_DOWN_LEFT ;
else if ( mouseCoords . x > box . x + box . width - CORNER )
direction = BORDERICON_DOWN_RIGHT ;
else
direction = BORDERICON_DOWN ;
} else {
if ( mouseCoords . x < box . x + CORNER )
direction = BORDERICON_LEFT ;
else if ( mouseCoords . x > box . x + box . width - CORNER )
direction = BORDERICON_RIGHT ;
}
}
if ( direction = = m_eBorderIconDirection )
return ;
m_eBorderIconDirection = direction ;
switch ( direction ) {
case BORDERICON_NONE : unsetCursorImage ( ) ; break ;
case BORDERICON_UP : setCursorImageUntilUnset ( " top_side " ) ; break ;
case BORDERICON_DOWN : setCursorImageUntilUnset ( " bottom_side " ) ; break ;
case BORDERICON_LEFT : setCursorImageUntilUnset ( " left_side " ) ; break ;
case BORDERICON_RIGHT : setCursorImageUntilUnset ( " right_side " ) ; break ;
case BORDERICON_UP_LEFT : setCursorImageUntilUnset ( " top_left_corner " ) ; break ;
case BORDERICON_DOWN_LEFT : setCursorImageUntilUnset ( " bottom_left_corner " ) ; break ;
case BORDERICON_UP_RIGHT : setCursorImageUntilUnset ( " top_right_corner " ) ; break ;
case BORDERICON_DOWN_RIGHT : setCursorImageUntilUnset ( " bottom_right_corner " ) ; break ;
}
}