You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
271 lines
10 KiB
271 lines
10 KiB
/* |
|
* Copyright 2014 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) 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 14 of version 3 of the license. |
|
* |
|
* 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 "waylandclienttest.h" |
|
// KWin::Wayland |
|
#include <KWayland/Client/buffer.h> |
|
#include <KWayland/Client/compositor.h> |
|
#include <KWayland/Client/connection_thread.h> |
|
#include <KWayland/Client/event_queue.h> |
|
#include <KWayland/Client/keyboard.h> |
|
#include <KWayland/Client/output.h> |
|
#include <KWayland/Client/pointer.h> |
|
#include <KWayland/Client/registry.h> |
|
#include <KWayland/Client/seat.h> |
|
#include <KWayland/Client/shell.h> |
|
#include <KWayland/Client/shm_pool.h> |
|
#include <KWayland/Client/surface.h> |
|
// Qt |
|
#include <QAbstractEventDispatcher> |
|
#include <QCoreApplication> |
|
#include <QImage> |
|
#include <QPainter> |
|
#include <QThread> |
|
#include <QTimer> |
|
|
|
#include <linux/input.h> |
|
|
|
using namespace KWayland::Client; |
|
|
|
static Qt::GlobalColor s_colors[] = { |
|
Qt::white, |
|
Qt::red, |
|
Qt::green, |
|
Qt::blue, |
|
Qt::black |
|
}; |
|
static int s_colorIndex = 0; |
|
|
|
WaylandClientTest::WaylandClientTest(QObject *parent) |
|
: QObject(parent) |
|
, m_connectionThread(new QThread(this)) |
|
, m_connectionThreadObject(new ConnectionThread(nullptr)) |
|
, m_eventQueue(nullptr) |
|
, m_compositor(nullptr) |
|
, m_output(nullptr) |
|
, m_surface(nullptr) |
|
, m_shm(nullptr) |
|
, m_shellSurface(nullptr) |
|
, m_timer(new QTimer(this)) |
|
{ |
|
init(); |
|
} |
|
|
|
WaylandClientTest::~WaylandClientTest() |
|
{ |
|
m_connectionThread->quit(); |
|
m_connectionThread->wait(); |
|
m_connectionThreadObject->deleteLater(); |
|
} |
|
|
|
void WaylandClientTest::init() |
|
{ |
|
connect(m_connectionThreadObject, &ConnectionThread::connected, this, |
|
[this]() { |
|
// create the event queue for the main gui thread |
|
m_eventQueue = new EventQueue(this); |
|
m_eventQueue->setup(m_connectionThreadObject); |
|
// setup registry |
|
Registry *registry = new Registry(this); |
|
setupRegistry(registry); |
|
}, |
|
Qt::QueuedConnection); |
|
|
|
m_connectionThreadObject->moveToThread(m_connectionThread); |
|
m_connectionThread->start(); |
|
|
|
m_connectionThreadObject->initConnection(); |
|
|
|
connect(m_timer, &QTimer::timeout, this, |
|
[this]() { |
|
s_colorIndex = (s_colorIndex + 1) % 5; |
|
render(); |
|
} |
|
); |
|
m_timer->setInterval(1000); |
|
m_timer->start(); |
|
} |
|
|
|
void WaylandClientTest::setupRegistry(Registry *registry) |
|
{ |
|
connect(registry, &Registry::compositorAnnounced, this, |
|
[this, registry](quint32 name) { |
|
m_compositor = registry->createCompositor(name, 1, this); |
|
m_surface = m_compositor->createSurface(this); |
|
m_transient.surface = m_compositor->createSurface(this); |
|
} |
|
); |
|
connect(registry, &Registry::shellAnnounced, this, |
|
[this, registry](quint32 name) { |
|
Shell *shell = registry->createShell(name, 1, this); |
|
m_shellSurface = shell->createSurface(m_surface, m_surface); |
|
m_transient.shellSurface = shell->createSurface(m_transient.surface, m_transient.surface); |
|
m_transient.shellSurface->setTransient(m_surface, QPoint(100, 100)); |
|
connect(m_shellSurface, &ShellSurface::sizeChanged, this, static_cast<void(WaylandClientTest::*)(const QSize&)>(&WaylandClientTest::render)); |
|
render(QSize(200, 200)); |
|
} |
|
); |
|
connect(registry, &Registry::outputAnnounced, this, |
|
[this, registry](quint32 name) { |
|
if (m_output) { |
|
return; |
|
} |
|
m_output = registry->createOutput(name, 2, this); |
|
} |
|
); |
|
connect(registry, &Registry::shmAnnounced, this, |
|
[this, registry](quint32 name) { |
|
m_shm = registry->createShmPool(name, 1, this); |
|
} |
|
); |
|
connect(registry, &Registry::seatAnnounced, this, |
|
[this, registry](quint32 name) { |
|
Seat *s = registry->createSeat(name, 2, this); |
|
connect(s, &Seat::hasKeyboardChanged, this, |
|
[this, s](bool has) { |
|
if (!has) { |
|
return; |
|
} |
|
Keyboard *k = s->createKeyboard(this); |
|
connect(k, &Keyboard::keyChanged, this, |
|
[this](quint32 key, Keyboard::KeyState state) { |
|
if (key == KEY_Q && state == Keyboard::KeyState::Released) { |
|
QCoreApplication::instance()->quit(); |
|
} |
|
if (key == KEY_F && state == Keyboard::KeyState::Released) { |
|
if (m_shellSurface) { |
|
static bool s_fullscreen = false; |
|
s_fullscreen = !s_fullscreen; |
|
if (s_fullscreen) { |
|
m_shellSurface->setFullscreen(); |
|
} else { |
|
m_shellSurface->setToplevel(); |
|
} |
|
} |
|
} |
|
if (key == KEY_T && state == Keyboard::KeyState::Released) { |
|
if (m_transient.visible) { |
|
m_transient.surface->attachBuffer(Buffer::Ptr()); |
|
m_transient.surface->commit(Surface::CommitFlag::None); |
|
m_transient.visible = false; |
|
} else { |
|
const QSize size(200, 200); |
|
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(s_colors[s_colorIndex]); |
|
|
|
m_transient.surface->attachBuffer(*buffer); |
|
m_transient.surface->damage(QRect(QPoint(0, 0), size)); |
|
m_transient.surface->commit(Surface::CommitFlag::None); |
|
buffer->setUsed(false); |
|
m_transient.visible = true; |
|
} |
|
} |
|
if (key == KEY_K && state == Keyboard::KeyState::Released) { |
|
delete m_shellSurface; |
|
m_shellSurface = nullptr; |
|
delete m_surface; |
|
m_surface = nullptr; |
|
m_connectionThreadObject->flush(); |
|
} |
|
} |
|
); |
|
} |
|
); |
|
connect(s, &Seat::hasPointerChanged, this, |
|
[this, s](bool has) { |
|
if (!has) { |
|
return; |
|
} |
|
Pointer *p = s->createPointer(this); |
|
connect(p, &Pointer::buttonStateChanged, this, |
|
[this](quint32 serial, quint32 time, quint32 button, Pointer::ButtonState state) { |
|
Q_UNUSED(serial) |
|
Q_UNUSED(time) |
|
if (state == Pointer::ButtonState::Released) { |
|
if (button == BTN_LEFT) { |
|
if (m_timer->isActive()) { |
|
m_timer->stop(); |
|
} else { |
|
m_timer->start(); |
|
} |
|
} |
|
if (button == BTN_RIGHT) { |
|
QCoreApplication::instance()->quit(); |
|
} |
|
if (button == BTN_MIDDLE) { |
|
if (m_shellSurface) { |
|
static bool s_maximized = false; |
|
s_maximized = !s_maximized; |
|
if (s_maximized) { |
|
m_shellSurface->setMaximized(); |
|
} else { |
|
m_shellSurface->setToplevel(); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
); |
|
} |
|
); |
|
} |
|
); |
|
registry->create(m_connectionThreadObject->display()); |
|
registry->setEventQueue(m_eventQueue); |
|
registry->setup(); |
|
} |
|
|
|
void WaylandClientTest::render(const QSize &size) |
|
{ |
|
m_currentSize = size; |
|
render(); |
|
} |
|
|
|
void WaylandClientTest::render() |
|
{ |
|
if (!m_shm || !m_surface || !m_surface->isValid() || !m_currentSize.isValid()) { |
|
return; |
|
} |
|
auto buffer = m_shm->getBuffer(m_currentSize, m_currentSize.width() * 4).toStrongRef(); |
|
buffer->setUsed(true); |
|
QImage image(buffer->address(), m_currentSize.width(), m_currentSize.height(), QImage::Format_ARGB32_Premultiplied); |
|
image.fill(s_colors[s_colorIndex]); |
|
|
|
QPainter p; |
|
p.begin(&image); |
|
QImage icon(QStringLiteral(SOURCE_DIR) + QStringLiteral("/48-apps-kwin.png")); |
|
p.drawImage(QPoint(0, 0), icon); |
|
p.end(); |
|
|
|
m_surface->attachBuffer(*buffer); |
|
m_surface->damage(QRect(QPoint(0, 0), m_currentSize)); |
|
m_surface->commit(Surface::CommitFlag::None); |
|
buffer->setUsed(false); |
|
} |
|
|
|
int main(int argc, char **argv) |
|
{ |
|
QCoreApplication app(argc, argv); |
|
|
|
new WaylandClientTest(&app); |
|
|
|
return app.exec(); |
|
}
|
|
|