2016-06-07 09:43:56 +00:00
|
|
|
|
/********************************************************************
|
|
|
|
|
KWin - the KDE window manager
|
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
|
|
Copyright (C) 2016 Martin Gräßlin <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 "kwin_wayland_test.h"
|
|
|
|
|
#include "cursor.h"
|
2016-07-01 14:03:13 +00:00
|
|
|
|
#include "effects.h"
|
2016-06-07 09:43:56 +00:00
|
|
|
|
#include "platform.h"
|
|
|
|
|
#include "shell_client.h"
|
|
|
|
|
#include "screens.h"
|
|
|
|
|
#include "wayland_server.h"
|
2016-06-14 11:46:28 +00:00
|
|
|
|
#include "workspace.h"
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
#include <KWayland/Client/connection_thread.h>
|
|
|
|
|
#include <KWayland/Client/compositor.h>
|
|
|
|
|
#include <KWayland/Client/shell.h>
|
2016-09-12 12:14:20 +00:00
|
|
|
|
#include <KWayland/Client/server_decoration.h>
|
2016-06-07 09:43:56 +00:00
|
|
|
|
#include <KWayland/Client/surface.h>
|
2016-09-12 10:03:18 +00:00
|
|
|
|
#include <KWayland/Client/xdgshell.h>
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
#include <KWayland/Server/shell_interface.h>
|
|
|
|
|
|
|
|
|
|
using namespace KWin;
|
|
|
|
|
using namespace KWayland::Client;
|
|
|
|
|
|
|
|
|
|
static const QString s_socketName = QStringLiteral("wayland_test_kwin_shell_client-0");
|
|
|
|
|
|
|
|
|
|
class TestShellClient : public QObject
|
|
|
|
|
{
|
|
|
|
|
Q_OBJECT
|
|
|
|
|
private Q_SLOTS:
|
|
|
|
|
void initTestCase();
|
|
|
|
|
void init();
|
|
|
|
|
void cleanup();
|
|
|
|
|
|
2016-04-22 12:13:37 +00:00
|
|
|
|
void testMapUnmapMap_data();
|
2016-06-07 09:43:56 +00:00
|
|
|
|
void testMapUnmapMap();
|
2016-07-01 14:03:13 +00:00
|
|
|
|
void testDesktopPresenceChanged();
|
2016-08-08 12:55:26 +00:00
|
|
|
|
void testTransientPositionAfterRemap();
|
2016-08-11 15:08:08 +00:00
|
|
|
|
void testMinimizeActiveWindow_data();
|
|
|
|
|
void testMinimizeActiveWindow();
|
2016-09-12 10:03:18 +00:00
|
|
|
|
void testFullscreen_data();
|
|
|
|
|
void testFullscreen();
|
2016-09-12 12:41:03 +00:00
|
|
|
|
void testMaximizedToFullscreen_data();
|
|
|
|
|
void testMaximizedToFullscreen();
|
2016-09-14 12:08:47 +00:00
|
|
|
|
void testWindowOpensLargerThanScreen_data();
|
|
|
|
|
void testWindowOpensLargerThanScreen();
|
2016-10-11 14:24:10 +00:00
|
|
|
|
void testHidden_data();
|
|
|
|
|
void testHidden();
|
2016-10-27 13:59:08 +00:00
|
|
|
|
void testDesktopFileName();
|
2016-10-31 14:50:14 +00:00
|
|
|
|
void testCaptionSimplified();
|
2016-06-07 09:43:56 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
void TestShellClient::initTestCase()
|
|
|
|
|
{
|
|
|
|
|
qRegisterMetaType<KWin::ShellClient*>();
|
|
|
|
|
qRegisterMetaType<KWin::AbstractClient*>();
|
|
|
|
|
QSignalSpy workspaceCreatedSpy(kwinApp(), &Application::workspaceCreated);
|
|
|
|
|
QVERIFY(workspaceCreatedSpy.isValid());
|
|
|
|
|
kwinApp()->platform()->setInitialWindowSize(QSize(1280, 1024));
|
|
|
|
|
QMetaObject::invokeMethod(kwinApp()->platform(), "setOutputCount", Qt::DirectConnection, Q_ARG(int, 2));
|
2016-07-04 07:09:03 +00:00
|
|
|
|
QVERIFY(waylandServer()->init(s_socketName.toLocal8Bit()));
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
kwinApp()->start();
|
|
|
|
|
QVERIFY(workspaceCreatedSpy.wait());
|
|
|
|
|
QCOMPARE(screens()->count(), 2);
|
|
|
|
|
QCOMPARE(screens()->geometry(0), QRect(0, 0, 1280, 1024));
|
|
|
|
|
QCOMPARE(screens()->geometry(1), QRect(1280, 0, 1280, 1024));
|
|
|
|
|
waylandServer()->initWorkspace();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::init()
|
|
|
|
|
{
|
2016-09-12 12:14:20 +00:00
|
|
|
|
QVERIFY(Test::setupWaylandConnection(s_socketName, Test::AdditionalWaylandInterface::Decoration));
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
screens()->setCurrent(0);
|
|
|
|
|
KWin::Cursor::setPos(QPoint(1280, 512));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::cleanup()
|
|
|
|
|
{
|
2016-06-30 11:32:54 +00:00
|
|
|
|
Test::destroyWaylandConnection();
|
2016-06-07 09:43:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-04-22 12:13:37 +00:00
|
|
|
|
void TestShellClient::testMapUnmapMap_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5;
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-07 09:43:56 +00:00
|
|
|
|
void TestShellClient::testMapUnmapMap()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that mapping a previously mapped window works correctly
|
|
|
|
|
QSignalSpy clientAddedSpy(waylandServer(), &WaylandServer::shellClientAdded);
|
|
|
|
|
QVERIFY(clientAddedSpy.isValid());
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QSignalSpy effectsWindowShownSpy(effects, &EffectsHandler::windowShown);
|
|
|
|
|
QVERIFY(effectsWindowShownSpy.isValid());
|
|
|
|
|
QSignalSpy effectsWindowHiddenSpy(effects, &EffectsHandler::windowHidden);
|
|
|
|
|
QVERIFY(effectsWindowHiddenSpy.isValid());
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
2016-06-30 11:32:54 +00:00
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
2016-04-22 12:13:37 +00:00
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
// now let's render
|
2016-06-30 11:32:54 +00:00
|
|
|
|
Test::render(surface.data(), QSize(100, 50), Qt::blue);
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
QVERIFY(clientAddedSpy.isEmpty());
|
|
|
|
|
QVERIFY(clientAddedSpy.wait());
|
|
|
|
|
auto client = clientAddedSpy.first().first().value<ShellClient*>();
|
|
|
|
|
QVERIFY(client);
|
|
|
|
|
QVERIFY(client->isShown(true));
|
2016-07-04 13:06:20 +00:00
|
|
|
|
QCOMPARE(client->isHiddenInternal(), false);
|
|
|
|
|
QCOMPARE(client->readyForPainting(), true);
|
2016-08-09 11:50:29 +00:00
|
|
|
|
QCOMPARE(client->depth(), 32);
|
|
|
|
|
QVERIFY(client->hasAlpha());
|
2016-06-14 11:46:28 +00:00
|
|
|
|
QCOMPARE(workspace()->activeClient(), client);
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QVERIFY(effectsWindowShownSpy.isEmpty());
|
2016-08-29 05:53:26 +00:00
|
|
|
|
QVERIFY(client->isMaximizable());
|
|
|
|
|
QVERIFY(client->isMovable());
|
|
|
|
|
QVERIFY(client->isMovableAcrossScreens());
|
|
|
|
|
QVERIFY(client->isResizable());
|
|
|
|
|
QVERIFY(client->property("maximizable").toBool());
|
|
|
|
|
QVERIFY(client->property("moveable").toBool());
|
|
|
|
|
QVERIFY(client->property("moveableAcrossScreens").toBool());
|
|
|
|
|
QVERIFY(client->property("resizeable").toBool());
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
// now unmap
|
|
|
|
|
QSignalSpy hiddenSpy(client, &ShellClient::windowHidden);
|
|
|
|
|
QVERIFY(hiddenSpy.isValid());
|
|
|
|
|
QSignalSpy windowClosedSpy(client, &ShellClient::windowClosed);
|
|
|
|
|
QVERIFY(windowClosedSpy.isValid());
|
|
|
|
|
surface->attachBuffer(Buffer::Ptr());
|
|
|
|
|
surface->commit(Surface::CommitFlag::None);
|
|
|
|
|
QVERIFY(hiddenSpy.wait());
|
2016-07-04 13:06:20 +00:00
|
|
|
|
QCOMPARE(client->readyForPainting(), true);
|
|
|
|
|
QCOMPARE(client->isHiddenInternal(), true);
|
2016-06-07 09:43:56 +00:00
|
|
|
|
QVERIFY(windowClosedSpy.isEmpty());
|
2016-06-14 11:46:28 +00:00
|
|
|
|
QVERIFY(!workspace()->activeClient());
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(effectsWindowHiddenSpy.count(), 1);
|
|
|
|
|
QCOMPARE(effectsWindowHiddenSpy.first().first().value<EffectWindow*>(), client->effectWindow());
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
QSignalSpy windowShownSpy(client, &ShellClient::windowShown);
|
|
|
|
|
QVERIFY(windowShownSpy.isValid());
|
2016-08-09 11:50:29 +00:00
|
|
|
|
Test::render(surface.data(), QSize(100, 50), Qt::blue, QImage::Format_RGB32);
|
2016-06-07 09:43:56 +00:00
|
|
|
|
QCOMPARE(clientAddedSpy.count(), 1);
|
|
|
|
|
QVERIFY(windowShownSpy.wait());
|
|
|
|
|
QCOMPARE(windowShownSpy.count(), 1);
|
|
|
|
|
QCOMPARE(clientAddedSpy.count(), 1);
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(client->readyForPainting(), true);
|
|
|
|
|
QCOMPARE(client->isHiddenInternal(), false);
|
2016-08-09 11:50:29 +00:00
|
|
|
|
QCOMPARE(client->depth(), 24);
|
|
|
|
|
QVERIFY(!client->hasAlpha());
|
2016-06-14 11:46:28 +00:00
|
|
|
|
QCOMPARE(workspace()->activeClient(), client);
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(effectsWindowShownSpy.count(), 1);
|
|
|
|
|
QCOMPARE(effectsWindowShownSpy.first().first().value<EffectWindow*>(), client->effectWindow());
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
// let's unmap again
|
|
|
|
|
surface->attachBuffer(Buffer::Ptr());
|
|
|
|
|
surface->commit(Surface::CommitFlag::None);
|
|
|
|
|
QVERIFY(hiddenSpy.wait());
|
|
|
|
|
QCOMPARE(hiddenSpy.count(), 2);
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(client->readyForPainting(), true);
|
|
|
|
|
QCOMPARE(client->isHiddenInternal(), true);
|
2016-06-07 09:43:56 +00:00
|
|
|
|
QVERIFY(windowClosedSpy.isEmpty());
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(effectsWindowHiddenSpy.count(), 2);
|
|
|
|
|
QCOMPARE(effectsWindowHiddenSpy.last().first().value<EffectWindow*>(), client->effectWindow());
|
2016-06-07 09:43:56 +00:00
|
|
|
|
|
|
|
|
|
shellSurface.reset();
|
|
|
|
|
surface.reset();
|
|
|
|
|
QVERIFY(windowClosedSpy.wait());
|
|
|
|
|
QCOMPARE(windowClosedSpy.count(), 1);
|
2016-07-04 13:37:30 +00:00
|
|
|
|
QCOMPARE(effectsWindowHiddenSpy.count(), 2);
|
2016-06-07 09:43:56 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-01 14:03:13 +00:00
|
|
|
|
void TestShellClient::testDesktopPresenceChanged()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that the desktop presence changed signals are properly emitted
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QScopedPointer<ShellSurface> shellSurface(Test::createShellSurface(surface.data()));
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QCOMPARE(c->desktop(), 1);
|
|
|
|
|
effects->setNumberOfDesktops(4);
|
|
|
|
|
QSignalSpy desktopPresenceChangedClientSpy(c, &ShellClient::desktopPresenceChanged);
|
|
|
|
|
QVERIFY(desktopPresenceChangedClientSpy.isValid());
|
|
|
|
|
QSignalSpy desktopPresenceChangedWorkspaceSpy(workspace(), &Workspace::desktopPresenceChanged);
|
|
|
|
|
QVERIFY(desktopPresenceChangedWorkspaceSpy.isValid());
|
|
|
|
|
QSignalSpy desktopPresenceChangedEffectsSpy(effects, &EffectsHandler::desktopPresenceChanged);
|
|
|
|
|
QVERIFY(desktopPresenceChangedEffectsSpy.isValid());
|
|
|
|
|
|
|
|
|
|
// let's change the desktop
|
|
|
|
|
workspace()->sendClientToDesktop(c, 2, false);
|
|
|
|
|
QCOMPARE(c->desktop(), 2);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedClientSpy.count(), 1);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedWorkspaceSpy.count(), 1);
|
|
|
|
|
// effects is delayed by one cycle
|
|
|
|
|
QCOMPARE(desktopPresenceChangedEffectsSpy.count(), 0);
|
|
|
|
|
QVERIFY(desktopPresenceChangedEffectsSpy.wait());
|
|
|
|
|
QCOMPARE(desktopPresenceChangedEffectsSpy.count(), 1);
|
|
|
|
|
|
|
|
|
|
// verify the arguments
|
|
|
|
|
QCOMPARE(desktopPresenceChangedClientSpy.first().at(0).value<AbstractClient*>(), c);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedClientSpy.first().at(1).toInt(), 1);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedWorkspaceSpy.first().at(0).value<AbstractClient*>(), c);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedWorkspaceSpy.first().at(1).toInt(), 1);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedEffectsSpy.first().at(0).value<EffectWindow*>(), c->effectWindow());
|
|
|
|
|
QCOMPARE(desktopPresenceChangedEffectsSpy.first().at(1).toInt(), 1);
|
|
|
|
|
QCOMPARE(desktopPresenceChangedEffectsSpy.first().at(2).toInt(), 2);
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-08 12:55:26 +00:00
|
|
|
|
void TestShellClient::testTransientPositionAfterRemap()
|
|
|
|
|
{
|
|
|
|
|
// this test simulates the situation that a transient window gets reused and the parent window
|
|
|
|
|
// moved between the two usages
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QScopedPointer<ShellSurface> shellSurface(Test::createShellSurface(surface.data()));
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
|
|
|
|
|
// create the Transient window
|
|
|
|
|
QScopedPointer<Surface> transientSurface(Test::createSurface());
|
|
|
|
|
QScopedPointer<ShellSurface> transientShellSurface(Test::createShellSurface(transientSurface.data()));
|
|
|
|
|
transientShellSurface->setTransient(surface.data(), QPoint(5, 10));
|
|
|
|
|
auto transient = Test::renderAndWaitForShown(transientSurface.data(), QSize(50, 40), Qt::blue);
|
|
|
|
|
QVERIFY(transient);
|
|
|
|
|
QCOMPARE(transient->geometry(), QRect(c->geometry().topLeft() + QPoint(5, 10), QSize(50, 40)));
|
|
|
|
|
|
|
|
|
|
// unmap the transient
|
|
|
|
|
QSignalSpy windowHiddenSpy(transient, &ShellClient::windowHidden);
|
|
|
|
|
QVERIFY(windowHiddenSpy.isValid());
|
|
|
|
|
transientSurface->attachBuffer(Buffer::Ptr());
|
|
|
|
|
transientSurface->commit(Surface::CommitFlag::None);
|
|
|
|
|
QVERIFY(windowHiddenSpy.wait());
|
|
|
|
|
|
|
|
|
|
// now move the parent surface
|
|
|
|
|
c->setGeometry(c->geometry().translated(5, 10));
|
|
|
|
|
|
|
|
|
|
// now map the transient again
|
|
|
|
|
QSignalSpy windowShownSpy(transient, &ShellClient::windowShown);
|
|
|
|
|
QVERIFY(windowShownSpy.isValid());
|
|
|
|
|
Test::render(transientSurface.data(), QSize(50, 40), Qt::blue);
|
|
|
|
|
QVERIFY(windowShownSpy.wait());
|
|
|
|
|
QCOMPARE(transient->geometry(), QRect(c->geometry().topLeft() + QPoint(5, 10), QSize(50, 40)));
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-11 15:08:08 +00:00
|
|
|
|
void TestShellClient::testMinimizeActiveWindow_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testMinimizeActiveWindow()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that when minimizing the active window it gets deactivated
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->isActive());
|
|
|
|
|
QCOMPARE(workspace()->activeClient(), c);
|
2016-09-12 08:01:02 +00:00
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
|
|
|
|
QVERIFY(c->isShown(true));
|
2016-08-11 15:08:08 +00:00
|
|
|
|
|
|
|
|
|
workspace()->slotWindowMinimize();
|
2016-09-12 08:01:02 +00:00
|
|
|
|
QVERIFY(!c->isShown(true));
|
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
2016-08-11 15:08:08 +00:00
|
|
|
|
QVERIFY(!c->isActive());
|
|
|
|
|
QVERIFY(!workspace()->activeClient());
|
2016-08-18 08:30:27 +00:00
|
|
|
|
QVERIFY(c->isMinimized());
|
|
|
|
|
|
|
|
|
|
// unminimize again
|
|
|
|
|
c->unminimize();
|
|
|
|
|
QVERIFY(!c->isMinimized());
|
|
|
|
|
QVERIFY(c->isActive());
|
2016-09-12 08:01:02 +00:00
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
|
|
|
|
QVERIFY(c->isShown(true));
|
2016-08-18 08:30:27 +00:00
|
|
|
|
QCOMPARE(workspace()->activeClient(), c);
|
2016-08-11 15:08:08 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-09-12 10:03:18 +00:00
|
|
|
|
void TestShellClient::testFullscreen_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
2016-09-12 12:14:20 +00:00
|
|
|
|
QTest::addColumn<ServerSideDecoration::Mode>("decoMode");
|
2016-09-12 10:03:18 +00:00
|
|
|
|
|
2016-09-12 12:14:20 +00:00
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell << ServerSideDecoration::Mode::Client;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5 << ServerSideDecoration::Mode::Client;
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell - deco") << Test::ShellSurfaceType::WlShell << ServerSideDecoration::Mode::Server;
|
|
|
|
|
QTest::newRow("xdgShellV5 - deco") << Test::ShellSurfaceType::XdgShellV5 << ServerSideDecoration::Mode::Server;
|
2016-09-12 10:03:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testFullscreen()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that a window can be properly fullscreened
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
2016-09-12 12:14:20 +00:00
|
|
|
|
|
|
|
|
|
// create deco
|
|
|
|
|
QScopedPointer<ServerSideDecoration> deco(Test::waylandServerSideDecoration()->create(surface.data()));
|
|
|
|
|
QSignalSpy decoSpy(deco.data(), &ServerSideDecoration::modeChanged);
|
|
|
|
|
QVERIFY(decoSpy.isValid());
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
QFETCH(ServerSideDecoration::Mode, decoMode);
|
|
|
|
|
deco->requestMode(decoMode);
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
QCOMPARE(deco->mode(), decoMode);
|
|
|
|
|
|
2016-09-12 10:03:18 +00:00
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->isActive());
|
2016-09-13 08:30:52 +00:00
|
|
|
|
QCOMPARE(c->layer(), NormalLayer);
|
2016-09-12 10:03:18 +00:00
|
|
|
|
QVERIFY(!c->isFullScreen());
|
2016-09-12 12:14:20 +00:00
|
|
|
|
QCOMPARE(c->clientSize(), QSize(100, 50));
|
2016-09-12 12:41:03 +00:00
|
|
|
|
QCOMPARE(c->isDecorated(), decoMode == ServerSideDecoration::Mode::Server);
|
2016-09-12 10:03:18 +00:00
|
|
|
|
QSignalSpy fullscreenChangedSpy(c, &ShellClient::fullScreenChanged);
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.isValid());
|
|
|
|
|
QSignalSpy geometryChangedSpy(c, &ShellClient::geometryChanged);
|
|
|
|
|
QVERIFY(geometryChangedSpy.isValid());
|
|
|
|
|
QSignalSpy sizeChangeRequestedSpy(shellSurface.data(), SIGNAL(sizeChanged(QSize)));
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.isValid());
|
|
|
|
|
|
|
|
|
|
// fullscreen the window
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Test::ShellSurfaceType::WlShell:
|
|
|
|
|
qobject_cast<ShellSurface*>(shellSurface.data())->setFullscreen();
|
|
|
|
|
break;
|
|
|
|
|
case Test::ShellSurfaceType::XdgShellV5:
|
|
|
|
|
qobject_cast<XdgShellSurface*>(shellSurface.data())->setFullscreen(true);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.wait());
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.count(), 1);
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.first().first().toSize(), QSize(screens()->size(0)));
|
|
|
|
|
// TODO: should switch to fullscreen once it's updated
|
|
|
|
|
QVERIFY(c->isFullScreen());
|
2016-09-12 12:14:20 +00:00
|
|
|
|
QCOMPARE(c->clientSize(), QSize(100, 50));
|
2016-09-12 10:03:18 +00:00
|
|
|
|
QVERIFY(geometryChangedSpy.isEmpty());
|
|
|
|
|
|
|
|
|
|
// render at the new size
|
|
|
|
|
Test::render(surface.data(), sizeChangeRequestedSpy.first().first().toSize(), Qt::red);
|
|
|
|
|
QVERIFY(geometryChangedSpy.wait());
|
|
|
|
|
QCOMPARE(geometryChangedSpy.count(), 1);
|
|
|
|
|
QVERIFY(c->isFullScreen());
|
2016-09-12 12:41:03 +00:00
|
|
|
|
QVERIFY(!c->isDecorated());
|
2016-09-12 10:03:18 +00:00
|
|
|
|
QCOMPARE(c->geometry(), QRect(QPoint(0, 0), sizeChangeRequestedSpy.first().first().toSize()));
|
2016-09-13 08:30:52 +00:00
|
|
|
|
QCOMPARE(c->layer(), ActiveLayer);
|
2016-09-12 10:03:18 +00:00
|
|
|
|
|
|
|
|
|
// swap back to normal
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Test::ShellSurfaceType::WlShell:
|
|
|
|
|
qobject_cast<ShellSurface*>(shellSurface.data())->setToplevel();
|
|
|
|
|
break;
|
|
|
|
|
case Test::ShellSurfaceType::XdgShellV5:
|
|
|
|
|
qobject_cast<XdgShellSurface*>(shellSurface.data())->setFullscreen(false);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.wait());
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.count(), 2);
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.last().first().toSize(), QSize(100, 50));
|
|
|
|
|
// TODO: should switch to fullscreen once it's updated
|
|
|
|
|
QVERIFY(!c->isFullScreen());
|
2016-09-13 08:30:52 +00:00
|
|
|
|
QCOMPARE(c->layer(), NormalLayer);
|
2016-09-12 12:41:03 +00:00
|
|
|
|
QCOMPARE(c->isDecorated(), decoMode == ServerSideDecoration::Mode::Server);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testMaximizedToFullscreen_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
|
|
|
|
QTest::addColumn<ServerSideDecoration::Mode>("decoMode");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell << ServerSideDecoration::Mode::Client;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5 << ServerSideDecoration::Mode::Client;
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell - deco") << Test::ShellSurfaceType::WlShell << ServerSideDecoration::Mode::Server;
|
|
|
|
|
QTest::newRow("xdgShellV5 - deco") << Test::ShellSurfaceType::XdgShellV5 << ServerSideDecoration::Mode::Server;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testMaximizedToFullscreen()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that a window can be properly fullscreened after maximizing
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
|
|
|
|
|
|
|
|
|
// create deco
|
|
|
|
|
QScopedPointer<ServerSideDecoration> deco(Test::waylandServerSideDecoration()->create(surface.data()));
|
|
|
|
|
QSignalSpy decoSpy(deco.data(), &ServerSideDecoration::modeChanged);
|
|
|
|
|
QVERIFY(decoSpy.isValid());
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
QFETCH(ServerSideDecoration::Mode, decoMode);
|
|
|
|
|
deco->requestMode(decoMode);
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
QCOMPARE(deco->mode(), decoMode);
|
|
|
|
|
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->isActive());
|
|
|
|
|
QVERIFY(!c->isFullScreen());
|
|
|
|
|
QCOMPARE(c->clientSize(), QSize(100, 50));
|
|
|
|
|
QCOMPARE(c->isDecorated(), decoMode == ServerSideDecoration::Mode::Server);
|
|
|
|
|
QSignalSpy fullscreenChangedSpy(c, &ShellClient::fullScreenChanged);
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.isValid());
|
|
|
|
|
QSignalSpy geometryChangedSpy(c, &ShellClient::geometryChanged);
|
|
|
|
|
QVERIFY(geometryChangedSpy.isValid());
|
|
|
|
|
QSignalSpy sizeChangeRequestedSpy(shellSurface.data(), SIGNAL(sizeChanged(QSize)));
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.isValid());
|
|
|
|
|
|
|
|
|
|
// change to maximize
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Test::ShellSurfaceType::WlShell:
|
|
|
|
|
qobject_cast<ShellSurface*>(shellSurface.data())->setMaximized();
|
|
|
|
|
break;
|
|
|
|
|
case Test::ShellSurfaceType::XdgShellV5:
|
|
|
|
|
qobject_cast<XdgShellSurface*>(shellSurface.data())->setMaximized(true);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.count(), 1);
|
|
|
|
|
QCOMPARE(c->maximizeMode(), MaximizeFull);
|
|
|
|
|
QCOMPARE(geometryChangedSpy.isEmpty(), false);
|
|
|
|
|
geometryChangedSpy.clear();
|
|
|
|
|
|
|
|
|
|
// fullscreen the window
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Test::ShellSurfaceType::WlShell:
|
|
|
|
|
qobject_cast<ShellSurface*>(shellSurface.data())->setFullscreen();
|
|
|
|
|
break;
|
|
|
|
|
case Test::ShellSurfaceType::XdgShellV5:
|
|
|
|
|
qobject_cast<XdgShellSurface*>(shellSurface.data())->setFullscreen(true);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.wait());
|
|
|
|
|
if (decoMode == ServerSideDecoration::Mode::Server) {
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.count(), 2);
|
|
|
|
|
}
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.last().first().toSize(), QSize(screens()->size(0)));
|
|
|
|
|
// TODO: should switch to fullscreen once it's updated
|
|
|
|
|
QVERIFY(c->isFullScreen());
|
|
|
|
|
QCOMPARE(c->clientSize(), QSize(100, 50));
|
|
|
|
|
QVERIFY(geometryChangedSpy.isEmpty());
|
|
|
|
|
|
|
|
|
|
// render at the new size
|
|
|
|
|
Test::render(surface.data(), sizeChangeRequestedSpy.last().first().toSize(), Qt::red);
|
|
|
|
|
QVERIFY(geometryChangedSpy.wait());
|
|
|
|
|
QCOMPARE(geometryChangedSpy.count(), 1);
|
|
|
|
|
QVERIFY(c->isFullScreen());
|
|
|
|
|
QVERIFY(!c->isDecorated());
|
|
|
|
|
QCOMPARE(c->geometry(), QRect(QPoint(0, 0), sizeChangeRequestedSpy.last().first().toSize()));
|
|
|
|
|
sizeChangeRequestedSpy.clear();
|
|
|
|
|
|
|
|
|
|
// swap back to normal
|
|
|
|
|
switch (type) {
|
|
|
|
|
case Test::ShellSurfaceType::WlShell:
|
|
|
|
|
qobject_cast<ShellSurface*>(shellSurface.data())->setToplevel();
|
|
|
|
|
break;
|
|
|
|
|
case Test::ShellSurfaceType::XdgShellV5:
|
|
|
|
|
qobject_cast<XdgShellSurface*>(shellSurface.data())->setFullscreen(false);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
QVERIFY(fullscreenChangedSpy.wait());
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.count(), 1);
|
|
|
|
|
QEXPECT_FAIL("wlShell - deco", "With decoration incorrect geometry requested", Continue);
|
|
|
|
|
QEXPECT_FAIL("xdgShellV5 - deco", "With decoration incorrect geometry requested", Continue);
|
|
|
|
|
QCOMPARE(sizeChangeRequestedSpy.last().first().toSize(), QSize(100, 50));
|
|
|
|
|
// TODO: should switch to fullscreen once it's updated
|
|
|
|
|
QVERIFY(!c->isFullScreen());
|
|
|
|
|
QCOMPARE(c->isDecorated(), decoMode == ServerSideDecoration::Mode::Server);
|
2016-09-12 10:03:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
2016-09-14 12:08:47 +00:00
|
|
|
|
void TestShellClient::testWindowOpensLargerThanScreen_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testWindowOpensLargerThanScreen()
|
|
|
|
|
{
|
|
|
|
|
// this test creates a window which is as large as the screen, but is decorated
|
|
|
|
|
// the window should get resized to fit into the screen, BUG: 366632
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
|
|
|
|
QSignalSpy sizeChangeRequestedSpy(shellSurface.data(), SIGNAL(sizeChanged(QSize)));
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.isValid());
|
|
|
|
|
|
|
|
|
|
// create deco
|
|
|
|
|
QScopedPointer<ServerSideDecoration> deco(Test::waylandServerSideDecoration()->create(surface.data()));
|
|
|
|
|
QSignalSpy decoSpy(deco.data(), &ServerSideDecoration::modeChanged);
|
|
|
|
|
QVERIFY(decoSpy.isValid());
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
deco->requestMode(ServerSideDecoration::Mode::Server);
|
|
|
|
|
QVERIFY(decoSpy.wait());
|
|
|
|
|
QCOMPARE(deco->mode(), ServerSideDecoration::Mode::Server);
|
|
|
|
|
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), screens()->size(0), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->isActive());
|
|
|
|
|
QCOMPARE(c->clientSize(), screens()->size(0));
|
|
|
|
|
QVERIFY(c->isDecorated());
|
|
|
|
|
QEXPECT_FAIL("", "BUG 366632", Continue);
|
|
|
|
|
QVERIFY(sizeChangeRequestedSpy.wait());
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-11 14:24:10 +00:00
|
|
|
|
void TestShellClient::testHidden_data()
|
|
|
|
|
{
|
|
|
|
|
QTest::addColumn<Test::ShellSurfaceType>("type");
|
|
|
|
|
|
|
|
|
|
QTest::newRow("wlShell") << Test::ShellSurfaceType::WlShell;
|
|
|
|
|
QTest::newRow("xdgShellV5") << Test::ShellSurfaceType::XdgShellV5;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void TestShellClient::testHidden()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that when hiding window it doesn't get shown
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
QFETCH(Test::ShellSurfaceType, type);
|
|
|
|
|
QScopedPointer<QObject> shellSurface(Test::createShellSurface(type, surface.data()));
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->isActive());
|
|
|
|
|
QCOMPARE(workspace()->activeClient(), c);
|
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
|
|
|
|
QVERIFY(c->isShown(true));
|
|
|
|
|
|
|
|
|
|
c->hideClient(true);
|
|
|
|
|
QVERIFY(!c->isShown(true));
|
|
|
|
|
QVERIFY(!c->isActive());
|
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
|
|
|
|
|
|
|
|
|
// unhide again
|
|
|
|
|
c->hideClient(false);
|
|
|
|
|
QVERIFY(c->isShown(true));
|
|
|
|
|
QVERIFY(c->wantsInput());
|
|
|
|
|
QVERIFY(c->wantsTabFocus());
|
|
|
|
|
|
|
|
|
|
//QCOMPARE(workspace()->activeClient(), c);
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-27 13:59:08 +00:00
|
|
|
|
void TestShellClient::testDesktopFileName()
|
|
|
|
|
{
|
2016-10-31 13:16:23 +00:00
|
|
|
|
QIcon::setThemeName(QStringLiteral("breeze"));
|
2016-10-27 13:59:08 +00:00
|
|
|
|
// this test verifies that desktop file name is passed correctly to the window
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
// only xdg-shell as ShellSurface misses the setter
|
|
|
|
|
QScopedPointer<XdgShellSurface> shellSurface(qobject_cast<XdgShellSurface*>(Test::createShellSurface(Test::ShellSurfaceType::XdgShellV5, surface.data())));
|
|
|
|
|
shellSurface->setAppId(QByteArrayLiteral("org.kde.foo"));
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QCOMPARE(c->desktopFileName(), QByteArrayLiteral("org.kde.foo"));
|
|
|
|
|
// the desktop file does not exist, so icon should be generic Wayland
|
|
|
|
|
QCOMPARE(c->icon().name(), QStringLiteral("wayland"));
|
|
|
|
|
|
|
|
|
|
QSignalSpy desktopFileNameChangedSpy(c, &AbstractClient::desktopFileNameChanged);
|
|
|
|
|
QVERIFY(desktopFileNameChangedSpy.isValid());
|
|
|
|
|
QSignalSpy iconChangedSpy(c, &ShellClient::iconChanged);
|
|
|
|
|
QVERIFY(iconChangedSpy.isValid());
|
|
|
|
|
shellSurface->setAppId(QByteArrayLiteral("org.kde.bar"));
|
|
|
|
|
QVERIFY(desktopFileNameChangedSpy.wait());
|
|
|
|
|
QCOMPARE(c->desktopFileName(), QByteArrayLiteral("org.kde.bar"));
|
|
|
|
|
// icon should still be wayland
|
|
|
|
|
QCOMPARE(c->icon().name(), QStringLiteral("wayland"));
|
|
|
|
|
QVERIFY(iconChangedSpy.isEmpty());
|
|
|
|
|
|
|
|
|
|
const QString dfPath = QFINDTESTDATA("data/example.desktop");
|
|
|
|
|
shellSurface->setAppId(dfPath.toUtf8());
|
|
|
|
|
QVERIFY(desktopFileNameChangedSpy.wait());
|
|
|
|
|
QCOMPARE(iconChangedSpy.count(), 1);
|
|
|
|
|
QCOMPARE(QString::fromUtf8(c->desktopFileName()), dfPath);
|
|
|
|
|
QCOMPARE(c->icon().name(), QStringLiteral("kwin"));
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-31 14:50:14 +00:00
|
|
|
|
void TestShellClient::testCaptionSimplified()
|
|
|
|
|
{
|
|
|
|
|
// this test verifies that caption is properly trimmed
|
|
|
|
|
// see BUG 323798 comment #12
|
|
|
|
|
QScopedPointer<Surface> surface(Test::createSurface());
|
|
|
|
|
// only done for xdg-shell as ShellSurface misses the setter
|
|
|
|
|
QScopedPointer<XdgShellSurface> shellSurface(qobject_cast<XdgShellSurface*>(Test::createShellSurface(Test::ShellSurfaceType::XdgShellV5, surface.data())));
|
|
|
|
|
const QString origTitle = QString::fromUtf8(QByteArrayLiteral("Was tun, wenn Schüler Autismus haben?\342\200\250\342\200\250\342\200\250 – Marlies Hübner - Mozilla Firefox"));
|
|
|
|
|
shellSurface->setTitle(origTitle);
|
|
|
|
|
auto c = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
|
|
|
|
QVERIFY(c);
|
|
|
|
|
QVERIFY(c->caption() != origTitle);
|
|
|
|
|
QCOMPARE(c->caption(), origTitle.simplified());
|
|
|
|
|
}
|
|
|
|
|
|
2016-06-07 09:43:56 +00:00
|
|
|
|
WAYLANDTEST_MAIN(TestShellClient)
|
|
|
|
|
#include "shell_client_test.moc"
|