2020-09-01 08:58:46 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
SPDX-FileCopyrightText: 2020 Marco Martin <mart@kde.org>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
#include "kwin_wayland_test.h"
|
2022-03-23 10:13:38 +00:00
|
|
|
|
2020-09-01 08:58:46 +00:00
|
|
|
#include "abstract_client.h"
|
2021-08-31 07:03:05 +00:00
|
|
|
#include "abstract_output.h"
|
2020-09-01 08:58:46 +00:00
|
|
|
#include "cursor.h"
|
|
|
|
#include "deleted.h"
|
2022-03-23 10:13:38 +00:00
|
|
|
#include "effects.h"
|
2020-09-29 14:46:32 +00:00
|
|
|
#include "inputmethod.h"
|
2022-03-23 10:13:38 +00:00
|
|
|
#include "platform.h"
|
2020-09-01 08:58:46 +00:00
|
|
|
#include "qwayland-input-method-unstable-v1.h"
|
2020-09-25 06:51:04 +00:00
|
|
|
#include "qwayland-text-input-unstable-v3.h"
|
2022-03-23 10:13:38 +00:00
|
|
|
#include "virtualkeyboard_dbus.h"
|
|
|
|
#include "wayland_server.h"
|
|
|
|
#include "workspace.h"
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
#include <KWaylandServer/clientconnection.h>
|
|
|
|
#include <KWaylandServer/display.h>
|
2021-07-20 23:41:38 +00:00
|
|
|
#include <KWaylandServer/seat_interface.h>
|
2020-09-01 08:58:46 +00:00
|
|
|
#include <KWaylandServer/surface_interface.h>
|
2022-03-23 10:13:38 +00:00
|
|
|
#include <QDBusConnection>
|
|
|
|
#include <QDBusMessage>
|
|
|
|
#include <QDBusPendingReply>
|
|
|
|
#include <QSignalSpy>
|
|
|
|
#include <QTest>
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
#include <KWayland/Client/compositor.h>
|
2021-12-29 17:19:43 +00:00
|
|
|
#include <KWayland/Client/keyboard.h>
|
2020-09-01 08:58:46 +00:00
|
|
|
#include <KWayland/Client/output.h>
|
2021-07-20 23:41:38 +00:00
|
|
|
#include <KWayland/Client/region.h>
|
2021-12-29 17:19:43 +00:00
|
|
|
#include <KWayland/Client/seat.h>
|
2020-09-01 08:58:46 +00:00
|
|
|
#include <KWayland/Client/surface.h>
|
|
|
|
#include <KWayland/Client/textinput.h>
|
2021-12-29 17:19:43 +00:00
|
|
|
#include <linux/input-event-codes.h>
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
using namespace KWin;
|
|
|
|
using namespace KWayland::Client;
|
|
|
|
using KWin::VirtualKeyboardDBus;
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
static const QString s_socketName = QStringLiteral("wayland_test_kwin_inputmethod-0");
|
2020-09-01 08:58:46 +00:00
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
class InputMethodTest : public QObject
|
2020-09-01 08:58:46 +00:00
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
private Q_SLOTS:
|
|
|
|
void initTestCase();
|
|
|
|
void init();
|
|
|
|
void cleanup();
|
|
|
|
|
|
|
|
void testOpenClose();
|
2020-09-25 06:51:04 +00:00
|
|
|
void testEnableDisableV3();
|
2021-07-20 23:41:38 +00:00
|
|
|
void testEnableActive();
|
2021-07-21 00:04:33 +00:00
|
|
|
void testHidePanel();
|
2021-07-30 18:29:01 +00:00
|
|
|
void testSwitchFocusedSurfaces();
|
2021-12-23 08:42:00 +00:00
|
|
|
void testV3Styling();
|
2021-12-23 22:33:30 +00:00
|
|
|
void testDisableShowInputPanel();
|
2021-12-29 17:19:43 +00:00
|
|
|
void testModifierForwarding();
|
2021-10-18 13:41:05 +00:00
|
|
|
|
|
|
|
private:
|
2022-03-23 10:13:38 +00:00
|
|
|
void touchNow()
|
|
|
|
{
|
2021-10-18 13:41:05 +00:00
|
|
|
static int time = 0;
|
2022-03-10 10:27:35 +00:00
|
|
|
Test::touchDown(0, {100, 100}, ++time);
|
|
|
|
Test::touchUp(0, ++time);
|
2021-10-18 13:41:05 +00:00
|
|
|
}
|
2020-09-01 08:58:46 +00:00
|
|
|
};
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
void InputMethodTest::initTestCase()
|
2020-09-01 08:58:46 +00:00
|
|
|
{
|
2021-02-22 16:03:16 +00:00
|
|
|
QDBusConnection::sessionBus().registerService(QStringLiteral("org.kde.kwin.testvirtualkeyboard"));
|
|
|
|
|
2020-09-01 08:58:46 +00:00
|
|
|
qRegisterMetaType<KWin::Deleted *>();
|
|
|
|
qRegisterMetaType<KWin::AbstractClient *>();
|
|
|
|
qRegisterMetaType<KWayland::Client::Output *>();
|
|
|
|
|
|
|
|
QSignalSpy applicationStartedSpy(kwinApp(), &Application::started);
|
|
|
|
QVERIFY(applicationStartedSpy.isValid());
|
|
|
|
kwinApp()->platform()->setInitialWindowSize(QSize(1280, 1024));
|
2020-12-09 13:06:15 +00:00
|
|
|
QVERIFY(waylandServer()->init(s_socketName));
|
2020-09-01 08:58:46 +00:00
|
|
|
QMetaObject::invokeMethod(kwinApp()->platform(), "setVirtualOutputs", Qt::DirectConnection, Q_ARG(int, 2));
|
|
|
|
|
2021-07-21 11:10:06 +00:00
|
|
|
static_cast<WaylandTestApplication *>(kwinApp())->setInputMethodServerToStart("internal");
|
2020-09-01 08:58:46 +00:00
|
|
|
kwinApp()->start();
|
|
|
|
QVERIFY(applicationStartedSpy.wait());
|
2021-08-31 07:03:05 +00:00
|
|
|
const auto outputs = kwinApp()->platform()->enabledOutputs();
|
|
|
|
QCOMPARE(outputs.count(), 2);
|
|
|
|
QCOMPARE(outputs[0]->geometry(), QRect(0, 0, 1280, 1024));
|
|
|
|
QCOMPARE(outputs[1]->geometry(), QRect(1280, 0, 1280, 1024));
|
2021-05-08 00:08:22 +00:00
|
|
|
Test::initWaylandWorkspace();
|
2020-09-01 08:58:46 +00:00
|
|
|
}
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
void InputMethodTest::init()
|
2020-09-01 08:58:46 +00:00
|
|
|
{
|
2021-10-18 13:41:05 +00:00
|
|
|
touchNow();
|
2022-03-23 10:13:38 +00:00
|
|
|
QVERIFY(Test::setupWaylandConnection(Test::AdditionalWaylandInterface::Seat | Test::AdditionalWaylandInterface::TextInputManagerV2 | Test::AdditionalWaylandInterface::InputMethodV1 | Test::AdditionalWaylandInterface::TextInputManagerV3));
|
2020-09-01 08:58:46 +00:00
|
|
|
|
2021-08-28 18:58:29 +00:00
|
|
|
workspace()->setActiveOutput(QPoint(640, 512));
|
2021-08-28 18:40:12 +00:00
|
|
|
KWin::Cursors::self()->mouse()->setPos(QPoint(640, 512));
|
2020-11-27 19:57:24 +00:00
|
|
|
|
2021-04-23 19:08:29 +00:00
|
|
|
InputMethod::self()->setEnabled(true);
|
2020-09-01 08:58:46 +00:00
|
|
|
}
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
void InputMethodTest::cleanup()
|
2020-09-01 08:58:46 +00:00
|
|
|
{
|
|
|
|
Test::destroyWaylandConnection();
|
|
|
|
}
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
void InputMethodTest::testOpenClose()
|
2020-09-01 08:58:46 +00:00
|
|
|
{
|
|
|
|
QSignalSpy clientAddedSpy(workspace(), &Workspace::clientAdded);
|
|
|
|
QSignalSpy clientRemovedSpy(workspace(), &Workspace::clientRemoved);
|
|
|
|
QVERIFY(clientAddedSpy.isValid());
|
|
|
|
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
2021-09-03 17:54:03 +00:00
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
2021-05-11 05:26:51 +00:00
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
2020-09-01 08:58:46 +00:00
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
QSignalSpy frameGeometryChangedSpy(client, &AbstractClient::frameGeometryChanged);
|
|
|
|
QVERIFY(frameGeometryChangedSpy.isValid());
|
2021-05-11 05:26:51 +00:00
|
|
|
QSignalSpy toplevelConfigureRequestedSpy(shellSurface.data(), &Test::XdgToplevel::configureRequested);
|
|
|
|
QSignalSpy surfaceConfigureRequestedSpy(shellSurface->xdgSurface(), &Test::XdgSurface::configureRequested);
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
QScopedPointer<TextInput> textInput(Test::waylandTextInputManager()->createTextInput(Test::waylandSeat()));
|
|
|
|
|
|
|
|
QVERIFY(!textInput.isNull());
|
|
|
|
textInput->enable(surface.data());
|
2021-05-11 05:26:51 +00:00
|
|
|
QVERIFY(surfaceConfigureRequestedSpy.wait());
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
// Show the keyboard
|
2021-10-18 13:41:05 +00:00
|
|
|
touchNow();
|
2020-09-01 08:58:46 +00:00
|
|
|
textInput->showInputPanel();
|
|
|
|
QVERIFY(clientAddedSpy.wait());
|
|
|
|
|
|
|
|
AbstractClient *keyboardClient = clientAddedSpy.last().first().value<AbstractClient *>();
|
|
|
|
QVERIFY(keyboardClient);
|
|
|
|
QVERIFY(keyboardClient->isInputMethod());
|
|
|
|
|
|
|
|
// Do the actual resize
|
2021-05-11 05:26:51 +00:00
|
|
|
QVERIFY(surfaceConfigureRequestedSpy.wait());
|
2020-09-01 08:58:46 +00:00
|
|
|
|
2021-05-11 05:26:51 +00:00
|
|
|
Test::render(surface.data(), toplevelConfigureRequestedSpy.last().first().value<QSize>(), Qt::red);
|
2020-09-01 08:58:46 +00:00
|
|
|
QVERIFY(frameGeometryChangedSpy.wait());
|
|
|
|
|
2022-03-08 11:53:02 +00:00
|
|
|
QCOMPARE(client->frameGeometry().height(), 1024 - keyboardClient->inputGeometry().height());
|
2020-09-01 08:58:46 +00:00
|
|
|
|
|
|
|
// Hide the keyboard
|
|
|
|
textInput->hideInputPanel();
|
|
|
|
|
2021-05-11 05:26:51 +00:00
|
|
|
QVERIFY(surfaceConfigureRequestedSpy.wait());
|
|
|
|
Test::render(surface.data(), toplevelConfigureRequestedSpy.last().first().value<QSize>(), Qt::red);
|
2020-09-01 08:58:46 +00:00
|
|
|
QVERIFY(frameGeometryChangedSpy.wait());
|
|
|
|
|
|
|
|
QCOMPARE(client->frameGeometry().height(), 1024);
|
|
|
|
|
|
|
|
// Destroy the test client.
|
|
|
|
shellSurface.reset();
|
|
|
|
QVERIFY(Test::waitForWindowDestroyed(client));
|
|
|
|
}
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
void InputMethodTest::testEnableDisableV3()
|
2020-09-25 06:51:04 +00:00
|
|
|
{
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
2021-09-03 17:54:03 +00:00
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
2021-05-11 05:26:51 +00:00
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
2020-09-25 06:51:04 +00:00
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
|
|
|
|
Test::TextInputV3 *textInputV3 = new Test::TextInputV3();
|
|
|
|
textInputV3->init(Test::waylandTextInputManagerV3()->get_text_input(*(Test::waylandSeat())));
|
|
|
|
textInputV3->enable();
|
|
|
|
|
2021-05-10 16:01:41 +00:00
|
|
|
QSignalSpy inputMethodActiveSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
// just enabling the text-input should not show it but rather on commit
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
2020-09-25 06:51:04 +00:00
|
|
|
textInputV3->commit();
|
2021-05-10 16:01:41 +00:00
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
2020-09-25 06:51:04 +00:00
|
|
|
|
|
|
|
// disable text input and ensure that it is not hiding input panel without commit
|
2021-05-10 16:01:41 +00:00
|
|
|
inputMethodActiveSpy.clear();
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
2020-09-25 06:51:04 +00:00
|
|
|
textInputV3->disable();
|
|
|
|
textInputV3->commit();
|
2021-05-10 16:01:41 +00:00
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
2020-09-25 06:51:04 +00:00
|
|
|
}
|
|
|
|
|
2021-07-20 23:41:38 +00:00
|
|
|
void InputMethodTest::testEnableActive()
|
|
|
|
{
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
QSignalSpy clientAddedSpy(workspace(), &Workspace::clientAdded);
|
|
|
|
QSignalSpy clientRemovedSpy(workspace(), &Workspace::clientRemoved);
|
|
|
|
|
|
|
|
QSignalSpy activateSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
2021-09-03 17:54:03 +00:00
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
2021-07-20 23:41:38 +00:00
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
QSignalSpy frameGeometryChangedSpy(client, &AbstractClient::frameGeometryChanged);
|
|
|
|
QVERIFY(frameGeometryChangedSpy.isValid());
|
|
|
|
QSignalSpy toplevelConfigureRequestedSpy(shellSurface.data(), &Test::XdgToplevel::configureRequested);
|
|
|
|
QSignalSpy surfaceConfigureRequestedSpy(shellSurface->xdgSurface(), &Test::XdgSurface::configureRequested);
|
|
|
|
|
|
|
|
QScopedPointer<TextInput> textInput(Test::waylandTextInputManager()->createTextInput(Test::waylandSeat()));
|
|
|
|
|
|
|
|
QVERIFY(!textInput.isNull());
|
|
|
|
textInput->enable(surface.data());
|
|
|
|
QVERIFY(surfaceConfigureRequestedSpy.wait());
|
|
|
|
QCOMPARE(clientAddedSpy.count(), 1);
|
|
|
|
|
|
|
|
// Show the keyboard
|
|
|
|
textInput->showInputPanel();
|
|
|
|
QVERIFY(clientAddedSpy.wait());
|
|
|
|
|
|
|
|
QCOMPARE(workspace()->activeClient(), client);
|
|
|
|
|
|
|
|
activateSpy.clear();
|
|
|
|
textInput->enable(surface.get());
|
|
|
|
textInput->showInputPanel();
|
|
|
|
activateSpy.wait(200);
|
|
|
|
QVERIFY(activateSpy.isEmpty());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
auto keyboardClient = Test::inputPanelClient();
|
|
|
|
QVERIFY(keyboardClient);
|
|
|
|
textInput->enable(surface.get());
|
|
|
|
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
// Destroy the test client.
|
|
|
|
shellSurface.reset();
|
|
|
|
QVERIFY(Test::waitForWindowDestroyed(client));
|
|
|
|
}
|
|
|
|
|
2021-07-21 00:04:33 +00:00
|
|
|
void InputMethodTest::testHidePanel()
|
|
|
|
{
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
|
2022-01-19 01:52:44 +00:00
|
|
|
touchNow();
|
2021-07-21 00:04:33 +00:00
|
|
|
QSignalSpy clientAddedSpy(workspace(), &Workspace::clientAdded);
|
|
|
|
QSignalSpy clientRemovedSpy(workspace(), &Workspace::clientRemoved);
|
|
|
|
QVERIFY(clientAddedSpy.isValid());
|
|
|
|
|
|
|
|
QSignalSpy activateSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
QScopedPointer<TextInput> textInput(Test::waylandTextInputManager()->createTextInput(Test::waylandSeat()));
|
|
|
|
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
2021-09-03 17:54:03 +00:00
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
2021-07-21 00:04:33 +00:00
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
waylandServer()->seat()->setFocusedTextInputSurface(client->surface());
|
|
|
|
|
2022-01-19 01:52:44 +00:00
|
|
|
textInput->enable(surface.get());
|
|
|
|
textInput->showInputPanel();
|
|
|
|
QVERIFY(clientAddedSpy.wait());
|
|
|
|
|
2021-07-21 00:04:33 +00:00
|
|
|
QCOMPARE(workspace()->activeClient(), client);
|
|
|
|
|
|
|
|
QCOMPARE(clientAddedSpy.count(), 2);
|
|
|
|
QVERIFY(activateSpy.count() || activateSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
auto keyboardClient = Test::inputPanelClient();
|
|
|
|
auto ipsurface = Test::inputPanelSurface();
|
|
|
|
QVERIFY(keyboardClient);
|
|
|
|
clientRemovedSpy.clear();
|
|
|
|
delete ipsurface;
|
|
|
|
QVERIFY(InputMethod::self()->isVisible());
|
|
|
|
QVERIFY(clientRemovedSpy.count() || clientRemovedSpy.wait());
|
|
|
|
QVERIFY(!InputMethod::self()->isVisible());
|
|
|
|
|
|
|
|
// Destroy the test client.
|
|
|
|
shellSurface.reset();
|
|
|
|
QVERIFY(Test::waitForWindowDestroyed(client));
|
|
|
|
}
|
|
|
|
|
2021-07-30 18:29:01 +00:00
|
|
|
void InputMethodTest::testSwitchFocusedSurfaces()
|
|
|
|
{
|
2022-01-19 01:52:44 +00:00
|
|
|
touchNow();
|
2021-07-30 18:29:01 +00:00
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
QSignalSpy clientAddedSpy(workspace(), &Workspace::clientAdded);
|
|
|
|
QSignalSpy clientRemovedSpy(workspace(), &Workspace::clientRemoved);
|
|
|
|
QVERIFY(clientAddedSpy.isValid());
|
|
|
|
|
|
|
|
QSignalSpy activateSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
QScopedPointer<TextInput> textInput(Test::waylandTextInputManager()->createTextInput(Test::waylandSeat()));
|
|
|
|
|
|
|
|
QVector<AbstractClient *> clients;
|
2021-09-03 17:54:03 +00:00
|
|
|
QVector<KWayland::Client::Surface *> surfaces;
|
2021-07-30 18:29:01 +00:00
|
|
|
QVector<Test::XdgToplevel *> toplevels;
|
|
|
|
// We create 3 surfaces
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
auto surface = Test::createSurface();
|
|
|
|
auto shellSurface = Test::createXdgToplevelSurface(surface);
|
|
|
|
clients += Test::renderAndWaitForShown(surface, QSize(1280, 1024), Qt::red);
|
|
|
|
QCOMPARE(workspace()->activeClient(), clients.constLast());
|
|
|
|
surfaces += surface;
|
|
|
|
toplevels += shellSurface;
|
|
|
|
}
|
2022-01-19 01:52:44 +00:00
|
|
|
QCOMPARE(clientAddedSpy.count(), 3);
|
2021-07-30 18:29:01 +00:00
|
|
|
waylandServer()->seat()->setFocusedTextInputSurface(clients.constFirst()->surface());
|
|
|
|
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
textInput->enable(surfaces.last());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
waylandServer()->seat()->setFocusedTextInputSurface(clients.first()->surface());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
activateSpy.clear();
|
|
|
|
waylandServer()->seat()->setFocusedTextInputSurface(clients.last()->surface());
|
|
|
|
QVERIFY(activateSpy.count() || activateSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
activateSpy.clear();
|
|
|
|
waylandServer()->seat()->setFocusedTextInputSurface(clients.first()->surface());
|
|
|
|
QVERIFY(activateSpy.count() || activateSpy.wait());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
// Destroy the test client.
|
|
|
|
for (int i = 0; i < clients.count(); ++i) {
|
|
|
|
delete toplevels[i];
|
|
|
|
QVERIFY(Test::waitForWindowDestroyed(clients[i]));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-23 08:42:00 +00:00
|
|
|
void InputMethodTest::testV3Styling()
|
|
|
|
{
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
|
|
|
|
Test::TextInputV3 *textInputV3 = new Test::TextInputV3();
|
|
|
|
textInputV3->init(Test::waylandTextInputManagerV3()->get_text_input(*(Test::waylandSeat())));
|
|
|
|
textInputV3->enable();
|
|
|
|
|
|
|
|
QSignalSpy inputMethodActiveSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
QSignalSpy inputMethodActivateSpy(Test::inputMethod(), &Test::MockInputMethod::activate);
|
|
|
|
// just enabling the text-input should not show it but rather on commit
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
textInputV3->commit();
|
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
QVERIFY(inputMethodActivateSpy.wait());
|
|
|
|
auto context = Test::inputMethod()->context();
|
|
|
|
QSignalSpy textInputPreeditSpy(textInputV3, &Test::TextInputV3::preeditString);
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 0);
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 0, 3, 7);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCD", "ABCD");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCD"));
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 0);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 0);
|
|
|
|
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 1);
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 0, 3, 7);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCDE", "ABCDE");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCDE"));
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 1);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 1);
|
|
|
|
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 2);
|
|
|
|
// Use selection for [2, 2+2)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 2, 2, 6);
|
|
|
|
// Use high light for [3, 3+3)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 3, 3, 4);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCDEF", "ABCDEF");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCDEF"));
|
|
|
|
// Merged range should be [2, 6)
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 2);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 6);
|
|
|
|
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 2);
|
|
|
|
// Use selection for [0, 0+2)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 0, 2, 6);
|
|
|
|
// Use high light for [3, 3+3)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 3, 3, 4);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCDEF", "ABCDEF");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCDEF"));
|
|
|
|
// Merged range should be none, because of the disjunction highlight.
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 2);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 2);
|
|
|
|
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 1);
|
|
|
|
// Use selection for [0, 0+2)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 0, 2, 6);
|
|
|
|
// Use high light for [2, 2+3)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 2, 3, 4);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCDEF", "ABCDEF");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCDEF"));
|
|
|
|
// Merged range should be none, starting offset does not match.
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 1);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 1);
|
|
|
|
|
|
|
|
// Use different order of styling and cursor
|
|
|
|
// Use high light for [3, 3+3)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 3, 3, 4);
|
|
|
|
zwp_input_method_context_v1_preedit_cursor(context, 1);
|
|
|
|
// Use selection for [1, 1+2)
|
|
|
|
zwp_input_method_context_v1_preedit_styling(context, 1, 2, 6);
|
|
|
|
zwp_input_method_context_v1_preedit_string(context, 0, "ABCDEF", "ABCDEF");
|
|
|
|
QVERIFY(textInputPreeditSpy.wait());
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(0), QString("ABCDEF"));
|
|
|
|
// Merged range should be [1,6).
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(1), 1);
|
|
|
|
QCOMPARE(textInputPreeditSpy.last().at(2), 6);
|
|
|
|
}
|
|
|
|
|
2021-12-23 22:33:30 +00:00
|
|
|
void InputMethodTest::testDisableShowInputPanel()
|
|
|
|
{
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
|
|
|
|
QScopedPointer<KWayland::Client::TextInput> textInputV2(Test::waylandTextInputManager()->createTextInput(Test::waylandSeat()));
|
|
|
|
|
|
|
|
QSignalSpy inputMethodActiveSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
// just enabling the text-input should not show it but rather on commit
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
textInputV2->enable(surface.get());
|
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
// disable text input and ensure that it is not hiding input panel without commit
|
|
|
|
inputMethodActiveSpy.clear();
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
textInputV2->disable(surface.get());
|
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
|
|
|
|
QSignalSpy requestShowInputPanelSpy(waylandServer()->seat()->textInputV2(), &KWaylandServer::TextInputV2Interface::requestShowInputPanel);
|
|
|
|
textInputV2->showInputPanel();
|
|
|
|
QVERIFY(requestShowInputPanelSpy.count() || requestShowInputPanelSpy.wait());
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
}
|
|
|
|
|
2021-12-29 17:19:43 +00:00
|
|
|
void InputMethodTest::testModifierForwarding()
|
|
|
|
{
|
|
|
|
// Create an xdg_toplevel surface and wait for the compositor to catch up.
|
|
|
|
QScopedPointer<KWayland::Client::Surface> surface(Test::createSurface());
|
|
|
|
QScopedPointer<Test::XdgToplevel> shellSurface(Test::createXdgToplevelSurface(surface.data()));
|
|
|
|
AbstractClient *client = Test::renderAndWaitForShown(surface.data(), QSize(1280, 1024), Qt::red);
|
|
|
|
QVERIFY(client);
|
|
|
|
QVERIFY(client->isActive());
|
|
|
|
QCOMPARE(client->frameGeometry().size(), QSize(1280, 1024));
|
|
|
|
|
|
|
|
Test::TextInputV3 *textInputV3 = new Test::TextInputV3();
|
|
|
|
textInputV3->init(Test::waylandTextInputManagerV3()->get_text_input(*(Test::waylandSeat())));
|
|
|
|
textInputV3->enable();
|
|
|
|
|
|
|
|
QSignalSpy inputMethodActiveSpy(InputMethod::self(), &InputMethod::activeChanged);
|
|
|
|
QSignalSpy inputMethodActivateSpy(Test::inputMethod(), &Test::MockInputMethod::activate);
|
|
|
|
// just enabling the text-input should not show it but rather on commit
|
|
|
|
QVERIFY(!InputMethod::self()->isActive());
|
|
|
|
textInputV3->commit();
|
|
|
|
QVERIFY(inputMethodActiveSpy.count() || inputMethodActiveSpy.wait());
|
|
|
|
QVERIFY(InputMethod::self()->isActive());
|
|
|
|
QVERIFY(inputMethodActivateSpy.wait());
|
|
|
|
auto context = Test::inputMethod()->context();
|
|
|
|
QScopedPointer<KWayland::Client::Keyboard> keyboardGrab(new KWayland::Client::Keyboard);
|
|
|
|
keyboardGrab->setup(zwp_input_method_context_v1_grab_keyboard(context));
|
|
|
|
QSignalSpy modifierSpy(keyboardGrab.get(), &Keyboard::modifiersChanged);
|
|
|
|
// Wait for initial modifiers update
|
|
|
|
QVERIFY(modifierSpy.wait());
|
|
|
|
|
|
|
|
quint32 timestamp = 1;
|
|
|
|
|
|
|
|
QSignalSpy keySpy(keyboardGrab.get(), &Keyboard::keyChanged);
|
|
|
|
bool keyChanged = false;
|
|
|
|
bool modifiersChanged = false;
|
|
|
|
// We want to verify the order of two signals, so SignalSpy is not very useful here.
|
|
|
|
auto keyChangedConnection = connect(keyboardGrab.get(), &Keyboard::keyChanged, [&keyChanged, &modifiersChanged]() {
|
|
|
|
QVERIFY(!modifiersChanged);
|
|
|
|
keyChanged = true;
|
|
|
|
});
|
|
|
|
auto modifiersChangedConnection = connect(keyboardGrab.get(), &Keyboard::modifiersChanged, [&keyChanged, &modifiersChanged]() {
|
|
|
|
QVERIFY(keyChanged);
|
|
|
|
modifiersChanged = true;
|
|
|
|
});
|
2022-03-10 10:27:35 +00:00
|
|
|
Test::keyboardKeyPressed(KEY_LEFTCTRL, timestamp++);
|
2021-12-29 17:19:43 +00:00
|
|
|
QVERIFY(keySpy.count() == 1 || keySpy.wait());
|
|
|
|
QVERIFY(modifierSpy.count() == 2 || modifierSpy.wait());
|
|
|
|
disconnect(keyChangedConnection);
|
|
|
|
disconnect(modifiersChangedConnection);
|
|
|
|
|
2022-03-10 10:27:35 +00:00
|
|
|
Test::keyboardKeyPressed(KEY_A, timestamp++);
|
2021-12-29 17:19:43 +00:00
|
|
|
QVERIFY(keySpy.count() == 2 || keySpy.wait());
|
|
|
|
QVERIFY(modifierSpy.count() == 2 || modifierSpy.wait());
|
|
|
|
|
|
|
|
// verify the order of key and modifiers again. Key first, then modifiers.
|
|
|
|
keyChanged = false;
|
|
|
|
modifiersChanged = false;
|
|
|
|
keyChangedConnection = connect(keyboardGrab.get(), &Keyboard::keyChanged, [&keyChanged, &modifiersChanged]() {
|
|
|
|
QVERIFY(!modifiersChanged);
|
|
|
|
keyChanged = true;
|
|
|
|
});
|
|
|
|
modifiersChangedConnection = connect(keyboardGrab.get(), &Keyboard::modifiersChanged, [&keyChanged, &modifiersChanged]() {
|
|
|
|
QVERIFY(keyChanged);
|
|
|
|
modifiersChanged = true;
|
|
|
|
});
|
2022-03-10 10:27:35 +00:00
|
|
|
Test::keyboardKeyReleased(KEY_LEFTCTRL, timestamp++);
|
2021-12-29 17:19:43 +00:00
|
|
|
QVERIFY(keySpy.count() == 3 || keySpy.wait());
|
|
|
|
QVERIFY(modifierSpy.count() == 3 || modifierSpy.wait());
|
|
|
|
disconnect(keyChangedConnection);
|
|
|
|
disconnect(modifiersChangedConnection);
|
|
|
|
}
|
|
|
|
|
2020-09-30 10:28:42 +00:00
|
|
|
WAYLANDTEST_MAIN(InputMethodTest)
|
2020-09-01 08:58:46 +00:00
|
|
|
|
2020-09-30 10:46:32 +00:00
|
|
|
#include "inputmethod_test.moc"
|