2016-03-14 09:23:52 +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"
|
2016-04-07 07:24:17 +00:00
|
|
|
#include "platform.h"
|
2016-03-14 09:23:52 +00:00
|
|
|
#include "debug_console.h"
|
|
|
|
#include "screens.h"
|
|
|
|
#include "shell_client.h"
|
|
|
|
#include "wayland_server.h"
|
|
|
|
#include "xcbutils.h"
|
|
|
|
|
|
|
|
#include <KWayland/Client/connection_thread.h>
|
|
|
|
#include <KWayland/Client/compositor.h>
|
|
|
|
#include <KWayland/Client/event_queue.h>
|
|
|
|
#include <KWayland/Client/registry.h>
|
|
|
|
#include <KWayland/Client/shell.h>
|
|
|
|
#include <KWayland/Client/shm_pool.h>
|
|
|
|
#include <KWayland/Client/surface.h>
|
|
|
|
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QRasterWindow>
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
static const QString s_socketName = QStringLiteral("wayland_test_kwin_debug_console-0");
|
|
|
|
|
|
|
|
class DebugConsoleTest : public QObject
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
private Q_SLOTS:
|
|
|
|
void initTestCase();
|
|
|
|
void topLevelTest_data();
|
|
|
|
void topLevelTest();
|
|
|
|
void testX11Client();
|
|
|
|
void testX11Unmanaged();
|
|
|
|
void testWaylandClient();
|
|
|
|
void testInternalWindow();
|
|
|
|
};
|
|
|
|
|
|
|
|
void DebugConsoleTest::initTestCase()
|
|
|
|
{
|
|
|
|
qRegisterMetaType<KWin::ShellClient*>();
|
|
|
|
qRegisterMetaType<KWin::AbstractClient*>();
|
|
|
|
QSignalSpy workspaceCreatedSpy(kwinApp(), &Application::workspaceCreated);
|
|
|
|
QVERIFY(workspaceCreatedSpy.isValid());
|
2016-04-07 06:28:35 +00:00
|
|
|
kwinApp()->platform()->setInitialWindowSize(QSize(1280, 1024));
|
|
|
|
QMetaObject::invokeMethod(kwinApp()->platform(), "setOutputCount", Qt::DirectConnection, Q_ARG(int, 2));
|
2016-03-14 09:23:52 +00:00
|
|
|
waylandServer()->init(s_socketName.toLocal8Bit());
|
|
|
|
|
|
|
|
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));
|
|
|
|
setenv("QT_QPA_PLATFORM", "wayland", true);
|
|
|
|
waylandServer()->initWorkspace();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleTest::topLevelTest_data()
|
|
|
|
{
|
|
|
|
QTest::addColumn<int>("row");
|
|
|
|
QTest::addColumn<int>("column");
|
|
|
|
QTest::addColumn<bool>("expectedValid");
|
|
|
|
|
|
|
|
// this tests various combinations of row/column on the top level whether they are valid
|
|
|
|
// valid are rows 0-4 with column 0, everything else is invalid
|
|
|
|
QTest::newRow("0/0") << 0 << 0 << true;
|
|
|
|
QTest::newRow("0/1") << 0 << 1 << false;
|
|
|
|
QTest::newRow("0/3") << 0 << 3 << false;
|
|
|
|
QTest::newRow("1/0") << 1 << 0 << true;
|
|
|
|
QTest::newRow("1/1") << 1 << 1 << false;
|
|
|
|
QTest::newRow("1/3") << 1 << 3 << false;
|
|
|
|
QTest::newRow("2/0") << 2 << 0 << true;
|
|
|
|
QTest::newRow("3/0") << 3 << 0 << true;
|
|
|
|
QTest::newRow("4/0") << 4 << 0 << false;
|
|
|
|
QTest::newRow("100/0") << 4 << 0 << false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleTest::topLevelTest()
|
|
|
|
{
|
|
|
|
DebugConsoleModel model;
|
|
|
|
QCOMPARE(model.rowCount(QModelIndex()), 4);
|
|
|
|
QCOMPARE(model.columnCount(QModelIndex()), 2);
|
|
|
|
QFETCH(int, row);
|
|
|
|
QFETCH(int, column);
|
|
|
|
const QModelIndex index = model.index(row, column, QModelIndex());
|
|
|
|
QTEST(index.isValid(), "expectedValid");
|
|
|
|
if (index.isValid()) {
|
|
|
|
QVERIFY(!model.parent(index).isValid());
|
|
|
|
QVERIFY(model.data(index, Qt::DisplayRole).isValid());
|
|
|
|
QCOMPARE(model.data(index, Qt::DisplayRole).userType(), int(QMetaType::QString));
|
|
|
|
for (int i = Qt::DecorationRole; i <= Qt::UserRole; i++) {
|
|
|
|
QVERIFY(!model.data(index, i).isValid());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleTest::testX11Client()
|
|
|
|
{
|
|
|
|
DebugConsoleModel model;
|
|
|
|
QModelIndex x11TopLevelIndex = model.index(0, 0, QModelIndex());
|
|
|
|
QVERIFY(x11TopLevelIndex.isValid());
|
|
|
|
// we don't have any windows yet
|
|
|
|
QCOMPARE(model.rowCount(x11TopLevelIndex), 0);
|
|
|
|
QVERIFY(!model.hasChildren(x11TopLevelIndex));
|
|
|
|
// child index must be invalid
|
|
|
|
QVERIFY(!model.index(0, 0, x11TopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 1, x11TopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, x11TopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, x11TopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// start glxgears, to get a window, which should be added to the model
|
|
|
|
QSignalSpy rowsInsertedSpy(&model, &QAbstractItemModel::rowsInserted);
|
|
|
|
QVERIFY(rowsInsertedSpy.isValid());
|
|
|
|
|
|
|
|
QProcess glxgears;
|
|
|
|
glxgears.start(QStringLiteral("glxgears"));
|
|
|
|
QVERIFY(glxgears.waitForStarted());
|
|
|
|
|
|
|
|
QVERIFY(rowsInsertedSpy.wait());
|
|
|
|
QCOMPARE(rowsInsertedSpy.count(), 1);
|
|
|
|
QVERIFY(model.hasChildren(x11TopLevelIndex));
|
|
|
|
QCOMPARE(model.rowCount(x11TopLevelIndex), 1);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(0).value<QModelIndex>(), x11TopLevelIndex);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
QModelIndex clientIndex = model.index(0, 0, x11TopLevelIndex);
|
|
|
|
QVERIFY(clientIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(clientIndex), x11TopLevelIndex);
|
|
|
|
QVERIFY(model.hasChildren(clientIndex));
|
|
|
|
QVERIFY(model.rowCount(clientIndex) != 0);
|
|
|
|
QCOMPARE(model.columnCount(clientIndex), 2);
|
|
|
|
// other indexes are still invalid
|
|
|
|
QVERIFY(!model.index(0, 1, x11TopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, x11TopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, x11TopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// the clientIndex has children and those are properties
|
|
|
|
for (int i = 0; i < model.rowCount(clientIndex); i++) {
|
|
|
|
const QModelIndex propNameIndex = model.index(i, 0, clientIndex);
|
|
|
|
QVERIFY(propNameIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propNameIndex), clientIndex);
|
|
|
|
QVERIFY(!model.hasChildren(propNameIndex));
|
|
|
|
QVERIFY(!model.index(0, 0, propNameIndex).isValid());
|
|
|
|
QVERIFY(model.data(propNameIndex, Qt::DisplayRole).isValid());
|
|
|
|
QCOMPARE(model.data(propNameIndex, Qt::DisplayRole).userType(), int(QMetaType::QString));
|
|
|
|
|
|
|
|
// and the value
|
|
|
|
const QModelIndex propValueIndex = model.index(i, 1, clientIndex);
|
|
|
|
QVERIFY(propValueIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propValueIndex), clientIndex);
|
|
|
|
QVERIFY(!model.index(0, 0, propValueIndex).isValid());
|
|
|
|
QVERIFY(!model.hasChildren(propValueIndex));
|
|
|
|
// TODO: how to test whether the values actually work?
|
|
|
|
|
|
|
|
// and on third column we should not get an index any more
|
|
|
|
QVERIFY(!model.index(i, 2, clientIndex).isValid());
|
|
|
|
}
|
|
|
|
// row after count should be invalid
|
|
|
|
QVERIFY(!model.index(model.rowCount(clientIndex), 0, clientIndex).isValid());
|
|
|
|
|
|
|
|
// creating a second model should be initialized directly with the X11 child
|
|
|
|
DebugConsoleModel model2;
|
|
|
|
QVERIFY(model2.hasChildren(model2.index(0, 0, QModelIndex())));
|
|
|
|
|
|
|
|
// now close the window again, it should be removed from the model
|
|
|
|
QSignalSpy rowsRemovedSpy(&model, &QAbstractItemModel::rowsRemoved);
|
|
|
|
QVERIFY(rowsRemovedSpy.isValid());
|
|
|
|
|
|
|
|
glxgears.terminate();
|
|
|
|
QVERIFY(glxgears.waitForFinished());
|
|
|
|
|
|
|
|
QVERIFY(rowsRemovedSpy.wait());
|
|
|
|
QCOMPARE(rowsRemovedSpy.count(), 1);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().first().value<QModelIndex>(), x11TopLevelIndex);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
// the child should be gone again
|
|
|
|
QVERIFY(!model.hasChildren(x11TopLevelIndex));
|
|
|
|
QVERIFY(!model2.hasChildren(model2.index(0, 0, QModelIndex())));
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleTest::testX11Unmanaged()
|
|
|
|
{
|
|
|
|
DebugConsoleModel model;
|
|
|
|
QModelIndex unmanagedTopLevelIndex = model.index(1, 0, QModelIndex());
|
|
|
|
QVERIFY(unmanagedTopLevelIndex.isValid());
|
|
|
|
// we don't have any windows yet
|
|
|
|
QCOMPARE(model.rowCount(unmanagedTopLevelIndex), 0);
|
|
|
|
QVERIFY(!model.hasChildren(unmanagedTopLevelIndex));
|
|
|
|
// child index must be invalid
|
|
|
|
QVERIFY(!model.index(0, 0, unmanagedTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 1, unmanagedTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, unmanagedTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, unmanagedTopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// we need to create an unmanaged window
|
|
|
|
QSignalSpy rowsInsertedSpy(&model, &QAbstractItemModel::rowsInserted);
|
|
|
|
QVERIFY(rowsInsertedSpy.isValid());
|
|
|
|
|
|
|
|
// let's create an override redirect window
|
|
|
|
const uint32_t values[] = {true};
|
|
|
|
Xcb::Window window(QRect(0, 0, 10, 10), XCB_CW_OVERRIDE_REDIRECT, values);
|
|
|
|
window.map();
|
|
|
|
|
|
|
|
QVERIFY(rowsInsertedSpy.wait());
|
|
|
|
QCOMPARE(rowsInsertedSpy.count(), 1);
|
|
|
|
QVERIFY(model.hasChildren(unmanagedTopLevelIndex));
|
|
|
|
QCOMPARE(model.rowCount(unmanagedTopLevelIndex), 1);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(0).value<QModelIndex>(), unmanagedTopLevelIndex);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
QModelIndex clientIndex = model.index(0, 0, unmanagedTopLevelIndex);
|
|
|
|
QVERIFY(clientIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(clientIndex), unmanagedTopLevelIndex);
|
|
|
|
QVERIFY(model.hasChildren(clientIndex));
|
|
|
|
QVERIFY(model.rowCount(clientIndex) != 0);
|
|
|
|
QCOMPARE(model.columnCount(clientIndex), 2);
|
|
|
|
// other indexes are still invalid
|
|
|
|
QVERIFY(!model.index(0, 1, unmanagedTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, unmanagedTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, unmanagedTopLevelIndex).isValid());
|
|
|
|
|
|
|
|
QCOMPARE(model.data(clientIndex, Qt::DisplayRole).toString(), QString::number(window));
|
|
|
|
|
|
|
|
// the clientIndex has children and those are properties
|
|
|
|
for (int i = 0; i < model.rowCount(clientIndex); i++) {
|
|
|
|
const QModelIndex propNameIndex = model.index(i, 0, clientIndex);
|
|
|
|
QVERIFY(propNameIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propNameIndex), clientIndex);
|
|
|
|
QVERIFY(!model.hasChildren(propNameIndex));
|
|
|
|
QVERIFY(!model.index(0, 0, propNameIndex).isValid());
|
|
|
|
QVERIFY(model.data(propNameIndex, Qt::DisplayRole).isValid());
|
|
|
|
QCOMPARE(model.data(propNameIndex, Qt::DisplayRole).userType(), int(QMetaType::QString));
|
|
|
|
|
|
|
|
// and the value
|
|
|
|
const QModelIndex propValueIndex = model.index(i, 1, clientIndex);
|
|
|
|
QVERIFY(propValueIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propValueIndex), clientIndex);
|
|
|
|
QVERIFY(!model.index(0, 0, propValueIndex).isValid());
|
|
|
|
QVERIFY(!model.hasChildren(propValueIndex));
|
|
|
|
// TODO: how to test whether the values actually work?
|
|
|
|
|
|
|
|
// and on third column we should not get an index any more
|
|
|
|
QVERIFY(!model.index(i, 2, clientIndex).isValid());
|
|
|
|
}
|
|
|
|
// row after count should be invalid
|
|
|
|
QVERIFY(!model.index(model.rowCount(clientIndex), 0, clientIndex).isValid());
|
|
|
|
|
|
|
|
// creating a second model should be initialized directly with the X11 child
|
|
|
|
DebugConsoleModel model2;
|
|
|
|
QVERIFY(model2.hasChildren(model2.index(1, 0, QModelIndex())));
|
|
|
|
|
|
|
|
// now close the window again, it should be removed from the model
|
|
|
|
QSignalSpy rowsRemovedSpy(&model, &QAbstractItemModel::rowsRemoved);
|
|
|
|
QVERIFY(rowsRemovedSpy.isValid());
|
|
|
|
|
|
|
|
window.unmap();
|
|
|
|
|
|
|
|
QVERIFY(rowsRemovedSpy.wait());
|
|
|
|
QCOMPARE(rowsRemovedSpy.count(), 1);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().first().value<QModelIndex>(), unmanagedTopLevelIndex);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
// the child should be gone again
|
|
|
|
QVERIFY(!model.hasChildren(unmanagedTopLevelIndex));
|
|
|
|
QVERIFY(!model2.hasChildren(model2.index(1, 0, QModelIndex())));
|
|
|
|
}
|
|
|
|
|
|
|
|
void DebugConsoleTest::testWaylandClient()
|
|
|
|
{
|
|
|
|
DebugConsoleModel model;
|
|
|
|
QModelIndex waylandTopLevelIndex = model.index(2, 0, QModelIndex());
|
|
|
|
QVERIFY(waylandTopLevelIndex.isValid());
|
|
|
|
|
|
|
|
// we don't have any windows yet
|
|
|
|
QCOMPARE(model.rowCount(waylandTopLevelIndex), 0);
|
|
|
|
QVERIFY(!model.hasChildren(waylandTopLevelIndex));
|
|
|
|
// child index must be invalid
|
|
|
|
QVERIFY(!model.index(0, 0, waylandTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 1, waylandTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, waylandTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, waylandTopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// we need to create a wayland window
|
|
|
|
QSignalSpy rowsInsertedSpy(&model, &QAbstractItemModel::rowsInserted);
|
|
|
|
QVERIFY(rowsInsertedSpy.isValid());
|
|
|
|
|
|
|
|
// create our connection
|
|
|
|
using namespace KWayland::Client;
|
|
|
|
// setup connection
|
|
|
|
QScopedPointer<ConnectionThread> connection(new ConnectionThread);
|
|
|
|
QSignalSpy connectedSpy(connection.data(), &ConnectionThread::connected);
|
|
|
|
QVERIFY(connectedSpy.isValid());
|
|
|
|
connection->setSocketName(s_socketName);
|
|
|
|
|
|
|
|
QScopedPointer<QThread> thread(new QThread);
|
|
|
|
connection->moveToThread(thread.data());
|
|
|
|
thread->start();
|
|
|
|
|
|
|
|
connection->initConnection();
|
|
|
|
QVERIFY(connectedSpy.wait());
|
|
|
|
|
|
|
|
QScopedPointer<EventQueue> queue(new EventQueue);
|
|
|
|
QVERIFY(!queue->isValid());
|
|
|
|
queue->setup(connection.data());
|
|
|
|
QVERIFY(queue->isValid());
|
|
|
|
|
|
|
|
Registry registry;
|
|
|
|
registry.setEventQueue(queue.data());
|
|
|
|
QSignalSpy interfacesAnnouncedSpy(®istry, &Registry::interfacesAnnounced);
|
|
|
|
QVERIFY(interfacesAnnouncedSpy.isValid());
|
|
|
|
registry.create(connection.data());
|
|
|
|
QVERIFY(registry.isValid());
|
|
|
|
registry.setup();
|
|
|
|
QVERIFY(interfacesAnnouncedSpy.wait());
|
|
|
|
|
|
|
|
QScopedPointer<Compositor> compositor(registry.createCompositor(registry.interface(Registry::Interface::Compositor).name, registry.interface(Registry::Interface::Compositor).version));
|
|
|
|
QVERIFY(compositor->isValid());
|
|
|
|
QScopedPointer<ShmPool> shm(registry.createShmPool(registry.interface(Registry::Interface::Shm).name, registry.interface(Registry::Interface::Shm).version));
|
|
|
|
QVERIFY(shm->isValid());
|
|
|
|
QScopedPointer<Shell> shell(registry.createShell(registry.interface(Registry::Interface::Shell).name, registry.interface(Registry::Interface::Shell).version));
|
|
|
|
QVERIFY(shell->isValid());
|
|
|
|
|
|
|
|
// create the Surface and ShellSurface
|
|
|
|
QScopedPointer<Surface> surface(compositor->createSurface());
|
|
|
|
QVERIFY(surface->isValid());
|
|
|
|
QScopedPointer<ShellSurface> shellSurface(shell->createSurface(surface.data()));
|
|
|
|
QVERIFY(shellSurface->isValid());
|
|
|
|
QImage img(10, 10, QImage::Format_ARGB32);
|
|
|
|
img.fill(Qt::red);
|
|
|
|
surface->attachBuffer(shm->createBuffer(img));
|
|
|
|
surface->damage(QRect(0, 0, 10, 10));
|
|
|
|
surface->commit(Surface::CommitFlag::None);
|
|
|
|
|
|
|
|
// now we have the window, it should be added to our model
|
|
|
|
QVERIFY(rowsInsertedSpy.wait());
|
|
|
|
QCOMPARE(rowsInsertedSpy.count(), 1);
|
|
|
|
|
|
|
|
QVERIFY(model.hasChildren(waylandTopLevelIndex));
|
|
|
|
QCOMPARE(model.rowCount(waylandTopLevelIndex), 1);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(0).value<QModelIndex>(), waylandTopLevelIndex);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
QModelIndex clientIndex = model.index(0, 0, waylandTopLevelIndex);
|
|
|
|
QVERIFY(clientIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(clientIndex), waylandTopLevelIndex);
|
|
|
|
QVERIFY(model.hasChildren(clientIndex));
|
|
|
|
QVERIFY(model.rowCount(clientIndex) != 0);
|
|
|
|
QCOMPARE(model.columnCount(clientIndex), 2);
|
|
|
|
// other indexes are still invalid
|
|
|
|
QVERIFY(!model.index(0, 1, waylandTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(0, 2, waylandTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(1, 0, waylandTopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// the clientIndex has children and those are properties
|
|
|
|
for (int i = 0; i < model.rowCount(clientIndex); i++) {
|
|
|
|
const QModelIndex propNameIndex = model.index(i, 0, clientIndex);
|
|
|
|
QVERIFY(propNameIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propNameIndex), clientIndex);
|
|
|
|
QVERIFY(!model.hasChildren(propNameIndex));
|
|
|
|
QVERIFY(!model.index(0, 0, propNameIndex).isValid());
|
|
|
|
QVERIFY(model.data(propNameIndex, Qt::DisplayRole).isValid());
|
|
|
|
QCOMPARE(model.data(propNameIndex, Qt::DisplayRole).userType(), int(QMetaType::QString));
|
|
|
|
|
|
|
|
// and the value
|
|
|
|
const QModelIndex propValueIndex = model.index(i, 1, clientIndex);
|
|
|
|
QVERIFY(propValueIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propValueIndex), clientIndex);
|
|
|
|
QVERIFY(!model.index(0, 0, propValueIndex).isValid());
|
|
|
|
QVERIFY(!model.hasChildren(propValueIndex));
|
|
|
|
// TODO: how to test whether the values actually work?
|
|
|
|
|
|
|
|
// and on third column we should not get an index any more
|
|
|
|
QVERIFY(!model.index(i, 2, clientIndex).isValid());
|
|
|
|
}
|
|
|
|
// row after count should be invalid
|
|
|
|
QVERIFY(!model.index(model.rowCount(clientIndex), 0, clientIndex).isValid());
|
|
|
|
|
|
|
|
// creating a second model should be initialized directly with the X11 child
|
|
|
|
DebugConsoleModel model2;
|
|
|
|
QVERIFY(model2.hasChildren(model2.index(2, 0, QModelIndex())));
|
|
|
|
|
|
|
|
// now close the window again, it should be removed from the model
|
|
|
|
QSignalSpy rowsRemovedSpy(&model, &QAbstractItemModel::rowsRemoved);
|
|
|
|
QVERIFY(rowsRemovedSpy.isValid());
|
|
|
|
|
|
|
|
surface->attachBuffer(Buffer::Ptr());
|
|
|
|
surface->commit(Surface::CommitFlag::None);
|
|
|
|
shellSurface.reset();
|
|
|
|
connection->flush();
|
|
|
|
qDebug() << rowsRemovedSpy.count();
|
|
|
|
QEXPECT_FAIL("", "Deleting a ShellSurface does not result in the server removing the ShellClient", Continue);
|
|
|
|
QVERIFY(rowsRemovedSpy.wait());
|
|
|
|
// also destroy the connection to ensure the ShellSurface goes away
|
2016-03-29 09:08:04 +00:00
|
|
|
surface.reset();
|
2016-03-14 09:23:52 +00:00
|
|
|
shell.reset();
|
|
|
|
shm.reset();
|
|
|
|
compositor.reset();
|
|
|
|
connection.take()->deleteLater();
|
|
|
|
thread->quit();
|
|
|
|
QVERIFY(thread->wait());
|
|
|
|
|
|
|
|
QVERIFY(rowsRemovedSpy.wait());
|
|
|
|
QCOMPARE(rowsRemovedSpy.count(), 1);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().first().value<QModelIndex>(), waylandTopLevelIndex);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(1).value<int>(), 0);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().at(2).value<int>(), 0);
|
|
|
|
|
|
|
|
// the child should be gone again
|
|
|
|
QVERIFY(!model.hasChildren(waylandTopLevelIndex));
|
|
|
|
QVERIFY(!model2.hasChildren(model2.index(2, 0, QModelIndex())));
|
|
|
|
}
|
|
|
|
|
|
|
|
class HelperWindow : public QRasterWindow
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
|
|
|
public:
|
|
|
|
HelperWindow() : QRasterWindow(nullptr) {}
|
|
|
|
~HelperWindow() = default;
|
|
|
|
|
|
|
|
Q_SIGNALS:
|
|
|
|
void entered();
|
|
|
|
void left();
|
|
|
|
void mouseMoved(const QPoint &global);
|
|
|
|
void mousePressed();
|
|
|
|
void mouseReleased();
|
|
|
|
void wheel();
|
|
|
|
void keyPressed();
|
|
|
|
void keyReleased();
|
|
|
|
|
|
|
|
protected:
|
|
|
|
void paintEvent(QPaintEvent *event) override {
|
|
|
|
Q_UNUSED(event)
|
|
|
|
QPainter p(this);
|
|
|
|
p.fillRect(0, 0, width(), height(), Qt::red);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
void DebugConsoleTest::testInternalWindow()
|
|
|
|
{
|
|
|
|
DebugConsoleModel model;
|
|
|
|
QModelIndex internalTopLevelIndex = model.index(3, 0, QModelIndex());
|
|
|
|
QVERIFY(internalTopLevelIndex.isValid());
|
|
|
|
|
|
|
|
// there might already be some internal windows, so we cannot reliable test whether there are children
|
|
|
|
// given that we just test whether adding a window works.
|
|
|
|
|
|
|
|
QSignalSpy rowsInsertedSpy(&model, &QAbstractItemModel::rowsInserted);
|
|
|
|
QVERIFY(rowsInsertedSpy.isValid());
|
|
|
|
|
|
|
|
QScopedPointer<HelperWindow> w(new HelperWindow);
|
|
|
|
w->setGeometry(0, 0, 100, 100);
|
|
|
|
w->show();
|
|
|
|
|
|
|
|
QVERIFY(rowsInsertedSpy.wait());
|
|
|
|
QCOMPARE(rowsInsertedSpy.count(), 1);
|
|
|
|
QCOMPARE(rowsInsertedSpy.first().first().value<QModelIndex>(), internalTopLevelIndex);
|
|
|
|
|
|
|
|
QModelIndex clientIndex = model.index(rowsInsertedSpy.first().last().toInt(), 0, internalTopLevelIndex);
|
|
|
|
QVERIFY(clientIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(clientIndex), internalTopLevelIndex);
|
|
|
|
QVERIFY(model.hasChildren(clientIndex));
|
|
|
|
QVERIFY(model.rowCount(clientIndex) != 0);
|
|
|
|
QCOMPARE(model.columnCount(clientIndex), 2);
|
|
|
|
// other indexes are still invalid
|
|
|
|
QVERIFY(!model.index(rowsInsertedSpy.first().last().toInt(), 1, internalTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(rowsInsertedSpy.first().last().toInt(), 2, internalTopLevelIndex).isValid());
|
|
|
|
QVERIFY(!model.index(rowsInsertedSpy.first().last().toInt() + 1, 0, internalTopLevelIndex).isValid());
|
|
|
|
|
|
|
|
// the wayland shell client top level should not have gained this window
|
|
|
|
QVERIFY(!model.hasChildren(model.index(2, 0, QModelIndex())));
|
|
|
|
|
|
|
|
// the clientIndex has children and those are properties
|
|
|
|
for (int i = 0; i < model.rowCount(clientIndex); i++) {
|
|
|
|
const QModelIndex propNameIndex = model.index(i, 0, clientIndex);
|
|
|
|
QVERIFY(propNameIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propNameIndex), clientIndex);
|
|
|
|
QVERIFY(!model.hasChildren(propNameIndex));
|
|
|
|
QVERIFY(!model.index(0, 0, propNameIndex).isValid());
|
|
|
|
QVERIFY(model.data(propNameIndex, Qt::DisplayRole).isValid());
|
|
|
|
QCOMPARE(model.data(propNameIndex, Qt::DisplayRole).userType(), int(QMetaType::QString));
|
|
|
|
|
|
|
|
// and the value
|
|
|
|
const QModelIndex propValueIndex = model.index(i, 1, clientIndex);
|
|
|
|
QVERIFY(propValueIndex.isValid());
|
|
|
|
QCOMPARE(model.parent(propValueIndex), clientIndex);
|
|
|
|
QVERIFY(!model.index(0, 0, propValueIndex).isValid());
|
|
|
|
QVERIFY(!model.hasChildren(propValueIndex));
|
|
|
|
// TODO: how to test whether the values actually work?
|
|
|
|
|
|
|
|
// and on third column we should not get an index any more
|
|
|
|
QVERIFY(!model.index(i, 2, clientIndex).isValid());
|
|
|
|
}
|
|
|
|
// row after count should be invalid
|
|
|
|
QVERIFY(!model.index(model.rowCount(clientIndex), 0, clientIndex).isValid());
|
|
|
|
|
|
|
|
// now close the window again, it should be removed from the model
|
|
|
|
QSignalSpy rowsRemovedSpy(&model, &QAbstractItemModel::rowsRemoved);
|
|
|
|
QVERIFY(rowsRemovedSpy.isValid());
|
|
|
|
|
|
|
|
w->hide();
|
|
|
|
w.reset();
|
|
|
|
|
|
|
|
QVERIFY(rowsRemovedSpy.wait());
|
|
|
|
QCOMPARE(rowsRemovedSpy.count(), 1);
|
|
|
|
QCOMPARE(rowsRemovedSpy.first().first().value<QModelIndex>(), internalTopLevelIndex);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
WAYLANDTEST_MAIN(KWin::DebugConsoleTest)
|
|
|
|
#include "debug_console_test.moc"
|