2022-03-17 15:53:45 +01:00
|
|
|
#include "InputManager.hpp"
|
|
|
|
#include "../Compositor.hpp"
|
|
|
|
|
2022-03-24 15:57:46 +01:00
|
|
|
void CInputManager::onMouseMoved(wlr_pointer_motion_event* e) {
|
2022-03-17 15:53:45 +01:00
|
|
|
// TODO: sensitivity
|
|
|
|
|
2022-03-19 20:30:21 +01:00
|
|
|
float sensitivity = g_pConfigManager->getFloat("general:sensitivity");
|
2022-03-17 15:53:45 +01:00
|
|
|
|
2022-03-24 15:57:46 +01:00
|
|
|
wlr_cursor_move(g_pCompositor->m_sWLRCursor, &e->pointer->base, e->delta_x * sensitivity, e->delta_y * sensitivity);
|
2022-03-18 20:42:49 +01:00
|
|
|
|
2022-03-18 23:52:36 +01:00
|
|
|
mouseMoveUnified(e->time_msec);
|
2022-03-18 20:42:49 +01:00
|
|
|
// todo: pointer
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputManager::mouseMoveUnified(uint32_t time) {
|
|
|
|
|
2022-03-31 17:25:23 +02:00
|
|
|
// update stuff
|
|
|
|
updateDragIcon();
|
|
|
|
|
2022-04-01 23:31:12 +02:00
|
|
|
|
2022-03-31 17:25:23 +02:00
|
|
|
// focus
|
2022-03-20 14:36:55 +01:00
|
|
|
wlr_surface* foundSurface = nullptr;
|
|
|
|
Vector2D mouseCoords = getMouseCoordsInternal();
|
2022-03-31 17:25:23 +02:00
|
|
|
|
2022-03-20 14:36:55 +01:00
|
|
|
const auto PMONITOR = g_pCompositor->getMonitorFromCursor();
|
2022-03-30 20:16:23 +02:00
|
|
|
if (PMONITOR)
|
|
|
|
g_pCompositor->m_pLastMonitor = PMONITOR;
|
2022-03-31 17:25:23 +02:00
|
|
|
|
2022-03-22 21:59:14 +01:00
|
|
|
Vector2D surfaceCoords;
|
2022-03-30 20:16:23 +02:00
|
|
|
Vector2D surfacePos = Vector2D(-1337, -1337);
|
2022-03-20 14:36:55 +01:00
|
|
|
|
2022-03-21 19:18:33 +01:00
|
|
|
// first, we check if the workspace doesnt have a fullscreen window
|
|
|
|
const auto PWORKSPACE = g_pCompositor->getWorkspaceByID(PMONITOR->activeWorkspace);
|
|
|
|
if (PWORKSPACE->hasFullscreenWindow) {
|
|
|
|
const auto PFULLSCREENWINDOW = g_pCompositor->getFullscreenWindowOnWorkspace(PWORKSPACE->ID);
|
|
|
|
|
|
|
|
// should never ever happen but who knows
|
|
|
|
if (PFULLSCREENWINDOW) {
|
|
|
|
foundSurface = g_pXWaylandManager->getWindowSurface(PFULLSCREENWINDOW);
|
|
|
|
if (foundSurface)
|
|
|
|
surfacePos = PFULLSCREENWINDOW->m_vRealPosition;
|
2022-03-30 20:16:23 +02:00
|
|
|
|
|
|
|
for (auto& w : g_pCompositor->m_lWindows) {
|
|
|
|
wlr_box box = {w.m_vRealPosition.x, w.m_vRealPosition.y, w.m_vRealSize.x, w.m_vRealSize.y};
|
|
|
|
if (w.m_iWorkspaceID == PFULLSCREENWINDOW->m_iWorkspaceID && w.m_bIsMapped && w.m_bCreatedOverFullscreen && wlr_box_contains_point(&box, mouseCoords.x, mouseCoords.y)) {
|
|
|
|
foundSurface = g_pXWaylandManager->getWindowSurface(&w);
|
|
|
|
if (foundSurface)
|
|
|
|
surfacePos = w.m_vRealPosition;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2022-03-21 19:18:33 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// then surfaces on top
|
|
|
|
if (!foundSurface)
|
2022-03-22 21:59:14 +01:00
|
|
|
foundSurface = g_pCompositor->vectorToLayerSurface(mouseCoords, &PMONITOR->m_aLayerSurfaceLists[ZWLR_LAYER_SHELL_V1_LAYER_OVERLAY], &surfaceCoords);
|
2022-03-21 19:18:33 +01:00
|
|
|
|
2022-03-20 14:36:55 +01:00
|
|
|
if (!foundSurface)
|
2022-03-22 21:59:14 +01:00
|
|
|
foundSurface = g_pCompositor->vectorToLayerSurface(mouseCoords, &PMONITOR->m_aLayerSurfaceLists[ZWLR_LAYER_SHELL_V1_LAYER_TOP], &surfaceCoords);
|
2022-03-20 14:36:55 +01:00
|
|
|
|
|
|
|
// then windows
|
2022-04-01 23:31:12 +02:00
|
|
|
const auto PWINDOWIDEAL = g_pCompositor->vectorToWindowIdeal(mouseCoords);
|
|
|
|
if (!foundSurface && PWINDOWIDEAL) {
|
|
|
|
foundSurface = g_pXWaylandManager->getWindowSurface(PWINDOWIDEAL);
|
|
|
|
if (foundSurface)
|
|
|
|
surfacePos = PWINDOWIDEAL->m_vRealPosition;
|
2022-03-20 14:36:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// then surfaces below
|
|
|
|
if (!foundSurface)
|
2022-03-22 21:59:14 +01:00
|
|
|
foundSurface = g_pCompositor->vectorToLayerSurface(mouseCoords, &PMONITOR->m_aLayerSurfaceLists[ZWLR_LAYER_SHELL_V1_LAYER_BOTTOM], &surfaceCoords);
|
2022-03-20 14:36:55 +01:00
|
|
|
|
|
|
|
if (!foundSurface)
|
2022-03-22 21:59:14 +01:00
|
|
|
foundSurface = g_pCompositor->vectorToLayerSurface(mouseCoords, &PMONITOR->m_aLayerSurfaceLists[ZWLR_LAYER_SHELL_V1_LAYER_BACKGROUND], &surfaceCoords);
|
2022-03-20 14:36:55 +01:00
|
|
|
|
2022-03-18 23:52:36 +01:00
|
|
|
|
2022-03-20 14:36:55 +01:00
|
|
|
if (!foundSurface) {
|
2022-03-18 23:52:36 +01:00
|
|
|
wlr_xcursor_manager_set_cursor_image(g_pCompositor->m_sWLRXCursorMgr, "left_ptr", g_pCompositor->m_sWLRCursor);
|
|
|
|
|
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
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
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-03-30 20:16:23 +02:00
|
|
|
Vector2D surfaceLocal = surfacePos == Vector2D(-1337, -1337) ? surfaceCoords : Vector2D(g_pCompositor->m_sWLRCursor->x, g_pCompositor->m_sWLRCursor->y) - surfacePos;
|
2022-03-18 23:52:36 +01:00
|
|
|
|
2022-03-22 18:29:13 +01: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-19 20:30:21 +01:00
|
|
|
|
2022-03-22 21:59:14 +01:00
|
|
|
g_pCompositor->focusSurface(foundSurface);
|
|
|
|
|
2022-03-20 11:14:24 +01:00
|
|
|
g_pLayoutManager->getCurrentLayout()->onMouseMove(getMouseCoordsInternal());
|
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
|
|
|
|
2022-03-22 18:29:13 +01:00
|
|
|
const auto PKEYBOARD = wlr_seat_get_keyboard(g_pCompositor->m_sSeat.seat);
|
2022-03-20 18:23:16 +01:00
|
|
|
|
2022-03-18 20:42:49 +01:00
|
|
|
switch (e->state) {
|
|
|
|
case WLR_BUTTON_PRESSED:
|
2022-03-20 18:23:39 +01:00
|
|
|
if ((e->button == BTN_LEFT || e->button == BTN_RIGHT) && wlr_keyboard_get_modifiers(PKEYBOARD) == (uint32_t)g_pConfigManager->getInt("general:main_mod_internal")) {
|
2022-03-20 11:14:24 +01:00
|
|
|
currentlyDraggedWindow = g_pCompositor->windowFloatingFromCursor();
|
|
|
|
dragButton = e->button;
|
|
|
|
|
|
|
|
g_pLayoutManager->getCurrentLayout()->onBeginDragWindow();
|
2022-03-21 19:18:33 +01:00
|
|
|
|
|
|
|
return;
|
2022-03-20 11:14:24 +01:00
|
|
|
}
|
2022-03-18 20:42:49 +01:00
|
|
|
break;
|
|
|
|
case WLR_BUTTON_RELEASED:
|
2022-03-20 11:14:24 +01:00
|
|
|
currentlyDraggedWindow = nullptr;
|
|
|
|
dragButton = -1;
|
2022-03-18 20:42:49 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
Debug::log(LOG, "Seat notified of button %i (state %i) on surface %x", e->button, e->state, g_pCompositor->m_pLastFocus);
|
|
|
|
}
|
|
|
|
|
2022-03-17 16:56:33 +01: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-03-18 23:25:26 +01:00
|
|
|
m_lKeyboards.push_back(SKeyboard());
|
2022-03-17 20:55:04 +01:00
|
|
|
|
2022-03-18 23:25:26 +01:00
|
|
|
const auto PNEWKEYBOARD = &m_lKeyboards.back();
|
2022-03-17 20:55:04 +01:00
|
|
|
|
|
|
|
PNEWKEYBOARD->keyboard = keyboard;
|
|
|
|
|
|
|
|
xkb_rule_names rules;
|
|
|
|
|
|
|
|
const auto CONTEXT = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
|
|
|
|
const auto KEYMAP = xkb_keymap_new_from_names(CONTEXT, &rules, XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
|
|
|
|
wlr_keyboard_set_keymap(keyboard->keyboard, KEYMAP);
|
|
|
|
xkb_keymap_unref(KEYMAP);
|
|
|
|
xkb_context_unref(CONTEXT);
|
|
|
|
wlr_keyboard_set_repeat_info(keyboard->keyboard, 25, 600);
|
|
|
|
|
2022-03-28 22:31:39 +02:00
|
|
|
PNEWKEYBOARD->hyprListener_keyboardMod.initCallback(&keyboard->keyboard->events.modifiers, &Events::listener_keyboardMod, PNEWKEYBOARD, "Keyboard");
|
|
|
|
PNEWKEYBOARD->hyprListener_keyboardKey.initCallback(&keyboard->keyboard->events.key, &Events::listener_keyboardKey, PNEWKEYBOARD, "Keyboard");
|
|
|
|
PNEWKEYBOARD->hyprListener_keyboardDestroy.initCallback(&keyboard->events.destroy, &Events::listener_keyboardDestroy, PNEWKEYBOARD, "Keyboard");
|
2022-03-17 20:55:04 +01:00
|
|
|
|
2022-03-24 15:57:46 +01:00
|
|
|
wlr_seat_set_keyboard(g_pCompositor->m_sSeat.seat, keyboard->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
|
|
|
|
|
|
|
setKeyboardLayout();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputManager::setKeyboardLayout() {
|
|
|
|
|
|
|
|
const auto RULES = g_pConfigManager->getString("input:kb_rules");
|
|
|
|
const auto MODEL = g_pConfigManager->getString("input:kb_model");
|
|
|
|
const auto LAYOUT = g_pConfigManager->getString("input:kb_layout");
|
|
|
|
const auto VARIANT = g_pConfigManager->getString("input:kb_variant");
|
|
|
|
const auto OPTIONS = g_pConfigManager->getString("input:kb_options");
|
|
|
|
|
|
|
|
xkb_rule_names rules = {
|
|
|
|
.rules = RULES.c_str(),
|
|
|
|
.model = MODEL.c_str(),
|
|
|
|
.layout = LAYOUT.c_str(),
|
|
|
|
.variant = VARIANT.c_str(),
|
|
|
|
.options = OPTIONS.c_str()
|
|
|
|
};
|
|
|
|
|
|
|
|
const auto CONTEXT = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
|
|
|
|
const auto KEYMAP = xkb_keymap_new_from_names(CONTEXT, &rules, XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
|
|
|
|
if (!KEYMAP) {
|
|
|
|
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);
|
|
|
|
xkb_context_unref(CONTEXT);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: configure devices one by one
|
|
|
|
for (auto& k : m_lKeyboards)
|
|
|
|
wlr_keyboard_set_keymap(k.keyboard->keyboard, KEYMAP);
|
|
|
|
|
|
|
|
xkb_keymap_unref(KEYMAP);
|
|
|
|
xkb_context_unref(CONTEXT);
|
|
|
|
|
|
|
|
Debug::log(LOG, "Set the keyboard layout to %s and variant to %s", rules.layout, rules.variant);
|
2022-03-17 20:55:04 +01:00
|
|
|
}
|
|
|
|
|
2022-03-18 22:53:27 +01:00
|
|
|
void CInputManager::newMouse(wlr_input_device* mouse) {
|
|
|
|
if (wlr_input_device_is_libinput(mouse)) {
|
|
|
|
const auto LIBINPUTDEV = (libinput_device*)wlr_libinput_get_device_handle(mouse);
|
|
|
|
|
|
|
|
if (libinput_device_config_tap_get_finger_count(LIBINPUTDEV)) // this is for tapping (like on a laptop)
|
|
|
|
libinput_device_config_tap_set_enabled(LIBINPUTDEV, LIBINPUT_CONFIG_TAP_ENABLED);
|
|
|
|
|
|
|
|
if (libinput_device_config_scroll_has_natural_scroll(LIBINPUTDEV))
|
|
|
|
libinput_device_config_scroll_set_natural_scroll_enabled(LIBINPUTDEV, 0 /* Natural */);
|
|
|
|
}
|
|
|
|
|
|
|
|
wlr_cursor_attach_input_device(g_pCompositor->m_sWLRCursor, mouse);
|
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-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
|
|
|
|
|
|
|
m_lKeyboards.remove(*pKeyboard);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CInputManager::destroyMouse(wlr_input_device* mouse) {
|
|
|
|
//
|
|
|
|
}
|
|
|
|
|
2022-03-24 15:57:46 +01:00
|
|
|
void CInputManager::onKeyboardKey(wlr_keyboard_key_event* e, SKeyboard* pKeyboard) {
|
2022-03-18 23:06:45 +01:00
|
|
|
const auto KEYCODE = e->keycode + 8; // Because to xkbcommon it's +8 from libinput
|
2022-03-17 20:55:04 +01:00
|
|
|
|
2022-03-18 23:06:45 +01:00
|
|
|
const xkb_keysym_t* keysyms;
|
|
|
|
int syms = xkb_state_key_get_syms(pKeyboard->keyboard->keyboard->xkb_state, KEYCODE, &keysyms);
|
|
|
|
|
|
|
|
const auto MODS = wlr_keyboard_get_modifiers(pKeyboard->keyboard->keyboard);
|
|
|
|
|
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-03-19 22:03:40 +01:00
|
|
|
bool found = false;
|
2022-03-18 23:06:45 +01:00
|
|
|
if (e->state == WL_KEYBOARD_KEY_STATE_PRESSED) {
|
2022-03-22 22:22:59 +01:00
|
|
|
Debug::log(LOG, "Pressed key %i, with the MODMASK being %i", e->keycode, MODS);
|
|
|
|
|
2022-03-19 17:48:18 +01:00
|
|
|
for (int i = 0; i < syms; ++i)
|
2022-03-19 22:03:40 +01:00
|
|
|
found = g_pKeybindManager->handleKeybinds(MODS, keysyms[i]) || found;
|
2022-03-19 11:27:19 +01:00
|
|
|
} else if (e->state == WL_KEYBOARD_KEY_STATE_RELEASED) {
|
2022-03-19 17:48:18 +01:00
|
|
|
// hee hee
|
2022-03-18 23:06:45 +01:00
|
|
|
}
|
|
|
|
|
2022-03-19 22:03:40 +01:00
|
|
|
if (!found) {
|
2022-03-24 15:57:46 +01:00
|
|
|
wlr_seat_set_keyboard(g_pCompositor->m_sSeat.seat, pKeyboard->keyboard->keyboard);
|
2022-03-22 18:29:13 +01:00
|
|
|
wlr_seat_keyboard_notify_key(g_pCompositor->m_sSeat.seat, e->time_msec, e->keycode, e->state);
|
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-03-24 15:57:46 +01:00
|
|
|
wlr_seat_set_keyboard(g_pCompositor->m_sSeat.seat, pKeyboard->keyboard->keyboard);
|
2022-03-22 18:29:13 +01:00
|
|
|
wlr_seat_keyboard_notify_modifiers(g_pCompositor->m_sSeat.seat, &pKeyboard->keyboard->keyboard->modifiers);
|
2022-03-21 19:28:43 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CInputManager::refocus() {
|
|
|
|
mouseMoveUnified(0);
|
2022-03-31 17:25:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CInputManager::updateDragIcon() {
|
|
|
|
if (!g_pInputManager->m_sDrag.dragIcon)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (g_pInputManager->m_sDrag.dragIcon->drag->grab_type) {
|
|
|
|
case WLR_DRAG_GRAB_KEYBOARD:
|
|
|
|
break;
|
|
|
|
case WLR_DRAG_GRAB_KEYBOARD_POINTER:
|
|
|
|
g_pInputManager->m_sDrag.pos = g_pInputManager->getMouseCoordsInternal();
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2022-03-17 19:03:15 +01:00
|
|
|
}
|