/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2017 Martin Flöser This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . *********************************************************************/ #include "x11_filter.h" #include "effects.h" #include "screenedge.h" #include "tabbox/tabbox.h" #include #include namespace KWin { namespace TabBox { X11Filter::X11Filter() : X11EventFilter(QVector{XCB_KEY_PRESS, XCB_KEY_RELEASE, XCB_MOTION_NOTIFY, XCB_BUTTON_PRESS, XCB_BUTTON_RELEASE}) { } template static bool passToEffects(T *e) { const auto tab = TabBox::TabBox::self(); if (!tab->isShown() && tab->isDisplayed()) { if (effects && static_cast(effects)->checkInputWindowEvent(e)) { return true; } } return false; } bool X11Filter::event(xcb_generic_event_t *event) { if (!TabBox::TabBox::self()->isGrabbed()) { return false; } const uint8_t eventType = event->response_type & ~0x80; switch (eventType) { case XCB_BUTTON_PRESS: case XCB_BUTTON_RELEASE: { auto e = reinterpret_cast(event); xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME); if (passToEffects(e)) { return true; } if (eventType == XCB_BUTTON_PRESS) { return buttonPress(e); } return false; } case XCB_MOTION_NOTIFY: { return motion(event); } case XCB_KEY_PRESS: { keyPress(event); return true; } case XCB_KEY_RELEASE: keyRelease(event); return true; } return false; } bool X11Filter::buttonPress(xcb_button_press_event_t *event) { // press outside Tabbox? const auto tab = TabBox::TabBox::self(); QPoint pos(event->root_x, event->root_y); if ((!tab->isShown() && tab->isDisplayed()) || (!tabBox->containsPos(pos) && (event->detail == XCB_BUTTON_INDEX_1 || event->detail == XCB_BUTTON_INDEX_2 || event->detail == XCB_BUTTON_INDEX_3))) { tab->close(); // click outside closes tab return true; } if (event->detail == XCB_BUTTON_INDEX_5 || event->detail == XCB_BUTTON_INDEX_4) { // mouse wheel event const QModelIndex index = tabBox->nextPrev(event->detail == XCB_BUTTON_INDEX_5); if (index.isValid()) { tab->setCurrentIndex(index); } return true; } return false; } bool X11Filter::motion(xcb_generic_event_t *event) { auto *mouseEvent = reinterpret_cast(event); const QPoint rootPos(mouseEvent->root_x, mouseEvent->root_y); // TODO: this should be in ScreenEdges directly ScreenEdges::self()->check(rootPos, QDateTime::fromMSecsSinceEpoch(xTime()), true); xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME); if (passToEffects(mouseEvent)) { return true; } return false; } void X11Filter::keyPress(xcb_generic_event_t *event) { int keyQt; xcb_key_press_event_t *keyEvent = reinterpret_cast(event); KKeyServer::xcbKeyPressEventToQt(keyEvent, &keyQt); TabBox::TabBox::self()->keyPress(keyQt); } void X11Filter::keyRelease(xcb_generic_event_t *event) { const auto ev = reinterpret_cast(event); unsigned int mk = ev->state & (KKeyServer::modXShift() | KKeyServer::modXCtrl() | KKeyServer::modXAlt() | KKeyServer::modXMeta()); // ev.state is state before the key release, so just checking mk being 0 isn't enough // using XQueryPointer() also doesn't seem to work well, so the check that all // modifiers are released: only one modifier is active and the currently released // key is this modifier - if yes, release the grab int mod_index = -1; for (int i = XCB_MAP_INDEX_SHIFT; i <= XCB_MAP_INDEX_5; ++i) if ((mk & (1 << i)) != 0) { if (mod_index >= 0) return; mod_index = i; } bool release = false; if (mod_index == -1) release = true; else { Xcb::ModifierMapping xmk; if (xmk) { xcb_keycode_t *keycodes = xmk.keycodes(); const int maxIndex = xmk.size(); for (int i = 0; i < xmk->keycodes_per_modifier; ++i) { const int index = xmk->keycodes_per_modifier * mod_index + i; if (index >= maxIndex) { continue; } if (keycodes[index] == ev->detail) { release = true; } } } } if (release) { TabBox::TabBox::self()->modifiersReleased(); } } } }