KWin
Loading...
Searching...
No Matches
x11_filter.cpp
Go to the documentation of this file.
1/*
2 KWin - the KDE window manager
3 This file is part of the KDE project.
4
5 SPDX-FileCopyrightText: 2017 Martin Flöser <mgraesslin@kde.org>
6
7 SPDX-License-Identifier: GPL-2.0-or-later
8*/
9#include "x11_filter.h"
10
12#include "screenedge.h"
13#include "tabbox/tabbox.h"
14#include "utils/xcbutils.h"
15#include "workspace.h"
16
17#include <KKeyServer>
18
19#include <xcb/xcb.h>
20
21namespace KWin
22{
23namespace TabBox
24{
25
27 : X11EventFilter(QList<int>{XCB_KEY_PRESS, XCB_KEY_RELEASE, XCB_MOTION_NOTIFY, XCB_BUTTON_PRESS, XCB_BUTTON_RELEASE})
28{
29}
30
31bool X11Filter::event(xcb_generic_event_t *event)
32{
33 const auto tab = workspace()->tabbox();
34 if (!tab->isGrabbed()) {
35 return false;
36 }
37 const uint8_t eventType = event->response_type & ~0x80;
38 switch (eventType) {
39 case XCB_BUTTON_PRESS:
40 case XCB_BUTTON_RELEASE: {
41 auto e = reinterpret_cast<xcb_button_press_event_t *>(event);
42 xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME);
43 if (!tab->isShown() && tab->isDisplayed()) {
45 // pass on to effects, effects will filter out the event
46 return false;
47 }
48 }
49 if (eventType == XCB_BUTTON_PRESS) {
50 return buttonPress(e);
51 }
52 return false;
53 }
54 case XCB_MOTION_NOTIFY: {
55 motion(event);
56 break;
57 }
58 case XCB_KEY_PRESS: {
59 keyPress(event);
60 return true;
61 }
62 case XCB_KEY_RELEASE:
63 keyRelease(event);
64 return true;
65 }
66 return false;
67}
68bool X11Filter::buttonPress(xcb_button_press_event_t *event)
69{
70 // press outside Tabbox?
71 const auto tab = workspace()->tabbox();
72 QPoint pos(event->root_x, event->root_y);
73 if ((!tab->isShown() && tab->isDisplayed())
74 || (!tabBox->containsPos(pos) && (event->detail == XCB_BUTTON_INDEX_1 || event->detail == XCB_BUTTON_INDEX_2 || event->detail == XCB_BUTTON_INDEX_3))) {
75 tab->close(); // click outside closes tab
76 return true;
77 }
78 if (event->detail == XCB_BUTTON_INDEX_5 || event->detail == XCB_BUTTON_INDEX_4) {
79 // mouse wheel event
80 const QModelIndex index = tabBox->nextPrev(event->detail == XCB_BUTTON_INDEX_5);
81 if (index.isValid()) {
82 tab->setCurrentIndex(index);
83 }
84 return true;
85 }
86 return false;
87}
88
89void X11Filter::motion(xcb_generic_event_t *event)
90{
91 auto *mouseEvent = reinterpret_cast<xcb_motion_notify_event_t *>(event);
92 const QPoint rootPos(mouseEvent->root_x, mouseEvent->root_y);
93 // TODO: this should be in ScreenEdges directly
94 workspace()->screenEdges()->check(rootPos, QDateTime::fromMSecsSinceEpoch(xTime(), Qt::UTC), true);
95 xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME);
96}
97
98void X11Filter::keyPress(xcb_generic_event_t *event)
99{
100 int keyQt;
101 xcb_key_press_event_t *keyEvent = reinterpret_cast<xcb_key_press_event_t *>(event);
102 KKeyServer::xcbKeyPressEventToQt(keyEvent, &keyQt);
103 workspace()->tabbox()->keyPress(keyQt);
104}
105
106void X11Filter::keyRelease(xcb_generic_event_t *event)
107{
108 const auto ev = reinterpret_cast<xcb_key_release_event_t *>(event);
109 unsigned int mk = ev->state & (KKeyServer::modXShift() | KKeyServer::modXCtrl() | KKeyServer::modXAlt() | KKeyServer::modXMeta());
110 // ev.state is state before the key release, so just checking mk being 0 isn't enough
111 // using XQueryPointer() also doesn't seem to work well, so the check that all
112 // modifiers are released: only one modifier is active and the currently released
113 // key is this modifier - if yes, release the grab
114 int mod_index = -1;
115 for (int i = XCB_MAP_INDEX_SHIFT;
116 i <= XCB_MAP_INDEX_5;
117 ++i) {
118 if ((mk & (1 << i)) != 0) {
119 if (mod_index >= 0) {
120 return;
121 }
122 mod_index = i;
123 }
124 }
125 bool release = false;
126 if (mod_index == -1) {
127 release = true;
128 } else {
129 Xcb::ModifierMapping xmk;
130 if (xmk) {
131 xcb_keycode_t *keycodes = xmk.keycodes();
132 const int maxIndex = xmk.size();
133 for (int i = 0; i < xmk->keycodes_per_modifier; ++i) {
134 const int index = xmk->keycodes_per_modifier * mod_index + i;
135 if (index >= maxIndex) {
136 continue;
137 }
138 if (keycodes[index] == ev->detail) {
139 release = true;
140 }
141 }
142 }
143 }
144 if (release) {
145 workspace()->tabbox()->modifiersReleased();
146 }
147}
148
149}
150}
bool isMouseInterception() const
void check(const QPoint &pos, const QDateTime &now, bool forceNoPushBack=false)
bool containsPos(const QPoint &pos) const
QModelIndex nextPrev(bool forward) const
bool event(xcb_generic_event_t *event) override
ScreenEdges * screenEdges() const
TabBoxHandler * tabBox
KWIN_EXPORT xcb_timestamp_t xTime()
Definition xcb.h:29
Workspace * workspace()
Definition workspace.h:830
KWIN_EXPORT xcb_connection_t * connection()
Definition xcb.h:19
EffectsHandler * effects