9b3d9e58a4
Summary: QDateTime::fromMSecSinceEpoch uses Qt::LocalTime by default. This involves an expensive localtime conversion. So instead force things to use UTC, as there is no need for timezone information when tracking durations. This is especially noticeable on Bedrock Linux, which uses a Fuse mounted /etc, which is slower than a plain /etc and causes quite some slowdown there. See https://github.com/bedrocklinux/bedrocklinux-userland/issues/140 for details. Test Plan: The screenedge unit test still passes. Reviewers: #kwin, davidedmundson Reviewed By: #kwin, davidedmundson Subscribers: zzag, anthonyfieroni, davidedmundson, kwin Tags: #kwin Differential Revision: https://phabricator.kde.org/D27114
162 lines
5.3 KiB
C++
162 lines
5.3 KiB
C++
/********************************************************************
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
Copyright (C) 2017 Martin Flöser <mgraesslin@kde.org>
|
|
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*********************************************************************/
|
|
#include "x11_filter.h"
|
|
|
|
#include "effects.h"
|
|
#include "screenedge.h"
|
|
#include "tabbox/tabbox.h"
|
|
|
|
#include <KKeyServer>
|
|
|
|
#include <xcb/xcb.h>
|
|
|
|
namespace KWin
|
|
{
|
|
namespace TabBox
|
|
{
|
|
|
|
X11Filter::X11Filter()
|
|
: X11EventFilter(QVector<int>{XCB_KEY_PRESS, XCB_KEY_RELEASE, XCB_MOTION_NOTIFY, XCB_BUTTON_PRESS, XCB_BUTTON_RELEASE})
|
|
{
|
|
}
|
|
|
|
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<xcb_button_press_event_t*>(event);
|
|
xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME);
|
|
const auto tab = TabBox::TabBox::self();
|
|
if (!tab->isShown() && tab->isDisplayed()) {
|
|
if (effects && static_cast<EffectsHandlerImpl*>(effects)->isMouseInterception()) {
|
|
// pass on to effects, effects will filter out the event
|
|
return false;
|
|
}
|
|
}
|
|
if (eventType == XCB_BUTTON_PRESS) {
|
|
return buttonPress(e);
|
|
}
|
|
return false;
|
|
}
|
|
case XCB_MOTION_NOTIFY: {
|
|
motion(event);
|
|
break;
|
|
}
|
|
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;
|
|
}
|
|
|
|
void X11Filter::motion(xcb_generic_event_t *event)
|
|
{
|
|
auto *mouseEvent = reinterpret_cast<xcb_motion_notify_event_t*>(event);
|
|
const QPoint rootPos(mouseEvent->root_x, mouseEvent->root_y);
|
|
// TODO: this should be in ScreenEdges directly
|
|
ScreenEdges::self()->check(rootPos, QDateTime::fromMSecsSinceEpoch(xTime(), Qt::UTC), true);
|
|
xcb_allow_events(connection(), XCB_ALLOW_ASYNC_POINTER, XCB_CURRENT_TIME);
|
|
}
|
|
|
|
void X11Filter::keyPress(xcb_generic_event_t *event)
|
|
{
|
|
int keyQt;
|
|
xcb_key_press_event_t *keyEvent = reinterpret_cast<xcb_key_press_event_t*>(event);
|
|
KKeyServer::xcbKeyPressEventToQt(keyEvent, &keyQt);
|
|
TabBox::TabBox::self()->keyPress(keyQt);
|
|
}
|
|
|
|
void X11Filter::keyRelease(xcb_generic_event_t *event)
|
|
{
|
|
const auto ev = reinterpret_cast<xcb_key_release_event_t*>(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();
|
|
}
|
|
}
|
|
|
|
}
|
|
}
|