From 6c406df7251824ef4d5f5495dad6bdf05d4835e4 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Martin=20Gr=C3=A4=C3=9Flin?= Date: Wed, 12 Nov 2014 14:05:46 +0100 Subject: [PATCH] Adding a test application for paste Counter part for the copyClient test app: the pasteClient test app creates and maps a shell surface and waits for selection announce, initiates the copy/paste mechanismn and prints out what got pasted. After success it closes itself. --- src/wayland/tests/CMakeLists.txt | 4 + src/wayland/tests/pasteclient.cpp | 202 ++++++++++++++++++++++++++++++ 2 files changed, 206 insertions(+) create mode 100644 src/wayland/tests/pasteclient.cpp diff --git a/src/wayland/tests/CMakeLists.txt b/src/wayland/tests/CMakeLists.txt index 6b127eec02..111c87fa7b 100644 --- a/src/wayland/tests/CMakeLists.txt +++ b/src/wayland/tests/CMakeLists.txt @@ -18,3 +18,7 @@ endif() add_executable(copyClient copyclient.cpp) target_link_libraries(copyClient KF5::WaylandClient) ecm_mark_as_test(copyClient) + +add_executable(pasteClient pasteclient.cpp) +target_link_libraries(pasteClient KF5::WaylandClient) +ecm_mark_as_test(pasteClient) diff --git a/src/wayland/tests/pasteclient.cpp b/src/wayland/tests/pasteclient.cpp new file mode 100644 index 0000000000..07139375e5 --- /dev/null +++ b/src/wayland/tests/pasteclient.cpp @@ -0,0 +1,202 @@ +/******************************************************************** +Copyright 2014 Martin Gräßlin + +This library is free software; you can redistribute it and/or +modify it under the terms of the GNU Lesser General Public +License as published by the Free Software Foundation; either +version 2.1 of the License, or (at your option) version 3, or any +later version accepted by the membership of KDE e.V. (or its +successor approved by the membership of KDE e.V.), which shall +act as a proxy defined in Section 6 of version 3 of the license. + +This library 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 +Lesser General Public License for more details. + +You should have received a copy of the GNU Lesser General Public +License along with this library. If not, see . +*********************************************************************/ +#include "../src/client/compositor.h" +#include "../src/client/connection_thread.h" +#include "../src/client/datadevice.h" +#include "../src/client/datadevicemanager.h" +#include "../src/client/dataoffer.h" +#include "../src/client/event_queue.h" +#include "../src/client/keyboard.h" +#include "../src/client/pointer.h" +#include "../src/client/registry.h" +#include "../src/client/seat.h" +#include "../src/client/shell.h" +#include "../src/client/shm_pool.h" +#include "../src/client/surface.h" +// Qt +#include +#include +#include +#include +#include +#include +// system +#include + +using namespace KWayland::Client; + +class PasteClient : public QObject +{ + Q_OBJECT +public: + explicit PasteClient(QObject *parent = nullptr); + virtual ~PasteClient(); + + void init(); + +private: + void setupRegistry(Registry *registry); + void render(); + QThread *m_connectionThread; + ConnectionThread *m_connectionThreadObject; + EventQueue *m_eventQueue = nullptr; + Compositor *m_compositor = nullptr; + DataDeviceManager *m_dataDeviceManager = nullptr; + DataDevice *m_dataDevice = nullptr; + Seat *m_seat = nullptr; + Shell *m_shell = nullptr; + ShellSurface *m_shellSurface = nullptr; + ShmPool *m_shm = nullptr; + Surface *m_surface = nullptr; +}; + +PasteClient::PasteClient(QObject *parent) + : QObject(parent) + , m_connectionThread(new QThread(this)) + , m_connectionThreadObject(new ConnectionThread()) +{ +} + +PasteClient::~PasteClient() +{ + m_connectionThread->quit(); + m_connectionThread->wait(); + m_connectionThreadObject->deleteLater(); +} + +void PasteClient::init() +{ + connect(m_connectionThreadObject, &ConnectionThread::connected, this, + [this] { + m_eventQueue = new EventQueue(this); + m_eventQueue->setup(m_connectionThreadObject); + + Registry *registry = new Registry(this); + setupRegistry(registry); + }, + Qt::QueuedConnection + ); + m_connectionThreadObject->moveToThread(m_connectionThread); + m_connectionThread->start(); + + m_connectionThreadObject->initConnection(); +} + +void PasteClient::setupRegistry(Registry *registry) +{ + connect(registry, &Registry::compositorAnnounced, this, + [this, registry](quint32 name, quint32 version) { + m_compositor = registry->createCompositor(name, version, this); + } + ); + connect(registry, &Registry::shellAnnounced, this, + [this, registry](quint32 name, quint32 version) { + m_shell = registry->createShell(name, version, this); + } + ); + connect(registry, &Registry::shmAnnounced, this, + [this, registry](quint32 name, quint32 version) { + m_shm = registry->createShmPool(name, version, this); + } + ); + connect(registry, &Registry::seatAnnounced, this, + [this, registry](quint32 name, quint32 version) { + m_seat = registry->createSeat(name, version, this); + } + ); + connect(registry, &Registry::dataDeviceManagerAnnounced, this, + [this, registry](quint32 name, quint32 version) { + m_dataDeviceManager = registry->createDataDeviceManager(name, version, this); + } + ); + connect(registry, &Registry::interfacesAnnounced, this, + [this] { + Q_ASSERT(m_compositor); + Q_ASSERT(m_dataDeviceManager); + Q_ASSERT(m_seat); + Q_ASSERT(m_shell); + Q_ASSERT(m_shm); + m_surface = m_compositor->createSurface(this); + Q_ASSERT(m_surface); + m_shellSurface = m_shell->createSurface(m_surface, this); + Q_ASSERT(m_shellSurface); + m_shellSurface->setFullscreen(); + connect(m_shellSurface, &ShellSurface::sizeChanged, this, &PasteClient::render); + + m_dataDevice = m_dataDeviceManager->getDataDevice(m_seat, this); + connect(m_dataDevice, &DataDevice::selectionOffered, this, + [this] { + auto dataOffer = m_dataDevice->offeredSelection(); + if (!dataOffer) { + return; + } + const auto &mimeTypes = dataOffer->offeredMimeTypes(); + auto it = std::find_if(mimeTypes.begin(), mimeTypes.end(), + [](const QMimeType &type) { + return type.inherits(QStringLiteral("text/plain")); + }); + if (it == mimeTypes.end()) { + return; + } + int pipeFds[2]; + if (pipe(pipeFds) != 0){ + return; + } + dataOffer->receive((*it).name(), pipeFds[1]); + close(pipeFds[1]); + QFile readPipe; + if (readPipe.open(pipeFds[0], QIODevice::ReadOnly)) { + qDebug() << "Pasted: " << readPipe.readLine(); + } + close(pipeFds[0]); + QCoreApplication::quit(); + } + ); + } + ); + registry->setEventQueue(m_eventQueue); + registry->create(m_connectionThreadObject); + registry->setup(); +} + +void PasteClient::render() +{ + const QSize &size = m_shellSurface->size(); + auto buffer = m_shm->getBuffer(size, size.width() * 4).toStrongRef(); + buffer->setUsed(true); + QImage image(buffer->address(), size.width(), size.height(), QImage::Format_ARGB32_Premultiplied); + image.fill(Qt::blue); + + m_surface->attachBuffer(*buffer); + m_surface->damage(QRect(QPoint(0, 0), size)); + m_surface->commit(Surface::CommitFlag::None); + buffer->setUsed(false); +} + +int main(int argc, char **argv) +{ + QCoreApplication app(argc, argv); + PasteClient client; + client.init(); + + return app.exec(); +} + +#include "pasteclient.moc"