/* SPDX-FileCopyrightText: 2014 Martin Gräßlin <mgraesslin@kde.org> SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL */ #include "core/graphicsbufferview.h" #include "wayland/compositor.h" #include "wayland/datadevicemanager.h" #include "wayland/display.h" #include "wayland/keyboard.h" #include "wayland/output.h" #include "wayland/pointer.h" #include "wayland/seat.h" #include "wayland/xdgshell.h" #include "fakeoutput.h" #include <QApplication> #include <QCommandLineParser> #include <QDateTime> #include <QFile> #include <QKeyEvent> #include <QMouseEvent> #include <QPainter> #include <QPointer> #include <QThreadPool> #include <QWidget> #include <iostream> #include <unistd.h> static int startXServer() { const QByteArray process = QByteArrayLiteral("Xwayland"); int pipeFds[2]; if (pipe(pipeFds) != 0) { std::cerr << "FATAL ERROR failed to create pipe to start X Server " << process.constData() << std::endl; exit(1); } pid_t pid = fork(); if (pid == 0) { // child process - should be turned into Xwayland // writes to pipe, closes read side close(pipeFds[0]); char fdbuf[16]; sprintf(fdbuf, "%d", pipeFds[1]); execlp(process.constData(), process.constData(), "-displayfd", fdbuf, "-rootless", (char *)nullptr); close(pipeFds[1]); exit(20); } // parent process - this is the wayland server // reads from pipe, closes write side close(pipeFds[1]); return pipeFds[0]; } static void readDisplayFromPipe(int pipe) { QFile readPipe; if (!readPipe.open(pipe, QIODevice::ReadOnly)) { std::cerr << "FATAL ERROR failed to open pipe to start X Server XWayland" << std::endl; exit(1); } QByteArray displayNumber = readPipe.readLine(); displayNumber.prepend(QByteArray(":")); displayNumber.remove(displayNumber.size() - 1, 1); std::cout << "X-Server started on display " << displayNumber.constData() << std::endl; setenv("DISPLAY", displayNumber.constData(), true); // close our pipe close(pipe); } class CompositorWindow : public QWidget { Q_OBJECT public: explicit CompositorWindow(QWidget *parent = nullptr); virtual ~CompositorWindow(); void surfaceCreated(KWin::XdgToplevelInterface *surface); void setSeat(const QPointer<KWin::SeatInterface> &seat); protected: void paintEvent(QPaintEvent *event) override; void keyPressEvent(QKeyEvent *event) override; void keyReleaseEvent(QKeyEvent *event) override; void mouseMoveEvent(QMouseEvent *event) override; void mousePressEvent(QMouseEvent *event) override; void mouseReleaseEvent(QMouseEvent *event) override; void wheelEvent(QWheelEvent *event) override; private: void updateFocus(); QList<KWin::XdgToplevelInterface *> m_stackingOrder; QPointer<KWin::SeatInterface> m_seat; }; CompositorWindow::CompositorWindow(QWidget *parent) : QWidget(parent) { setMouseTracking(true); } CompositorWindow::~CompositorWindow() = default; void CompositorWindow::surfaceCreated(KWin::XdgToplevelInterface *surface) { using namespace KWin; surface->sendConfigure(QSize(), XdgToplevelInterface::States()); m_stackingOrder << surface; connect(surface->surface(), &SurfaceInterface::damaged, this, static_cast<void (CompositorWindow::*)()>(&CompositorWindow::update)); connect(surface, &XdgToplevelInterface::destroyed, this, [surface, this] { m_stackingOrder.removeAll(surface); updateFocus(); update(); }); updateFocus(); } void CompositorWindow::updateFocus() { using namespace KWin; if (!m_seat || m_stackingOrder.isEmpty()) { return; } auto it = std::find_if(m_stackingOrder.constBegin(), m_stackingOrder.constEnd(), [](XdgToplevelInterface *toplevel) { return toplevel->surface()->isMapped(); }); if (it == m_stackingOrder.constEnd()) { return; } m_seat->notifyPointerEnter((*it)->surface(), m_seat->pointerPos()); m_seat->setFocusedKeyboardSurface((*it)->surface()); } void CompositorWindow::setSeat(const QPointer<KWin::SeatInterface> &seat) { m_seat = seat; } void CompositorWindow::paintEvent(QPaintEvent *event) { QWidget::paintEvent(event); QPainter p(this); for (auto window : m_stackingOrder) { KWin::SurfaceInterface *surface = window->surface(); if (!surface || !surface->isMapped()) { continue; } KWin::GraphicsBufferView view(surface->buffer()); if (view.image()) { p.drawImage(QPoint(0, 0), *view.image()); } surface->frameRendered(QDateTime::currentMSecsSinceEpoch()); } } void CompositorWindow::keyPressEvent(QKeyEvent *event) { QWidget::keyPressEvent(event); if (!m_seat) { return; } if (!m_seat->focusedKeyboardSurface()) { updateFocus(); } m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); m_seat->notifyKeyboardKey(event->nativeScanCode() - 8, KWin::KeyboardKeyState::Pressed); } void CompositorWindow::keyReleaseEvent(QKeyEvent *event) { QWidget::keyReleaseEvent(event); if (!m_seat) { return; } m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); m_seat->notifyKeyboardKey(event->nativeScanCode() - 8, KWin::KeyboardKeyState::Released); } void CompositorWindow::mouseMoveEvent(QMouseEvent *event) { QWidget::mouseMoveEvent(event); if (!m_seat->focusedPointerSurface()) { updateFocus(); } m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); m_seat->notifyPointerMotion(event->localPos().toPoint()); m_seat->notifyPointerFrame(); } void CompositorWindow::mousePressEvent(QMouseEvent *event) { QWidget::mousePressEvent(event); if (!m_seat->focusedPointerSurface()) { if (!m_stackingOrder.isEmpty()) { m_seat->notifyPointerEnter(m_stackingOrder.last()->surface(), event->globalPos()); } } m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); m_seat->notifyPointerButton(event->button(), KWin::PointerButtonState::Pressed); m_seat->notifyPointerFrame(); } void CompositorWindow::mouseReleaseEvent(QMouseEvent *event) { QWidget::mouseReleaseEvent(event); m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); m_seat->notifyPointerButton(event->button(), KWin::PointerButtonState::Released); m_seat->notifyPointerFrame(); } void CompositorWindow::wheelEvent(QWheelEvent *event) { QWidget::wheelEvent(event); m_seat->setTimestamp(std::chrono::milliseconds(event->timestamp())); const QPoint &angle = event->angleDelta() / (8 * 15); if (angle.x() != 0) { m_seat->notifyPointerAxis(Qt::Horizontal, angle.x(), 1, KWin::PointerAxisSource::Wheel); } if (angle.y() != 0) { m_seat->notifyPointerAxis(Qt::Vertical, angle.y(), 1, KWin::PointerAxisSource::Wheel); } m_seat->notifyPointerFrame(); } int main(int argc, char **argv) { using namespace KWin; QApplication app(argc, argv); QCommandLineParser parser; parser.addHelpOption(); QCommandLineOption xwaylandOption(QStringList{QStringLiteral("x"), QStringLiteral("xwayland")}, QStringLiteral("Start a rootless Xwayland server")); parser.addOption(xwaylandOption); parser.process(app); KWin::Display display; display.start(); new DataDeviceManagerInterface(&display); new CompositorInterface(&display, &display); XdgShellInterface *shell = new XdgShellInterface(&display); display.createShm(); const QSize windowSize(1024, 768); auto outputHandle = std::make_unique<FakeOutput>(); outputHandle->setPhysicalSize(QSize(269, 202)); outputHandle->setMode(windowSize, 60000); auto outputInterface = std::make_unique<OutputInterface>(&display, outputHandle.get()); SeatInterface *seat = new SeatInterface(&display); seat->setHasKeyboard(true); seat->setHasPointer(true); seat->setName(QStringLiteral("testSeat0")); CompositorWindow compositorWindow; compositorWindow.setSeat(seat); compositorWindow.setMinimumSize(windowSize); compositorWindow.setMaximumSize(windowSize); compositorWindow.setGeometry(QRect(QPoint(0, 0), windowSize)); compositorWindow.show(); QObject::connect(shell, &XdgShellInterface::toplevelCreated, &compositorWindow, &CompositorWindow::surfaceCreated); // start XWayland if (parser.isSet(xwaylandOption)) { // starts XWayland by forking and opening a pipe const int pipe = startXServer(); if (pipe == -1) { exit(1); } QThreadPool::globalInstance()->start([pipe] { readDisplayFromPipe(pipe); }); } return app.exec(); } #include "renderingservertest.moc"