/* SPDX-FileCopyrightText: 2021 Vlad Zahorodnii SPDX-License-Identifier: GPL-2.0-or-later */ #include "kwin_wayland_test.h" #include "core/output.h" #include "core/outputbackend.h" #include "core/outputconfiguration.h" #include "pointer_input.h" #include "wayland_server.h" #include "window.h" #include "workspace.h" #include using namespace std::chrono_literals; namespace KWin { static const QString s_socketName = QStringLiteral("wayland_test_output_changes-0"); class OutputChangesTest : public QObject { Q_OBJECT private Q_SLOTS: void initTestCase(); void init(); void cleanup(); void testWindowSticksToOutputAfterOutputIsDisabled(); void testWindowSticksToOutputAfterAnotherOutputIsDisabled(); void testWindowSticksToOutputAfterOutputIsMoved(); void testWindowSticksToOutputAfterOutputsAreSwappedLeftToRight(); void testWindowSticksToOutputAfterOutputsAreSwappedRightToLeft(); void testWindowRestoredAfterEnablingOutput(); void testMaximizedWindowRestoredAfterEnablingOutput(); void testFullScreenWindowRestoredAfterEnablingOutput(); void testWindowRestoredAfterChangingScale(); void testMaximizeStateRestoredAfterEnablingOutput(); void testWindowNotRestoredAfterMovingWindowAndEnablingOutput(); void testLaptopLidClosed(); }; void OutputChangesTest::initTestCase() { qRegisterMetaType(); QSignalSpy applicationStartedSpy(kwinApp(), &Application::started); QVERIFY(waylandServer()->init(s_socketName)); Test::setOutputConfig({ QRect(0, 0, 1280, 1024), QRect(1280, 0, 1280, 1024), }); kwinApp()->start(); QVERIFY(applicationStartedSpy.wait()); const auto outputs = workspace()->outputs(); QCOMPARE(outputs.count(), 2); QCOMPARE(outputs[0]->geometry(), QRect(0, 0, 1280, 1024)); QCOMPARE(outputs[1]->geometry(), QRect(1280, 0, 1280, 1024)); } void OutputChangesTest::init() { Test::setOutputConfig({ QRect(0, 0, 1280, 1024), QRect(1280, 0, 1280, 1024), }); QVERIFY(Test::setupWaylandConnection()); workspace()->setActiveOutput(QPoint(640, 512)); input()->pointer()->warp(QPoint(640, 512)); } void OutputChangesTest::cleanup() { Test::destroyWaylandConnection(); } void OutputChangesTest::testWindowSticksToOutputAfterOutputIsDisabled() { auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to some predefined position so the test is more robust. window->move(QPoint(42, 67)); QCOMPARE(window->frameGeometry(), QRect(42, 67, 100, 50)); // Disable the output where the window is on. OutputConfiguration config; { auto changeSet = config.changeSet(outputs[0]); changeSet->enabled = false; } workspace()->applyOutputConfiguration(config); // The window will be sent to the second output, which is at (1280, 0). QCOMPARE(window->frameGeometry(), QRect(1280 + 42, 0 + 67, 100, 50)); } void OutputChangesTest::testWindowSticksToOutputAfterAnotherOutputIsDisabled() { auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the second output. window->move(QPoint(1280 + 42, 67)); QCOMPARE(window->frameGeometry(), QRect(1280 + 42, 67, 100, 50)); // Disable the first output. OutputConfiguration config; { auto changeSet = config.changeSet(outputs[0]); changeSet->enabled = false; } { auto changeSet = config.changeSet(outputs[1]); changeSet->pos = QPoint(0, 0); } workspace()->applyOutputConfiguration(config); // The position of the window relative to its output should remain the same. QCOMPARE(window->frameGeometry(), QRect(42, 67, 100, 50)); } void OutputChangesTest::testWindowSticksToOutputAfterOutputIsMoved() { auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to some predefined position so the test is more robust. window->move(QPoint(42, 67)); QCOMPARE(window->frameGeometry(), QRect(42, 67, 100, 50)); // Disable the first output. OutputConfiguration config; { auto changeSet = config.changeSet(outputs[0]); changeSet->pos = QPoint(-10, 20); } workspace()->applyOutputConfiguration(config); // The position of the window relative to its output should remain the same. QCOMPARE(window->frameGeometry(), QRect(-10 + 42, 20 + 67, 100, 50)); } void OutputChangesTest::testWindowSticksToOutputAfterOutputsAreSwappedLeftToRight() { // This test verifies that a window placed on the left monitor sticks // to that monitor even after the monitors are swapped horizontally. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the left output. window->move(QPointF(0, 0)); QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->frameGeometry(), QRectF(0, 0, 100, 50)); // Swap outputs. OutputConfiguration config; { auto changeSet1 = config.changeSet(outputs[0]); changeSet1->pos = QPoint(1280, 0); auto changeSet2 = config.changeSet(outputs[1]); changeSet2->pos = QPoint(0, 0); } workspace()->applyOutputConfiguration(config); // The window should be still on its original output. QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 100, 50)); } void OutputChangesTest::testWindowSticksToOutputAfterOutputsAreSwappedRightToLeft() { // This test verifies that a window placed on the right monitor sticks // to that monitor even after the monitors are swapped horizontally. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the right output. window->move(QPointF(1280, 0)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 100, 50)); // Swap outputs. OutputConfiguration config; { auto changeSet1 = config.changeSet(outputs[0]); changeSet1->pos = QPoint(1280, 0); auto changeSet2 = config.changeSet(outputs[1]); changeSet2->pos = QPoint(0, 0); } workspace()->applyOutputConfiguration(config); // The window should be still on its original output. QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->frameGeometry(), QRectF(0, 0, 100, 50)); } void OutputChangesTest::testWindowRestoredAfterEnablingOutput() { // This test verifies that a window will be moved back to its original output when it's hotplugged. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the right output. window->move(QPointF(1280 + 50, 100)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->frameGeometry(), QRectF(1280 + 50, 100, 100, 50)); // Disable the right output. OutputConfiguration config1; { auto changeSet = config1.changeSet(outputs[1]); changeSet->enabled = false; } workspace()->applyOutputConfiguration(config1); // The window will be moved to the left monitor. QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->frameGeometry(), QRectF(50, 100, 100, 50)); // Enable the right monitor. OutputConfiguration config2; { auto changeSet = config2.changeSet(outputs[1]); changeSet->enabled = true; } workspace()->applyOutputConfiguration(config2); // The window will be moved back to the right monitor. QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->frameGeometry(), QRectF(1280 + 50, 100, 100, 50)); } void OutputChangesTest::testWindowNotRestoredAfterMovingWindowAndEnablingOutput() { // This test verifies that a window won't be moved to its original output when it's // hotplugged because the window was moved manually by the user. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the right output. window->move(QPointF(1280 + 50, 100)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->frameGeometry(), QRectF(1280 + 50, 100, 100, 50)); // Disable the right output. OutputConfiguration config1; { auto changeSet = config1.changeSet(outputs[1]); changeSet->enabled = false; } workspace()->applyOutputConfiguration(config1); // The window will be moved to the left monitor. QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->frameGeometry(), QRectF(50, 100, 100, 50)); // Pretend that the user moved the window. workspace()->slotWindowMove(); QVERIFY(window->isInteractiveMove()); window->keyPressEvent(Qt::Key_Right); window->keyPressEvent(Qt::Key_Enter); QCOMPARE(window->frameGeometry(), QRectF(58, 100, 100, 50)); // Enable the right monitor. OutputConfiguration config2; { auto changeSet = config2.changeSet(outputs[1]); changeSet->enabled = true; } workspace()->applyOutputConfiguration(config2); // The window is still on the left monitor because user manually moved it. QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->frameGeometry(), QRectF(58, 100, 100, 50)); } void OutputChangesTest::testMaximizedWindowRestoredAfterEnablingOutput() { // This test verifies that a maximized window will be moved to its original // output when it's re-enabled. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // kwin will send a configure event with the actived state. QSignalSpy toplevelConfigureRequestedSpy(shellSurface.get(), &Test::XdgToplevel::configureRequested); QSignalSpy surfaceConfigureRequestedSpy(shellSurface->xdgSurface(), &Test::XdgSurface::configureRequested); QVERIFY(surfaceConfigureRequestedSpy.wait()); // Move the window to the right monitor and make it maximized. QSignalSpy frameGeometryChangedSpy(window, &Window::frameGeometryChanged); window->move(QPointF(1280 + 50, 100)); window->maximize(MaximizeFull); QVERIFY(surfaceConfigureRequestedSpy.wait()); QCOMPARE(toplevelConfigureRequestedSpy.last().at(0).value(), QSize(1280, 1024)); shellSurface->xdgSurface()->ack_configure(surfaceConfigureRequestedSpy.last().at(0).value()); Test::render(surface.get(), QSize(1280, 1024), Qt::blue); QVERIFY(frameGeometryChangedSpy.wait()); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->maximizeMode(), MaximizeFull); QCOMPARE(window->requestedMaximizeMode(), MaximizeFull); QCOMPARE(window->geometryRestore(), QRectF(1280 + 50, 100, 100, 50)); // Disable the right output. OutputConfiguration config1; { auto changeSet = config1.changeSet(outputs[1]); changeSet->enabled = false; } workspace()->applyOutputConfiguration(config1); // The window will be moved to the left monitor, the geometry restore will be updated too. QCOMPARE(window->frameGeometry(), QRectF(0, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(0, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->maximizeMode(), MaximizeFull); QCOMPARE(window->requestedMaximizeMode(), MaximizeFull); QCOMPARE(window->geometryRestore(), QRectF(50, 100, 100, 50)); // Enable the right monitor. OutputConfiguration config2; { auto changeSet = config2.changeSet(outputs[1]); changeSet->enabled = true; } workspace()->applyOutputConfiguration(config2); // The window will be moved back to the right monitor, the geometry restore will be updated too. QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->maximizeMode(), MaximizeFull); QCOMPARE(window->requestedMaximizeMode(), MaximizeFull); QCOMPARE(window->geometryRestore(), QRectF(1280 + 50, 100, 100, 50)); } void OutputChangesTest::testFullScreenWindowRestoredAfterEnablingOutput() { // This test verifies that a fullscreen window will be moved to its original // output when it's re-enabled. const auto outputs = kwinApp()->outputBackend()->outputs(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // kwin will send a configure event with the actived state. QSignalSpy toplevelConfigureRequestedSpy(shellSurface.get(), &Test::XdgToplevel::configureRequested); QSignalSpy surfaceConfigureRequestedSpy(shellSurface->xdgSurface(), &Test::XdgSurface::configureRequested); QVERIFY(surfaceConfigureRequestedSpy.wait()); // Move the window to the right monitor and make it fullscreen. QSignalSpy frameGeometryChangedSpy(window, &Window::frameGeometryChanged); window->move(QPointF(1280 + 50, 100)); window->setFullScreen(true); QVERIFY(surfaceConfigureRequestedSpy.wait()); QCOMPARE(toplevelConfigureRequestedSpy.last().at(0).value(), QSize(1280, 1024)); shellSurface->xdgSurface()->ack_configure(surfaceConfigureRequestedSpy.last().at(0).value()); Test::render(surface.get(), QSize(1280, 1024), Qt::blue); QVERIFY(frameGeometryChangedSpy.wait()); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->isFullScreen(), true); QCOMPARE(window->isRequestedFullScreen(), true); QCOMPARE(window->fullscreenGeometryRestore(), QRectF(1280 + 50, 100, 100, 50)); // Disable the right output. OutputConfiguration config1; { auto changeSet = config1.changeSet(outputs[1]); changeSet->enabled = false; } workspace()->applyOutputConfiguration(config1); // The window will be moved to the left monitor, the geometry restore will be updated too. QCOMPARE(window->frameGeometry(), QRectF(0, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(0, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->isFullScreen(), true); QCOMPARE(window->isRequestedFullScreen(), true); QCOMPARE(window->fullscreenGeometryRestore(), QRectF(50, 100, 100, 50)); // Enable the right monitor. OutputConfiguration config2; { auto changeSet = config2.changeSet(outputs[1]); changeSet->enabled = true; } workspace()->applyOutputConfiguration(config2); // The window will be moved back to the right monitor, the geometry restore will be updated too. QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->isFullScreen(), true); QCOMPARE(window->isRequestedFullScreen(), true); QCOMPARE(window->fullscreenGeometryRestore(), QRectF(1280 + 50, 100, 100, 50)); } void OutputChangesTest::testWindowRestoredAfterChangingScale() { // This test verifies that a window will be moved to its original position after changing the scale of an output const auto output = kwinApp()->outputBackend()->outputs().front(); // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // Move the window to the bottom right const QPointF originalPosition(output->geometry().width() - window->width(), output->geometry().height() - window->height()); window->move(originalPosition); QCOMPARE(window->pos(), originalPosition); QCOMPARE(window->output(), output); // change the scale of the output OutputConfiguration config1; { auto changeSet = config1.changeSet(output); changeSet->scale = 2; } workspace()->applyOutputConfiguration(config1); // The window will be moved to still be in the monitor QCOMPARE(window->pos(), QPointF(output->geometry().width() - window->width(), output->geometry().height() - window->height())); QCOMPARE(window->output(), output); // Change scale back OutputConfiguration config2; { auto changeSet = config2.changeSet(output); changeSet->scale = 1; } workspace()->applyOutputConfiguration(config2); // The window will be moved back to where it was before QCOMPARE(window->pos(), originalPosition); QCOMPARE(window->output(), output); } void OutputChangesTest::testMaximizeStateRestoredAfterEnablingOutput() { // This test verifies that the window state will get restored after disabling and enabling an output, // even if its maximize state changed in the process const auto outputs = kwinApp()->outputBackend()->outputs(); // Disable the right output { OutputConfiguration config; auto changeSet = config.changeSet(outputs[1]); changeSet->enabled = false; workspace()->applyOutputConfiguration(config); } // Create a window. std::unique_ptr surface(Test::createSurface()); std::unique_ptr shellSurface(Test::createXdgToplevelSurface(surface.get())); auto window = Test::renderAndWaitForShown(surface.get(), QSize(100, 50), Qt::blue); QVERIFY(window); // kwin will send a configure event with the actived state. QSignalSpy toplevelConfigureRequestedSpy(shellSurface.get(), &Test::XdgToplevel::configureRequested); QSignalSpy surfaceConfigureRequestedSpy(shellSurface->xdgSurface(), &Test::XdgSurface::configureRequested); QVERIFY(surfaceConfigureRequestedSpy.wait()); const QRectF originalGeometry = window->moveResizeGeometry(); // Enable the right output { OutputConfiguration config; auto changeSet = config.changeSet(outputs[1]); changeSet->enabled = true; workspace()->applyOutputConfiguration(config); } // Move the window to the right monitor and make it maximized. QSignalSpy frameGeometryChangedSpy(window, &Window::frameGeometryChanged); window->move(QPointF(1280 + 50, 100)); window->maximize(MaximizeFull); QVERIFY(surfaceConfigureRequestedSpy.wait()); QCOMPARE(toplevelConfigureRequestedSpy.last().at(0).value(), QSize(1280, 1024)); shellSurface->xdgSurface()->ack_configure(surfaceConfigureRequestedSpy.last().at(0).value()); Test::render(surface.get(), QSize(1280, 1024), Qt::blue); QVERIFY(frameGeometryChangedSpy.wait()); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->maximizeMode(), MaximizeFull); QCOMPARE(window->requestedMaximizeMode(), MaximizeFull); QCOMPARE(window->geometryRestore(), QRectF(1280 + 50, 100, 100, 50)); // Disable the right output { OutputConfiguration config; auto changeSet = config.changeSet(outputs[1]); changeSet->enabled = false; workspace()->applyOutputConfiguration(config); } // The window will be moved to its prior position on the left monitor and unmaximized QVERIFY(surfaceConfigureRequestedSpy.wait()); QCOMPARE(toplevelConfigureRequestedSpy.last().at(0).value(), originalGeometry.size().toSize()); shellSurface->xdgSurface()->ack_configure(surfaceConfigureRequestedSpy.last().at(0).value()); Test::render(surface.get(), originalGeometry.size().toSize(), Qt::blue); QVERIFY(frameGeometryChangedSpy.wait()); QCOMPARE(window->frameGeometry(), originalGeometry); QCOMPARE(window->moveResizeGeometry(), originalGeometry); QCOMPARE(window->output(), outputs[0]); QCOMPARE(window->maximizeMode(), MaximizeRestore); QCOMPARE(window->requestedMaximizeMode(), MaximizeRestore); // Enable the right output again { OutputConfiguration config; auto changeSet = config.changeSet(outputs[1]); changeSet->enabled = true; workspace()->applyOutputConfiguration(config); } // The window will be moved back to the right monitor, maximized and the geometry restore will be updated QVERIFY(surfaceConfigureRequestedSpy.wait()); QCOMPARE(toplevelConfigureRequestedSpy.last().at(0).value(), outputs[1]->geometry().size()); shellSurface->xdgSurface()->ack_configure(surfaceConfigureRequestedSpy.last().at(0).value()); Test::render(surface.get(), outputs[1]->geometry().size(), Qt::blue); QVERIFY(frameGeometryChangedSpy.wait()); QCOMPARE(window->frameGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->moveResizeGeometry(), QRectF(1280, 0, 1280, 1024)); QCOMPARE(window->output(), outputs[1]); QCOMPARE(window->maximizeMode(), MaximizeFull); QCOMPARE(window->requestedMaximizeMode(), MaximizeFull); QCOMPARE(window->geometryRestore(), QRectF(1280 + 50, 100, 100, 50)); } void OutputChangesTest::testLaptopLidClosed() { QSKIP("Laptop lid handling is disabled until config writing is moved into KWin"); Test::setOutputConfig({ Test::OutputInfo{ .geometry = QRect(0, 0, 1280, 1024), .internal = true, }, Test::OutputInfo{ .geometry = QRect(1280, 0, 1280, 1024), .internal = false, }, }); const auto outputs = kwinApp()->outputBackend()->outputs(); const auto internal = outputs.front(); QVERIFY(internal->isInternal()); const auto external = outputs.back(); QVERIFY(!external->isInternal()); auto lidSwitch = std::make_unique(); lidSwitch->setLidSwitch(true); lidSwitch->setName("virtual lid switch"); input()->addInputDevice(lidSwitch.get()); auto timestamp = 1ms; Q_EMIT lidSwitch->switchToggledOff(timestamp++, lidSwitch.get()); QVERIFY(internal->isEnabled()); QVERIFY(external->isEnabled()); Q_EMIT lidSwitch->switchToggledOn(timestamp++, lidSwitch.get()); QVERIFY(!internal->isEnabled()); QVERIFY(external->isEnabled()); Q_EMIT lidSwitch->switchToggledOff(timestamp++, lidSwitch.get()); QVERIFY(internal->isEnabled()); QVERIFY(external->isEnabled()); input()->removeInputDevice(lidSwitch.get()); } } // namespace KWin WAYLANDTEST_MAIN(KWin::OutputChangesTest) #include "outputchanges_test.moc"