1e9b961761
When the tabbox grabs input, the key events won't be sent to the client. It's not good for a couple of reasons: first, it can fool the client into repeating a previously pressed key; and the server side and the client side can be out of sync after the task switcher is dismissed. In order to handle that properly, this change makes the tabbox reset the keyboard focus when it's shown or hidden. When the task switcher is dismissed, an enter event will be sent with the current state of the pressed keys. Ideally, the same needs to be done with the pointer focus but it's challenging to achieve with the current input pipeline design. An input grab abstraction is needed to handle pointer focus when the task switcher is active.
295 lines
11 KiB
C++
295 lines
11 KiB
C++
/*
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
SPDX-FileCopyrightText: 2016 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
#include "kwin_wayland_test.h"
|
|
|
|
#include "input.h"
|
|
#include "pointer_input.h"
|
|
#include "tabbox/tabbox.h"
|
|
#include "wayland_server.h"
|
|
#include "window.h"
|
|
#include "workspace.h"
|
|
|
|
#include <KConfigGroup>
|
|
#include <KWayland/Client/keyboard.h>
|
|
#include <KWayland/Client/seat.h>
|
|
#include <KWayland/Client/surface.h>
|
|
|
|
#include <linux/input.h>
|
|
|
|
using namespace KWin;
|
|
|
|
static const QString s_socketName = QStringLiteral("wayland_test_kwin_tabbox-0");
|
|
|
|
class TabBoxTest : public QObject
|
|
{
|
|
Q_OBJECT
|
|
private Q_SLOTS:
|
|
void initTestCase();
|
|
void init();
|
|
void cleanup();
|
|
|
|
void testMoveForward();
|
|
void testMoveBackward();
|
|
void testCapsLock();
|
|
void testKeyboardFocus();
|
|
};
|
|
|
|
void TabBoxTest::initTestCase()
|
|
{
|
|
qRegisterMetaType<KWin::Window *>();
|
|
QSignalSpy applicationStartedSpy(kwinApp(), &Application::started);
|
|
QVERIFY(waylandServer()->init(s_socketName));
|
|
Test::setOutputConfig({
|
|
QRect(0, 0, 1280, 1024),
|
|
QRect(1280, 0, 1280, 1024),
|
|
});
|
|
|
|
KSharedConfigPtr c = KSharedConfig::openConfig(QString(), KConfig::SimpleConfig);
|
|
c->group(QStringLiteral("TabBox")).writeEntry("ShowTabBox", false);
|
|
c->sync();
|
|
kwinApp()->setConfig(c);
|
|
qputenv("KWIN_XKB_DEFAULT_KEYMAP", "1");
|
|
|
|
kwinApp()->start();
|
|
QVERIFY(applicationStartedSpy.wait());
|
|
}
|
|
|
|
void TabBoxTest::init()
|
|
{
|
|
QVERIFY(Test::setupWaylandConnection(Test::AdditionalWaylandInterface::Seat));
|
|
workspace()->setActiveOutput(QPoint(640, 512));
|
|
KWin::input()->pointer()->warp(QPoint(640, 512));
|
|
}
|
|
|
|
void TabBoxTest::cleanup()
|
|
{
|
|
Test::destroyWaylandConnection();
|
|
}
|
|
|
|
void TabBoxTest::testCapsLock()
|
|
{
|
|
#if !KWIN_BUILD_GLOBALSHORTCUTS
|
|
QSKIP("Can't test shortcuts without shortcuts");
|
|
return;
|
|
#endif
|
|
|
|
// this test verifies that Alt+tab works correctly also when Capslock is on
|
|
// bug 368590
|
|
|
|
// first create three windows
|
|
std::unique_ptr<KWayland::Client::Surface> surface1(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface1(Test::createXdgToplevelSurface(surface1.get()));
|
|
auto c1 = Test::renderAndWaitForShown(surface1.get(), QSize(100, 50), Qt::blue);
|
|
QVERIFY(c1);
|
|
QVERIFY(c1->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface2(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface2(Test::createXdgToplevelSurface(surface2.get()));
|
|
auto c2 = Test::renderAndWaitForShown(surface2.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c2);
|
|
QVERIFY(c2->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface3(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface3(Test::createXdgToplevelSurface(surface3.get()));
|
|
auto c3 = Test::renderAndWaitForShown(surface3.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c3);
|
|
QVERIFY(c3->isActive());
|
|
|
|
// Setup tabbox signal spies
|
|
QSignalSpy tabboxAddedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxAdded);
|
|
QSignalSpy tabboxClosedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxClosed);
|
|
|
|
// enable capslock
|
|
quint32 timestamp = 0;
|
|
Test::keyboardKeyPressed(KEY_CAPSLOCK, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_CAPSLOCK, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::NoModifier);
|
|
|
|
// press alt+tab
|
|
Test::keyboardKeyPressed(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::AltModifier);
|
|
Test::keyboardKeyPressed(KEY_TAB, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_TAB, timestamp++);
|
|
|
|
QVERIFY(tabboxAddedSpy.wait());
|
|
QVERIFY(workspace()->tabbox()->isGrabbed());
|
|
|
|
// release alt
|
|
Test::keyboardKeyReleased(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(tabboxClosedSpy.count(), 1);
|
|
QCOMPARE(workspace()->tabbox()->isGrabbed(), false);
|
|
|
|
// release caps lock
|
|
Test::keyboardKeyPressed(KEY_CAPSLOCK, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_CAPSLOCK, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::NoModifier);
|
|
QCOMPARE(tabboxClosedSpy.count(), 1);
|
|
QCOMPARE(workspace()->tabbox()->isGrabbed(), false);
|
|
QCOMPARE(workspace()->activeWindow(), c2);
|
|
|
|
surface3.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c3));
|
|
surface2.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c2));
|
|
surface1.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c1));
|
|
}
|
|
|
|
void TabBoxTest::testMoveForward()
|
|
{
|
|
#if !KWIN_BUILD_GLOBALSHORTCUTS
|
|
QSKIP("Can't test shortcuts without shortcuts");
|
|
return;
|
|
#endif
|
|
|
|
// this test verifies that Alt+tab works correctly moving forward
|
|
|
|
// first create three windows
|
|
std::unique_ptr<KWayland::Client::Surface> surface1(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface1(Test::createXdgToplevelSurface(surface1.get()));
|
|
auto c1 = Test::renderAndWaitForShown(surface1.get(), QSize(100, 50), Qt::blue);
|
|
QVERIFY(c1);
|
|
QVERIFY(c1->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface2(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface2(Test::createXdgToplevelSurface(surface2.get()));
|
|
auto c2 = Test::renderAndWaitForShown(surface2.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c2);
|
|
QVERIFY(c2->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface3(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface3(Test::createXdgToplevelSurface(surface3.get()));
|
|
auto c3 = Test::renderAndWaitForShown(surface3.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c3);
|
|
QVERIFY(c3->isActive());
|
|
|
|
// Setup tabbox signal spies
|
|
QSignalSpy tabboxAddedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxAdded);
|
|
QSignalSpy tabboxClosedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxClosed);
|
|
|
|
// press alt+tab
|
|
quint32 timestamp = 0;
|
|
Test::keyboardKeyPressed(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::AltModifier);
|
|
Test::keyboardKeyPressed(KEY_TAB, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_TAB, timestamp++);
|
|
|
|
QVERIFY(tabboxAddedSpy.wait());
|
|
QVERIFY(workspace()->tabbox()->isGrabbed());
|
|
|
|
// release alt
|
|
Test::keyboardKeyReleased(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(tabboxClosedSpy.count(), 1);
|
|
QCOMPARE(workspace()->tabbox()->isGrabbed(), false);
|
|
QCOMPARE(workspace()->activeWindow(), c2);
|
|
|
|
surface3.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c3));
|
|
surface2.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c2));
|
|
surface1.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c1));
|
|
}
|
|
|
|
void TabBoxTest::testMoveBackward()
|
|
{
|
|
#if !KWIN_BUILD_GLOBALSHORTCUTS
|
|
QSKIP("Can't test shortcuts without shortcuts");
|
|
return;
|
|
#endif
|
|
|
|
// this test verifies that Alt+Shift+tab works correctly moving backward
|
|
|
|
// first create three windows
|
|
std::unique_ptr<KWayland::Client::Surface> surface1(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface1(Test::createXdgToplevelSurface(surface1.get()));
|
|
auto c1 = Test::renderAndWaitForShown(surface1.get(), QSize(100, 50), Qt::blue);
|
|
QVERIFY(c1);
|
|
QVERIFY(c1->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface2(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface2(Test::createXdgToplevelSurface(surface2.get()));
|
|
auto c2 = Test::renderAndWaitForShown(surface2.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c2);
|
|
QVERIFY(c2->isActive());
|
|
std::unique_ptr<KWayland::Client::Surface> surface3(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface3(Test::createXdgToplevelSurface(surface3.get()));
|
|
auto c3 = Test::renderAndWaitForShown(surface3.get(), QSize(100, 50), Qt::red);
|
|
QVERIFY(c3);
|
|
QVERIFY(c3->isActive());
|
|
|
|
// Setup tabbox signal spies
|
|
QSignalSpy tabboxAddedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxAdded);
|
|
QSignalSpy tabboxClosedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxClosed);
|
|
|
|
// press alt+shift+tab
|
|
quint32 timestamp = 0;
|
|
Test::keyboardKeyPressed(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::AltModifier);
|
|
Test::keyboardKeyPressed(KEY_LEFTSHIFT, timestamp++);
|
|
QCOMPARE(input()->keyboardModifiers(), Qt::AltModifier | Qt::ShiftModifier);
|
|
Test::keyboardKeyPressed(KEY_TAB, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_TAB, timestamp++);
|
|
|
|
QVERIFY(tabboxAddedSpy.wait());
|
|
QVERIFY(workspace()->tabbox()->isGrabbed());
|
|
|
|
// release alt
|
|
Test::keyboardKeyReleased(KEY_LEFTSHIFT, timestamp++);
|
|
QCOMPARE(tabboxClosedSpy.count(), 0);
|
|
Test::keyboardKeyReleased(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(tabboxClosedSpy.count(), 1);
|
|
QCOMPARE(workspace()->tabbox()->isGrabbed(), false);
|
|
QCOMPARE(workspace()->activeWindow(), c1);
|
|
|
|
surface3.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c3));
|
|
surface2.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c2));
|
|
surface1.reset();
|
|
QVERIFY(Test::waitForWindowClosed(c1));
|
|
}
|
|
|
|
void TabBoxTest::testKeyboardFocus()
|
|
{
|
|
// This test verifies that the keyboard focus will be withdrawn from the currently activated
|
|
// window when the task switcher is active and restored once the task switcher is dismissed.
|
|
|
|
QVERIFY(Test::waitForWaylandKeyboard());
|
|
|
|
std::unique_ptr<KWayland::Client::Keyboard> keyboard(Test::waylandSeat()->createKeyboard());
|
|
QSignalSpy enteredSpy(keyboard.get(), &KWayland::Client::Keyboard::entered);
|
|
QSignalSpy leftSpy(keyboard.get(), &KWayland::Client::Keyboard::left);
|
|
|
|
// add a window
|
|
std::unique_ptr<KWayland::Client::Surface> surface(Test::createSurface());
|
|
std::unique_ptr<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.get()));
|
|
Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue);
|
|
|
|
// the keyboard focus will be moved to the surface after it's mapped
|
|
QVERIFY(enteredSpy.wait());
|
|
|
|
QSignalSpy tabboxAddedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxAdded);
|
|
QSignalSpy tabboxClosedSpy(workspace()->tabbox(), &TabBox::TabBox::tabBoxClosed);
|
|
|
|
// press alt+tab
|
|
quint32 timestamp = 0;
|
|
Test::keyboardKeyPressed(KEY_LEFTALT, timestamp++);
|
|
Test::keyboardKeyPressed(KEY_TAB, timestamp++);
|
|
Test::keyboardKeyReleased(KEY_TAB, timestamp++);
|
|
QVERIFY(tabboxAddedSpy.wait());
|
|
|
|
// the surface should have no keyboard focus anymore because tabbox grabs input
|
|
QCOMPARE(leftSpy.count(), 1);
|
|
|
|
// release alt
|
|
Test::keyboardKeyReleased(KEY_LEFTALT, timestamp++);
|
|
QCOMPARE(tabboxClosedSpy.count(), 1);
|
|
|
|
// the surface should regain keyboard focus after the tabbox is dismissed
|
|
QVERIFY(enteredSpy.wait());
|
|
}
|
|
|
|
WAYLANDTEST_MAIN(TabBoxTest)
|
|
#include "tabbox_test.moc"
|