2022-03-19 17:48:18 +01:00
|
|
|
#include "KeybindManager.hpp"
|
|
|
|
|
|
|
|
void CKeybindManager::addKeybind(SKeybind kb) {
|
|
|
|
m_dKeybinds.push_back(kb);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t CKeybindManager::stringToModMask(std::string mods) {
|
|
|
|
uint32_t modMask = 0;
|
|
|
|
if (mods.find("SHIFT") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_SHIFT;
|
|
|
|
if (mods.find("CAPS") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_CAPS;
|
|
|
|
if (mods.find("CTRL") != std::string::npos || mods.find("CONTROL") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_CTRL;
|
|
|
|
if (mods.find("ALT") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_ALT;
|
|
|
|
if (mods.find("MOD2") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_MOD2;
|
|
|
|
if (mods.find("MOD3") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_MOD3;
|
|
|
|
if (mods.find("SUPER") != std::string::npos || mods.find("WIN") != std::string::npos || mods.find("LOGO") != std::string::npos || mods.find("MOD4") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_LOGO;
|
|
|
|
if (mods.find("MOD5") != std::string::npos)
|
|
|
|
modMask |= WLR_MODIFIER_MOD5;
|
|
|
|
|
|
|
|
return modMask;
|
|
|
|
}
|
|
|
|
|
2022-03-19 22:03:40 +01:00
|
|
|
bool CKeybindManager::handleKeybinds(const uint32_t& modmask, const xkb_keysym_t& key) {
|
|
|
|
bool found = false;
|
2022-03-27 19:32:50 +02:00
|
|
|
|
|
|
|
if (handleInternalKeybinds(key))
|
|
|
|
return true;
|
|
|
|
|
2022-03-19 17:48:18 +01:00
|
|
|
for (auto& k : m_dKeybinds) {
|
|
|
|
if (modmask != k.modmask)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// oMg such performance hit!!11!
|
|
|
|
// this little maneouver is gonna cost us 4µs
|
|
|
|
const auto KBKEY = xkb_keysym_from_name(k.key.c_str(), XKB_KEYSYM_CASE_INSENSITIVE);
|
2022-03-23 16:51:48 +01:00
|
|
|
const auto KBKEYUPPER = xkb_keysym_to_upper(KBKEY);
|
2022-04-10 16:47:19 +02:00
|
|
|
// small TODO: fix 0-9 keys and other modified ones with shift
|
2022-03-22 22:22:59 +01:00
|
|
|
|
2022-03-23 16:51:48 +01:00
|
|
|
if (key != KBKEY && key != KBKEYUPPER)
|
2022-03-19 17:48:18 +01:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// yes.
|
|
|
|
if (k.handler == "exec") { spawn(k.arg); }
|
|
|
|
else if (k.handler == "killactive") { killActive(k.arg); }
|
2022-03-20 11:14:24 +01:00
|
|
|
else if (k.handler == "togglefloating") { toggleActiveFloating(k.arg); }
|
2022-03-20 15:55:47 +01:00
|
|
|
else if (k.handler == "workspace") { changeworkspace(k.arg); }
|
2022-03-21 19:18:33 +01:00
|
|
|
else if (k.handler == "fullscreen") { fullscreenActive(k.arg); }
|
2022-03-23 16:51:48 +01:00
|
|
|
else if (k.handler == "movetoworkspace") { moveActiveToWorkspace(k.arg); }
|
2022-04-02 20:04:32 +02:00
|
|
|
else if (k.handler == "pseudo") { toggleActivePseudo(k.arg); }
|
2022-04-09 13:26:55 +02:00
|
|
|
else if (k.handler == "movefocus") { moveFocusTo(k.arg); }
|
2022-03-19 22:03:40 +01:00
|
|
|
|
|
|
|
found = true;
|
2022-03-19 17:48:18 +01:00
|
|
|
}
|
2022-03-19 22:03:40 +01:00
|
|
|
|
|
|
|
return found;
|
2022-03-19 17:48:18 +01:00
|
|
|
}
|
|
|
|
|
2022-03-27 19:32:50 +02:00
|
|
|
bool CKeybindManager::handleInternalKeybinds(xkb_keysym_t keysym) {
|
|
|
|
// Handles the CTRL+ALT+FX TTY keybinds
|
|
|
|
if (!(keysym >= XKB_KEY_XF86Switch_VT_1 && keysym <= XKB_KEY_XF86Switch_VT_12))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const auto PSESSION = wlr_backend_get_session(g_pCompositor->m_sWLRBackend);
|
|
|
|
if (PSESSION) {
|
|
|
|
const auto TTY = keysym - XKB_KEY_XF86Switch_VT_1 + 1;
|
|
|
|
wlr_session_change_vt(PSESSION, TTY);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-03-19 17:48:18 +01:00
|
|
|
// Dispatchers
|
|
|
|
|
|
|
|
void CKeybindManager::spawn(std::string args) {
|
|
|
|
args = "WAYLAND_DISPLAY=" + std::string(g_pCompositor->m_szWLDisplaySocket) + " DISPLAY=" + std::string(g_pXWaylandManager->m_sWLRXWayland->display_name) + " " + args;
|
|
|
|
Debug::log(LOG, "Executing %s", args.c_str());
|
|
|
|
if (fork() == 0) {
|
|
|
|
execl("/bin/sh", "/bin/sh", "-c", args.c_str(), nullptr);
|
|
|
|
|
|
|
|
_exit(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CKeybindManager::killActive(std::string args) {
|
2022-04-02 18:57:09 +02:00
|
|
|
if (g_pCompositor->m_pLastWindow && g_pCompositor->windowValidMapped(g_pCompositor->m_pLastWindow)) {
|
|
|
|
g_pXWaylandManager->sendCloseWindow(g_pCompositor->m_pLastWindow);
|
|
|
|
g_pCompositor->m_pLastFocus = nullptr;
|
|
|
|
g_pCompositor->m_pLastWindow = nullptr;
|
|
|
|
}
|
|
|
|
|
2022-03-20 11:14:24 +01:00
|
|
|
g_pCompositor->focusWindow(g_pCompositor->windowFromCursor());
|
2022-03-19 21:48:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CKeybindManager::clearKeybinds() {
|
|
|
|
m_dKeybinds.clear();
|
2022-03-20 11:14:24 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CKeybindManager::toggleActiveFloating(std::string args) {
|
2022-04-02 18:57:09 +02:00
|
|
|
const auto ACTIVEWINDOW = g_pCompositor->m_pLastWindow;
|
2022-03-20 11:14:24 +01:00
|
|
|
|
|
|
|
if (g_pCompositor->windowValidMapped(ACTIVEWINDOW)) {
|
|
|
|
ACTIVEWINDOW->m_bIsFloating = !ACTIVEWINDOW->m_bIsFloating;
|
2022-03-20 13:37:07 +01:00
|
|
|
|
|
|
|
ACTIVEWINDOW->m_vRealPosition = ACTIVEWINDOW->m_vRealPosition + Vector2D(5, 5);
|
|
|
|
ACTIVEWINDOW->m_vSize = ACTIVEWINDOW->m_vRealPosition - Vector2D(10, 10);
|
|
|
|
|
2022-03-20 11:14:24 +01:00
|
|
|
g_pLayoutManager->getCurrentLayout()->changeWindowFloatingMode(ACTIVEWINDOW);
|
|
|
|
}
|
2022-03-20 15:55:47 +01:00
|
|
|
}
|
|
|
|
|
2022-04-02 20:04:32 +02:00
|
|
|
void CKeybindManager::toggleActivePseudo(std::string args) {
|
|
|
|
const auto ACTIVEWINDOW = g_pCompositor->m_pLastWindow;
|
|
|
|
|
|
|
|
if (!g_pCompositor->windowValidMapped(ACTIVEWINDOW))
|
|
|
|
return;
|
|
|
|
|
|
|
|
ACTIVEWINDOW->m_bIsPseudotiled = !ACTIVEWINDOW->m_bIsPseudotiled;
|
|
|
|
|
|
|
|
g_pLayoutManager->getCurrentLayout()->recalculateWindow(ACTIVEWINDOW);
|
|
|
|
}
|
|
|
|
|
2022-03-20 15:55:47 +01:00
|
|
|
void CKeybindManager::changeworkspace(std::string args) {
|
|
|
|
int workspaceToChangeTo = 0;
|
|
|
|
try {
|
|
|
|
workspaceToChangeTo = stoi(args);
|
|
|
|
} catch (...) {
|
|
|
|
Debug::log(ERR, "Invalid arg \"%s\" in changeWorkspace!", args.c_str());
|
|
|
|
}
|
|
|
|
|
|
|
|
// if it exists, we warp to it
|
|
|
|
if (g_pCompositor->getWorkspaceByID(workspaceToChangeTo)) {
|
|
|
|
const auto PMONITOR = g_pCompositor->getMonitorFromID(g_pCompositor->getWorkspaceByID(workspaceToChangeTo)->monitorID);
|
|
|
|
|
|
|
|
// if it's not visible, make it visible.
|
2022-03-20 19:14:17 +01:00
|
|
|
if (!g_pCompositor->isWorkspaceVisible(workspaceToChangeTo)) {
|
|
|
|
const auto OLDWORKSPACEID = PMONITOR->activeWorkspace;
|
|
|
|
|
|
|
|
// change it
|
2022-03-20 15:55:47 +01:00
|
|
|
PMONITOR->activeWorkspace = workspaceToChangeTo;
|
|
|
|
|
2022-03-20 19:14:17 +01:00
|
|
|
// we need to move XWayland windows to narnia or otherwise they will still process our cursor and shit
|
|
|
|
// and that'd be annoying as hell
|
|
|
|
g_pCompositor->fixXWaylandWindowsOnWorkspace(OLDWORKSPACEID);
|
|
|
|
|
|
|
|
// and fix on the new workspace
|
|
|
|
g_pCompositor->fixXWaylandWindowsOnWorkspace(PMONITOR->activeWorkspace);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-03-20 15:55:47 +01:00
|
|
|
// If the monitor is not the one our cursor's at, warp to it.
|
|
|
|
if (PMONITOR != g_pCompositor->getMonitorFromCursor()) {
|
|
|
|
Vector2D middle = PMONITOR->vecPosition + PMONITOR->vecSize / 2.f;
|
|
|
|
wlr_cursor_warp(g_pCompositor->m_sWLRCursor, nullptr, middle.x, middle.y);
|
|
|
|
}
|
|
|
|
|
|
|
|
// focus the first window
|
|
|
|
g_pCompositor->focusWindow(g_pCompositor->getFirstWindowOnWorkspace(workspaceToChangeTo));
|
2022-03-20 19:14:17 +01:00
|
|
|
|
2022-03-20 15:55:47 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Workspace doesn't exist, create and switch
|
|
|
|
const auto PMONITOR = g_pCompositor->getMonitorFromCursor();
|
|
|
|
|
2022-03-20 19:28:57 +01:00
|
|
|
const auto OLDWORKSPACE = PMONITOR->activeWorkspace;
|
2022-03-20 19:27:47 +01:00
|
|
|
|
2022-03-20 15:55:47 +01:00
|
|
|
g_pCompositor->m_lWorkspaces.push_back(SWorkspace());
|
|
|
|
const auto PWORKSPACE = &g_pCompositor->m_lWorkspaces.back();
|
|
|
|
|
|
|
|
PWORKSPACE->ID = workspaceToChangeTo;
|
|
|
|
PWORKSPACE->monitorID = PMONITOR->ID;
|
|
|
|
|
|
|
|
PMONITOR->activeWorkspace = workspaceToChangeTo;
|
2022-03-20 19:28:57 +01:00
|
|
|
|
|
|
|
// we need to move XWayland windows to narnia or otherwise they will still process our cursor and shit
|
|
|
|
// and that'd be annoying as hell
|
2022-03-20 19:29:50 +01:00
|
|
|
g_pCompositor->fixXWaylandWindowsOnWorkspace(OLDWORKSPACE);
|
2022-03-21 19:18:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CKeybindManager::fullscreenActive(std::string args) {
|
2022-04-02 18:57:09 +02:00
|
|
|
const auto PWINDOW = g_pCompositor->m_pLastWindow;
|
2022-03-21 19:18:33 +01:00
|
|
|
|
|
|
|
if (!g_pCompositor->windowValidMapped(PWINDOW))
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_pLayoutManager->getCurrentLayout()->fullscreenRequestForWindow(PWINDOW);
|
2022-03-30 17:39:04 +02:00
|
|
|
|
|
|
|
g_pXWaylandManager->setWindowFullscreen(PWINDOW, PWINDOW->m_bIsFullscreen);
|
2022-03-30 20:16:23 +02:00
|
|
|
|
|
|
|
// make all windows on the same workspace under the fullscreen window
|
|
|
|
for (auto& w : g_pCompositor->m_lWindows) {
|
|
|
|
if (w.m_iWorkspaceID == PWINDOW->m_iWorkspaceID)
|
|
|
|
w.m_bCreatedOverFullscreen = false;
|
|
|
|
}
|
2022-03-23 16:51:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void CKeybindManager::moveActiveToWorkspace(std::string args) {
|
2022-04-02 18:57:09 +02:00
|
|
|
const auto PWINDOW = g_pCompositor->m_pLastWindow;
|
2022-03-23 16:51:48 +01:00
|
|
|
|
|
|
|
if (!g_pCompositor->windowValidMapped(PWINDOW))
|
|
|
|
return;
|
|
|
|
|
|
|
|
int workspaceID;
|
|
|
|
try {
|
|
|
|
workspaceID = stoi(args);
|
|
|
|
} catch( ... ) {
|
|
|
|
Debug::log(ERR, "Invalid movetoworkspace: %s", args.c_str());
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_pLayoutManager->getCurrentLayout()->onWindowRemoved(PWINDOW);
|
|
|
|
|
|
|
|
const auto OLDWORKSPACE = g_pCompositor->getWorkspaceByID(PWINDOW->m_iWorkspaceID);
|
|
|
|
|
|
|
|
// hack
|
|
|
|
g_pKeybindManager->changeworkspace(std::to_string(workspaceID));
|
|
|
|
|
|
|
|
const auto NEWWORKSPACE = g_pCompositor->getWorkspaceByID(workspaceID);
|
|
|
|
|
|
|
|
OLDWORKSPACE->hasFullscreenWindow = false;
|
|
|
|
|
|
|
|
PWINDOW->m_iWorkspaceID = workspaceID;
|
|
|
|
PWINDOW->m_iMonitorID = NEWWORKSPACE->monitorID;
|
|
|
|
PWINDOW->m_bIsFullscreen = false;
|
|
|
|
|
|
|
|
if (NEWWORKSPACE->hasFullscreenWindow) {
|
|
|
|
g_pCompositor->getFullscreenWindowOnWorkspace(workspaceID)->m_bIsFullscreen = false;
|
|
|
|
NEWWORKSPACE->hasFullscreenWindow = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Hack: So that the layout doesnt find our window at the cursor
|
|
|
|
PWINDOW->m_vPosition = Vector2D(-42069, -42069);
|
2022-04-10 21:45:24 +02:00
|
|
|
|
|
|
|
// Save the real position and size because the layout might set its own
|
|
|
|
const auto PSAVEDSIZE = PWINDOW->m_vRealSize;
|
|
|
|
const auto PSAVEDPOS = PWINDOW->m_vRealPosition;
|
2022-03-23 16:51:48 +01:00
|
|
|
g_pLayoutManager->getCurrentLayout()->onWindowCreated(PWINDOW);
|
2022-04-10 21:45:24 +02:00
|
|
|
// and restore it
|
|
|
|
PWINDOW->m_vRealPosition = PSAVEDPOS;
|
|
|
|
PWINDOW->m_vRealSize = PSAVEDSIZE;
|
2022-03-23 16:51:48 +01:00
|
|
|
|
|
|
|
if (PWINDOW->m_bIsFloating) {
|
|
|
|
PWINDOW->m_vRealPosition = PWINDOW->m_vRealPosition - g_pCompositor->getMonitorFromID(OLDWORKSPACE->monitorID)->vecPosition;
|
|
|
|
PWINDOW->m_vRealPosition = PWINDOW->m_vRealPosition + g_pCompositor->getMonitorFromID(NEWWORKSPACE->monitorID)->vecPosition;
|
2022-04-10 21:45:24 +02:00
|
|
|
PWINDOW->m_vEffectivePosition = PWINDOW->m_vRealPosition;
|
|
|
|
PWINDOW->m_vPosition = PWINDOW->m_vRealPosition;
|
2022-03-23 16:51:48 +01:00
|
|
|
}
|
2022-03-27 19:32:50 +02:00
|
|
|
}
|
2022-04-09 13:26:55 +02:00
|
|
|
|
|
|
|
void CKeybindManager::moveFocusTo(std::string args) {
|
|
|
|
char arg = args[0];
|
|
|
|
|
|
|
|
if (arg != 'l' && arg != 'r' && arg != 'u' && arg != 'd' && arg != 't' && arg != 'b') {
|
|
|
|
Debug::log(ERR, "Cannot move window in direction %c, unsupported direction. Supported: l,r,u/t,d/b", arg);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const auto PLASTWINDOW = g_pCompositor->m_pLastWindow;
|
|
|
|
|
|
|
|
if (!g_pCompositor->windowValidMapped(PLASTWINDOW))
|
|
|
|
return;
|
|
|
|
|
|
|
|
const auto PWINDOWTOCHANGETO = g_pCompositor->getWindowInDirection(PLASTWINDOW, arg);
|
|
|
|
|
2022-04-09 13:33:44 +02:00
|
|
|
if (PWINDOWTOCHANGETO) {
|
2022-04-09 13:26:55 +02:00
|
|
|
g_pCompositor->focusWindow(PWINDOWTOCHANGETO);
|
2022-04-09 13:33:44 +02:00
|
|
|
Vector2D middle = PWINDOWTOCHANGETO->m_vPosition + PWINDOWTOCHANGETO->m_vSize / 2.f;
|
|
|
|
wlr_cursor_warp(g_pCompositor->m_sWLRCursor, nullptr, middle.x, middle.y);
|
|
|
|
}
|
2022-04-09 13:26:55 +02:00
|
|
|
}
|