2014-09-02 07:34:31 +00:00
|
|
|
/********************************************************************
|
2014-09-17 13:57:56 +00:00
|
|
|
Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-17 13:57:56 +00:00
|
|
|
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.
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-17 13:57:56 +00:00
|
|
|
This library is distributed in the hope that it will be useful,
|
2014-09-02 07:34:31 +00:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2014-09-17 13:57:56 +00:00
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
Lesser General Public License for more details.
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-17 13:57:56 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
2014-09-02 07:34:31 +00:00
|
|
|
*********************************************************************/
|
|
|
|
#include "seat_interface.h"
|
2014-11-13 14:07:31 +00:00
|
|
|
#include "global_p.h"
|
2014-09-02 07:34:31 +00:00
|
|
|
#include "display.h"
|
|
|
|
#include "surface_interface.h"
|
2014-09-18 15:14:50 +00:00
|
|
|
// Qt
|
|
|
|
#include <QHash>
|
|
|
|
// System
|
2014-09-02 07:34:31 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
2014-09-18 15:14:50 +00:00
|
|
|
// Wayland
|
|
|
|
#include <wayland-server.h>
|
2014-09-02 09:11:26 +00:00
|
|
|
#ifndef WL_SEAT_NAME_SINCE_VERSION
|
|
|
|
#define WL_SEAT_NAME_SINCE_VERSION 2
|
|
|
|
#endif
|
2014-11-10 13:51:09 +00:00
|
|
|
// linux
|
|
|
|
#include <linux/input.h>
|
2014-09-02 09:11:26 +00:00
|
|
|
|
2014-09-17 14:10:38 +00:00
|
|
|
namespace KWayland
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
|
2014-09-17 14:10:38 +00:00
|
|
|
namespace Server
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
static const quint32 s_version = 3;
|
|
|
|
|
2014-11-13 14:07:31 +00:00
|
|
|
class SeatInterface::Private : public Global::Private
|
2014-09-18 14:35:28 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
Private(SeatInterface *q, Display *d);
|
2014-11-13 17:43:18 +00:00
|
|
|
void bind(wl_client *client, uint32_t version, uint32_t id) override;
|
2014-09-18 14:35:28 +00:00
|
|
|
void sendCapabilities(wl_resource *r);
|
|
|
|
void sendName(wl_resource *r);
|
|
|
|
|
|
|
|
QString name;
|
|
|
|
bool pointer = false;
|
|
|
|
bool keyboard = false;
|
|
|
|
bool touch = false;
|
|
|
|
QList<wl_resource*> resources;
|
2014-11-13 14:07:31 +00:00
|
|
|
PointerInterface *pointerInterface = nullptr;
|
|
|
|
KeyboardInterface *keyboardInterface = nullptr;
|
2014-09-18 14:35:28 +00:00
|
|
|
|
2014-11-05 14:29:18 +00:00
|
|
|
static SeatInterface *get(wl_resource *native) {
|
|
|
|
auto s = cast(native);
|
|
|
|
return s ? s->q : nullptr;
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
private:
|
|
|
|
static Private *cast(wl_resource *r);
|
|
|
|
static void unbind(wl_resource *r);
|
|
|
|
|
|
|
|
// interface
|
|
|
|
static void getPointerCallback(wl_client *client, wl_resource *resource, uint32_t id);
|
|
|
|
static void getKeyboardCallback(wl_client *client, wl_resource *resource, uint32_t id);
|
|
|
|
static void getTouchCallback(wl_client *client, wl_resource *resource, uint32_t id);
|
|
|
|
static const struct wl_seat_interface s_interface;
|
2014-11-05 14:29:18 +00:00
|
|
|
|
|
|
|
SeatInterface *q;
|
2014-09-18 14:35:28 +00:00
|
|
|
};
|
|
|
|
|
2014-11-13 14:07:31 +00:00
|
|
|
SeatInterface::Private::Private(SeatInterface *q, Display *display)
|
2014-11-13 17:43:18 +00:00
|
|
|
: Global::Private(display, &wl_seat_interface, s_version)
|
2014-11-05 14:29:18 +00:00
|
|
|
, q(q)
|
2014-09-18 14:35:28 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct wl_seat_interface SeatInterface::Private::s_interface = {
|
|
|
|
getPointerCallback,
|
|
|
|
getKeyboardCallback,
|
|
|
|
getTouchCallback
|
2014-09-02 07:34:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
SeatInterface::SeatInterface(Display *display, QObject *parent)
|
2014-11-13 14:07:31 +00:00
|
|
|
: Global(new Private(this, display), parent)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
|
|
|
d->pointerInterface = new PointerInterface(display, this);
|
|
|
|
d->keyboardInterface = new KeyboardInterface(display, this);
|
2014-09-02 07:34:31 +00:00
|
|
|
connect(this, &SeatInterface::nameChanged, this,
|
2014-11-13 14:07:31 +00:00
|
|
|
[this, d] {
|
2014-09-18 14:35:28 +00:00
|
|
|
for (auto it = d->resources.constBegin(); it != d->resources.constEnd(); ++it) {
|
|
|
|
d->sendName(*it);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2014-11-13 14:07:31 +00:00
|
|
|
auto sendCapabilitiesAll = [this, d] {
|
2014-09-18 14:35:28 +00:00
|
|
|
for (auto it = d->resources.constBegin(); it != d->resources.constEnd(); ++it) {
|
|
|
|
d->sendCapabilities(*it);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
connect(this, &SeatInterface::hasPointerChanged, this, sendCapabilitiesAll);
|
|
|
|
connect(this, &SeatInterface::hasKeyboardChanged, this, sendCapabilitiesAll);
|
|
|
|
connect(this, &SeatInterface::hasTouchChanged, this, sendCapabilitiesAll);
|
|
|
|
}
|
|
|
|
|
|
|
|
SeatInterface::~SeatInterface()
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
while (!d->resources.isEmpty()) {
|
|
|
|
wl_resource_destroy(d->resources.takeLast());
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::bind(wl_client *client, uint32_t version, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
wl_resource *r = wl_resource_create(client, &wl_seat_interface, qMin(s_version, version), id);
|
|
|
|
if (!r) {
|
|
|
|
wl_client_post_no_memory(client);
|
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
resources << r;
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
wl_resource_set_implementation(r, &s_interface, this, unbind);
|
2014-09-02 07:34:31 +00:00
|
|
|
|
|
|
|
sendCapabilities(r);
|
|
|
|
sendName(r);
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::unbind(wl_resource *r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:35:28 +00:00
|
|
|
cast(r)->resources.removeAll(r);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::sendName(wl_resource *r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
if (wl_resource_get_version(r) < WL_SEAT_NAME_SINCE_VERSION) {
|
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
wl_seat_send_name(r, name.toUtf8().constData());
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::sendCapabilities(wl_resource *r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
uint32_t capabilities = 0;
|
2014-09-18 14:35:28 +00:00
|
|
|
if (pointer) {
|
2014-09-02 07:34:31 +00:00
|
|
|
capabilities |= WL_SEAT_CAPABILITY_POINTER;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
if (keyboard) {
|
2014-09-02 07:34:31 +00:00
|
|
|
capabilities |= WL_SEAT_CAPABILITY_KEYBOARD;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
if (touch) {
|
2014-09-02 07:34:31 +00:00
|
|
|
capabilities |= WL_SEAT_CAPABILITY_TOUCH;
|
|
|
|
}
|
|
|
|
wl_seat_send_capabilities(r, capabilities);
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
SeatInterface::Private *SeatInterface::Private::cast(wl_resource *r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-11-05 14:29:18 +00:00
|
|
|
return r ? reinterpret_cast<SeatInterface::Private*>(wl_resource_get_user_data(r)) : nullptr;
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SeatInterface::setHasKeyboard(bool has)
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
if (d->keyboard == has) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
d->keyboard = has;
|
|
|
|
emit hasKeyboardChanged(d->keyboard);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SeatInterface::setHasPointer(bool has)
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
if (d->pointer == has) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
d->pointer = has;
|
|
|
|
emit hasPointerChanged(d->pointer);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SeatInterface::setHasTouch(bool has)
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
if (d->touch == has) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
d->touch = has;
|
|
|
|
emit hasTouchChanged(d->touch);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SeatInterface::setName(const QString &name)
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
if (d->name == name) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:35:28 +00:00
|
|
|
d->name = name;
|
|
|
|
emit nameChanged(d->name);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::getPointerCallback(wl_client *client, wl_resource *resource, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:35:28 +00:00
|
|
|
cast(resource)->pointerInterface->createInterface(client, resource, id);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::getKeyboardCallback(wl_client *client, wl_resource *resource, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:35:28 +00:00
|
|
|
cast(resource)->keyboardInterface->createInterfae(client, resource, id);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
void SeatInterface::Private::getTouchCallback(wl_client *client, wl_resource *resource, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
|
|
|
Q_UNUSED(resource)
|
|
|
|
Q_UNUSED(id)
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:35:28 +00:00
|
|
|
QString SeatInterface::name() const
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->name;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SeatInterface::hasPointer() const
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SeatInterface::hasKeyboard() const
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->keyboard;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SeatInterface::hasTouch() const
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->touch;
|
|
|
|
}
|
|
|
|
|
|
|
|
PointerInterface *SeatInterface::pointer()
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->pointerInterface;
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyboardInterface *SeatInterface::keyboard()
|
|
|
|
{
|
2014-11-13 14:07:31 +00:00
|
|
|
Q_D();
|
2014-09-18 14:35:28 +00:00
|
|
|
return d->keyboardInterface;
|
|
|
|
}
|
|
|
|
|
2014-11-05 14:29:18 +00:00
|
|
|
SeatInterface *SeatInterface::get(wl_resource *native)
|
|
|
|
{
|
|
|
|
return Private::get(native);
|
|
|
|
}
|
|
|
|
|
2014-11-13 14:07:31 +00:00
|
|
|
SeatInterface::Private *SeatInterface::d_func() const
|
|
|
|
{
|
|
|
|
return reinterpret_cast<Private*>(d.data());
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
/****************************************
|
|
|
|
* PointerInterface
|
|
|
|
***************************************/
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
class PointerInterface::Private
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Private(Display *display, SeatInterface *parent);
|
|
|
|
void createInterface(wl_client *client, wl_resource *parentResource, uint32_t id);
|
|
|
|
wl_resource *pointerForSurface(SurfaceInterface *surface) const;
|
|
|
|
void surfaceDeleted();
|
|
|
|
void updateButtonSerial(quint32 button, quint32 serial);
|
|
|
|
enum class ButtonState {
|
|
|
|
Released,
|
|
|
|
Pressed
|
|
|
|
};
|
|
|
|
void updateButtonState(quint32 button, ButtonState state);
|
|
|
|
|
|
|
|
Display *display;
|
|
|
|
SeatInterface *seat;
|
|
|
|
struct ResourceData {
|
|
|
|
wl_client *client = nullptr;
|
|
|
|
wl_resource *pointer = nullptr;
|
|
|
|
};
|
|
|
|
QList<ResourceData> resources;
|
|
|
|
quint32 eventTime = 0;
|
|
|
|
QPoint globalPos;
|
|
|
|
struct FocusedSurface {
|
|
|
|
SurfaceInterface *surface = nullptr;
|
|
|
|
QPoint offset = QPoint();
|
|
|
|
wl_resource *pointer = nullptr;
|
|
|
|
quint32 serial = 0;
|
|
|
|
};
|
|
|
|
FocusedSurface focusedSurface;
|
|
|
|
QHash<quint32, quint32> buttonSerials;
|
|
|
|
QHash<quint32, ButtonState> buttonStates;
|
|
|
|
QMetaObject::Connection destroyConnection;
|
|
|
|
|
|
|
|
private:
|
|
|
|
static PointerInterface::Private *cast(wl_resource *resource) {
|
|
|
|
return reinterpret_cast<PointerInterface::Private*>(wl_resource_get_user_data(resource));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unbind(wl_resource *resource);
|
|
|
|
// interface
|
|
|
|
static void setCursorCallback(wl_client *client, wl_resource *resource, uint32_t serial,
|
|
|
|
wl_resource *surface, int32_t hotspot_x, int32_t hotspot_y);
|
|
|
|
// since version 3
|
|
|
|
static void releaseCallback(wl_client *client, wl_resource *resource);
|
|
|
|
|
|
|
|
static const struct wl_pointer_interface s_interface;
|
|
|
|
};
|
|
|
|
|
|
|
|
PointerInterface::Private::Private(Display *display, SeatInterface *parent)
|
|
|
|
: display(display)
|
|
|
|
, seat(parent)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const struct wl_pointer_interface PointerInterface::Private::s_interface = {
|
|
|
|
setCursorCallback,
|
|
|
|
releaseCallback
|
2014-09-02 07:34:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
PointerInterface::PointerInterface(Display *display, SeatInterface *parent)
|
|
|
|
: QObject(parent)
|
2014-09-18 14:58:23 +00:00
|
|
|
, d(new Private(display, parent))
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
PointerInterface::~PointerInterface()
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
while (!d->resources.isEmpty()) {
|
|
|
|
auto data = d->resources.takeLast();
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_resource_destroy(data.pointer);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::createInterface(wl_client *client, wl_resource *parentResource, uint32_t id)
|
2014-09-18 14:58:23 +00:00
|
|
|
{
|
|
|
|
d->createInterface(client, parentResource, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::Private::createInterface(wl_client* client, wl_resource* parentResource, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
wl_resource *p = wl_resource_create(client, &wl_pointer_interface, wl_resource_get_version(parentResource), id);
|
|
|
|
if (!p) {
|
|
|
|
wl_resource_post_no_memory(parentResource);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ResourceData data;
|
|
|
|
data.client = client;
|
|
|
|
data.pointer = p;
|
2014-09-18 14:58:23 +00:00
|
|
|
resources << data;
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
wl_resource_set_implementation(p, &s_interface, this, unbind);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
wl_resource *PointerInterface::Private::pointerForSurface(SurfaceInterface *surface) const
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
if (!surface) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
for (auto it = resources.constBegin(); it != resources.constEnd(); ++it) {
|
2014-11-19 15:53:56 +00:00
|
|
|
if ((*it).client == *surface->client()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return (*it).pointer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::setFocusedSurface(SurfaceInterface *surface, const QPoint &surfacePosition)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
const quint32 serial = d->display->nextSerial();
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.pointer) {
|
2014-11-13 12:05:01 +00:00
|
|
|
wl_pointer_send_leave(d->focusedSurface.pointer, serial, d->focusedSurface.surface->resource());
|
2014-09-18 14:58:23 +00:00
|
|
|
disconnect(d->destroyConnection);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
d->focusedSurface.pointer = d->pointerForSurface(surface);
|
|
|
|
if (!d->focusedSurface.pointer) {
|
|
|
|
d->focusedSurface = Private::FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
d->focusedSurface.surface = surface;
|
|
|
|
d->focusedSurface.offset = surfacePosition;
|
|
|
|
d->focusedSurface.serial = serial;
|
|
|
|
d->destroyConnection = connect(d->focusedSurface.surface, &QObject::destroyed, this, [this] { d->surfaceDeleted(); });
|
|
|
|
|
|
|
|
const QPoint pos = d->globalPos - surfacePosition;
|
|
|
|
wl_pointer_send_enter(d->focusedSurface.pointer, d->focusedSurface.serial,
|
2014-11-13 12:05:01 +00:00
|
|
|
d->focusedSurface.surface->resource(),
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_fixed_from_int(pos.x()), wl_fixed_from_int(pos.y()));
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::surfaceDeleted()
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
focusedSurface = FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::setFocusedSurfacePosition(const QPoint &surfacePosition)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
if (!d->focusedSurface.surface) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
d->focusedSurface.offset = surfacePosition;
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::setGlobalPos(const QPoint &pos)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
if (d->globalPos == pos) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
d->globalPos = pos;
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.pointer) {
|
|
|
|
const QPoint pos = d->globalPos - d->focusedSurface.offset;
|
|
|
|
wl_pointer_send_motion(d->focusedSurface.pointer, d->eventTime,
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_fixed_from_int(pos.x()), wl_fixed_from_int(pos.y()));
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
emit globalPosChanged(d->globalPos);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PointerInterface::updateTimestamp(quint32 time)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
d->eventTime = time;
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 13:51:09 +00:00
|
|
|
static quint32 qtToWaylandButton(Qt::MouseButton button)
|
|
|
|
{
|
|
|
|
static const QHash<Qt::MouseButton, quint32> s_buttons({
|
|
|
|
{Qt::LeftButton, BTN_LEFT},
|
|
|
|
{Qt::RightButton, BTN_RIGHT},
|
|
|
|
{Qt::MiddleButton, BTN_MIDDLE},
|
|
|
|
{Qt::ExtraButton1, BTN_BACK}, // note: QtWayland maps BTN_SIDE
|
|
|
|
{Qt::ExtraButton2, BTN_FORWARD}, // note: QtWayland maps BTN_EXTRA
|
|
|
|
{Qt::ExtraButton3, BTN_TASK}, // note: QtWayland maps BTN_FORWARD
|
|
|
|
{Qt::ExtraButton4, BTN_EXTRA}, // note: QtWayland maps BTN_BACK
|
|
|
|
{Qt::ExtraButton5, BTN_SIDE}, // note: QtWayland maps BTN_TASK
|
|
|
|
{Qt::ExtraButton6, BTN_TASK + 1},
|
|
|
|
{Qt::ExtraButton7, BTN_TASK + 2},
|
|
|
|
{Qt::ExtraButton8, BTN_TASK + 3},
|
|
|
|
{Qt::ExtraButton9, BTN_TASK + 4},
|
|
|
|
{Qt::ExtraButton10, BTN_TASK + 5},
|
|
|
|
{Qt::ExtraButton11, BTN_TASK + 6},
|
|
|
|
{Qt::ExtraButton12, BTN_TASK + 7},
|
|
|
|
{Qt::ExtraButton13, BTN_TASK + 8}
|
|
|
|
// further mapping not possible, 0x120 is BTN_JOYSTICK
|
|
|
|
});
|
|
|
|
return s_buttons.value(button, 0);
|
|
|
|
};
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
void PointerInterface::buttonPressed(quint32 button)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
const quint32 serial = d->display->nextSerial();
|
|
|
|
d->updateButtonSerial(button, serial);
|
|
|
|
d->updateButtonState(button, Private::ButtonState::Pressed);
|
|
|
|
if (!d->focusedSurface.surface || !d->focusedSurface.pointer) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
wl_pointer_send_button(d->focusedSurface.pointer, serial, d->eventTime, button, WL_POINTER_BUTTON_STATE_PRESSED);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 13:51:09 +00:00
|
|
|
void PointerInterface::buttonPressed(Qt::MouseButton button)
|
|
|
|
{
|
|
|
|
const quint32 nativeButton = qtToWaylandButton(button);
|
|
|
|
if (nativeButton == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
buttonPressed(nativeButton);
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
void PointerInterface::buttonReleased(quint32 button)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
const quint32 serial = d->display->nextSerial();
|
|
|
|
d->updateButtonSerial(button, serial);
|
|
|
|
d->updateButtonState(button, Private::ButtonState::Released);
|
|
|
|
if (!d->focusedSurface.surface || !d->focusedSurface.pointer) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
wl_pointer_send_button(d->focusedSurface.pointer, serial, d->eventTime, button, WL_POINTER_BUTTON_STATE_RELEASED);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 13:51:09 +00:00
|
|
|
void PointerInterface::buttonReleased(Qt::MouseButton button)
|
|
|
|
{
|
|
|
|
const quint32 nativeButton = qtToWaylandButton(button);
|
|
|
|
if (nativeButton == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
buttonReleased(nativeButton);
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::updateButtonSerial(quint32 button, quint32 serial)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
auto it = buttonSerials.find(button);
|
|
|
|
if (it == buttonSerials.end()) {
|
|
|
|
buttonSerials.insert(button, serial);
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
it.value() = serial;
|
|
|
|
}
|
|
|
|
|
|
|
|
quint32 PointerInterface::buttonSerial(quint32 button) const
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
auto it = d->buttonSerials.constFind(button);
|
|
|
|
if (it == d->buttonSerials.constEnd()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return it.value();
|
|
|
|
}
|
|
|
|
|
2014-11-10 13:51:09 +00:00
|
|
|
quint32 PointerInterface::buttonSerial(Qt::MouseButton button) const
|
|
|
|
{
|
|
|
|
return buttonSerial(qtToWaylandButton(button));
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::updateButtonState(quint32 button, ButtonState state)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
auto it = buttonStates.find(button);
|
|
|
|
if (it == buttonStates.end()) {
|
|
|
|
buttonStates.insert(button, state);
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
it.value() = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PointerInterface::isButtonPressed(quint32 button) const
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
auto it = d->buttonStates.constFind(button);
|
|
|
|
if (it == d->buttonStates.constEnd()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return false;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
return it.value() == Private::ButtonState::Pressed ? true : false;
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-11-10 13:51:09 +00:00
|
|
|
bool PointerInterface::isButtonPressed(Qt::MouseButton button) const
|
|
|
|
{
|
|
|
|
const quint32 nativeButton = qtToWaylandButton(button);
|
|
|
|
if (nativeButton == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return isButtonPressed(nativeButton);
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
void PointerInterface::axis(Qt::Orientation orientation, quint32 delta)
|
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
if (!d->focusedSurface.surface || !d->focusedSurface.pointer) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
wl_pointer_send_axis(d->focusedSurface.pointer, d->eventTime,
|
2014-09-02 07:34:31 +00:00
|
|
|
(orientation == Qt::Vertical) ? WL_POINTER_AXIS_VERTICAL_SCROLL : WL_POINTER_AXIS_HORIZONTAL_SCROLL,
|
|
|
|
wl_fixed_from_int(delta));
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::unbind(wl_resource *resource)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 14:58:23 +00:00
|
|
|
auto p = cast(resource);
|
|
|
|
auto it = std::find_if(p->resources.begin(), p->resources.end(),
|
2014-09-02 07:34:31 +00:00
|
|
|
[resource](const ResourceData &data) {
|
|
|
|
return data.pointer == resource;
|
|
|
|
}
|
|
|
|
);
|
2014-09-18 14:58:23 +00:00
|
|
|
if (it == p->resources.end()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
if ((*it).pointer == p->focusedSurface.pointer) {
|
|
|
|
QObject::disconnect(p->destroyConnection);
|
|
|
|
p->focusedSurface = FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
2014-09-18 14:58:23 +00:00
|
|
|
p->resources.erase(it);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::setCursorCallback(wl_client *client, wl_resource *resource, uint32_t serial,
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_resource *surface, int32_t hotspot_x, int32_t hotspot_y)
|
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
|
|
|
Q_UNUSED(resource)
|
|
|
|
Q_UNUSED(serial)
|
|
|
|
Q_UNUSED(surface)
|
|
|
|
Q_UNUSED(hotspot_x)
|
|
|
|
Q_UNUSED(hotspot_y)
|
|
|
|
// TODO: implement
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
void PointerInterface::Private::releaseCallback(wl_client *client, wl_resource *resource)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
|
|
|
unbind(resource);
|
|
|
|
}
|
|
|
|
|
2014-09-18 14:58:23 +00:00
|
|
|
QPoint PointerInterface::globalPos() const
|
|
|
|
{
|
|
|
|
return d->globalPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
SurfaceInterface *PointerInterface::focusedSurface() const
|
|
|
|
{
|
|
|
|
return d->focusedSurface.surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint PointerInterface::focusedSurfacePosition() const
|
|
|
|
{
|
|
|
|
return d->focusedSurface.offset;
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
/****************************************
|
|
|
|
* KeyboardInterface
|
|
|
|
***************************************/
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
class KeyboardInterface::Private
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
Private(Display *d, SeatInterface *s);
|
|
|
|
void createInterfae(wl_client *client, wl_resource *parentResource, uint32_t id);
|
|
|
|
void surfaceDeleted();
|
|
|
|
wl_resource *keyboardForSurface(SurfaceInterface *surface) const;
|
|
|
|
void sendKeymap(wl_resource *r);
|
|
|
|
void sendKeymapToAll();
|
|
|
|
void sendModifiers(wl_resource *r);
|
|
|
|
enum class KeyState {
|
|
|
|
Released,
|
|
|
|
Pressed
|
|
|
|
};
|
|
|
|
void updateKey(quint32 key, KeyState state);
|
|
|
|
|
|
|
|
Display *display;
|
|
|
|
SeatInterface *seat;
|
|
|
|
struct ResourceData {
|
|
|
|
wl_client *client = nullptr;
|
|
|
|
wl_resource *keyboard = nullptr;
|
|
|
|
};
|
|
|
|
QList<ResourceData> resources;
|
|
|
|
struct Keymap {
|
|
|
|
int fd = -1;
|
|
|
|
quint32 size = 0;
|
|
|
|
bool xkbcommonCompatible = false;
|
|
|
|
};
|
|
|
|
Keymap keymap;
|
|
|
|
struct Modifiers {
|
|
|
|
quint32 depressed = 0;
|
|
|
|
quint32 latched = 0;
|
|
|
|
quint32 locked = 0;
|
|
|
|
quint32 group = 0;
|
|
|
|
};
|
|
|
|
Modifiers modifiers;
|
|
|
|
struct FocusedSurface {
|
|
|
|
SurfaceInterface *surface = nullptr;
|
|
|
|
wl_resource *keyboard = nullptr;
|
|
|
|
};
|
|
|
|
FocusedSurface focusedSurface;
|
|
|
|
QHash<quint32, KeyState> keyStates;
|
|
|
|
quint32 eventTime = 0;
|
|
|
|
QMetaObject::Connection destroyConnection;
|
|
|
|
|
|
|
|
private:
|
|
|
|
static Private *cast(wl_resource *resource) {
|
|
|
|
return reinterpret_cast<KeyboardInterface::Private*>(wl_resource_get_user_data(resource));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void unbind(wl_resource *resource);
|
|
|
|
// since version 3
|
|
|
|
static void releaseCallback(wl_client *client, wl_resource *resource);
|
|
|
|
|
|
|
|
static const struct wl_keyboard_interface s_interface;
|
|
|
|
};
|
|
|
|
|
|
|
|
KeyboardInterface::Private::Private(Display *d, SeatInterface *s)
|
|
|
|
: display(d)
|
|
|
|
, seat(s)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct wl_keyboard_interface KeyboardInterface::Private::s_interface {
|
|
|
|
releaseCallback
|
2014-09-02 07:34:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
KeyboardInterface::KeyboardInterface(Display *display, SeatInterface *parent)
|
|
|
|
: QObject(parent)
|
2014-09-18 15:14:50 +00:00
|
|
|
, d(new Private(display, parent))
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
KeyboardInterface::~KeyboardInterface()
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
while (!d->resources.isEmpty()) {
|
|
|
|
auto data = d->resources.takeLast();
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_resource_destroy(data.keyboard);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::createInterfae(wl_client *client, wl_resource *parentResource, uint32_t id)
|
2014-09-18 15:14:50 +00:00
|
|
|
{
|
|
|
|
d->createInterfae(client, parentResource, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::Private::createInterfae(wl_client *client, wl_resource *parentResource, uint32_t id)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
wl_resource *k = wl_resource_create(client, &wl_keyboard_interface, wl_resource_get_version(parentResource), id);
|
|
|
|
if (!k) {
|
|
|
|
wl_resource_post_no_memory(parentResource);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ResourceData data;
|
|
|
|
data.client = client;
|
|
|
|
data.keyboard = k;
|
2014-09-18 15:14:50 +00:00
|
|
|
resources << data;
|
2014-09-02 07:34:31 +00:00
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
wl_resource_set_implementation(k, &s_interface, this, unbind);
|
2014-09-02 07:34:31 +00:00
|
|
|
|
|
|
|
sendKeymap(k);
|
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::unbind(wl_resource *resource)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
auto k = cast(resource);
|
|
|
|
auto it = std::find_if(k->resources.begin(), k->resources.end(),
|
2014-09-02 07:34:31 +00:00
|
|
|
[resource](const ResourceData &data) {
|
|
|
|
return data.keyboard == resource;
|
|
|
|
}
|
|
|
|
);
|
2014-09-18 15:14:50 +00:00
|
|
|
if (it == k->resources.end()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 15:14:50 +00:00
|
|
|
if ((*it).keyboard == k->focusedSurface.keyboard) {
|
|
|
|
QObject::disconnect(k->destroyConnection);
|
|
|
|
k->focusedSurface = FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
2014-09-18 15:14:50 +00:00
|
|
|
k->resources.erase(it);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::surfaceDeleted()
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
focusedSurface = FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::releaseCallback(wl_client *client, wl_resource *resource)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
|
|
|
unbind(resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::setKeymap(int fd, quint32 size)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
d->keymap.xkbcommonCompatible = true;
|
|
|
|
d->keymap.fd = fd;
|
|
|
|
d->keymap.size = size;
|
|
|
|
d->sendKeymapToAll();
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::sendKeymap(wl_resource *r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
if (keymap.xkbcommonCompatible) {
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_keyboard_send_keymap(r,
|
|
|
|
WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
|
2014-09-18 15:14:50 +00:00
|
|
|
keymap.fd,
|
|
|
|
keymap.size);
|
2014-09-02 07:34:31 +00:00
|
|
|
} else {
|
|
|
|
int nullFd = open("/dev/null", O_RDONLY);
|
|
|
|
wl_keyboard_send_keymap(r, WL_KEYBOARD_KEYMAP_FORMAT_NO_KEYMAP, nullFd, 0);
|
|
|
|
close(nullFd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::sendKeymapToAll()
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
for (auto it = resources.constBegin(); it != resources.constEnd(); ++it) {
|
2014-09-02 07:34:31 +00:00
|
|
|
sendKeymap((*it).keyboard);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::sendModifiers(wl_resource* r)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
wl_keyboard_send_modifiers(r, display->nextSerial(), modifiers.depressed, modifiers.latched, modifiers.locked, modifiers.group);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::setFocusedSurface(SurfaceInterface *surface)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
const quint32 serial = d->display->nextSerial();
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.keyboard) {
|
2014-11-13 12:05:01 +00:00
|
|
|
wl_keyboard_send_leave(d->focusedSurface.keyboard, serial, d->focusedSurface.surface->resource());
|
2014-09-18 15:14:50 +00:00
|
|
|
disconnect(d->destroyConnection);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
2014-09-18 15:14:50 +00:00
|
|
|
d->focusedSurface.keyboard = d->keyboardForSurface(surface);
|
|
|
|
if (!d->focusedSurface.keyboard) {
|
|
|
|
d->focusedSurface = Private::FocusedSurface();
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
2014-09-18 15:14:50 +00:00
|
|
|
d->focusedSurface.surface = surface;
|
|
|
|
d->destroyConnection = connect(d->focusedSurface.surface, &QObject::destroyed, this, [this] { d->surfaceDeleted(); });
|
2014-09-02 07:34:31 +00:00
|
|
|
|
|
|
|
wl_array keys;
|
|
|
|
wl_array_init(&keys);
|
2014-09-18 15:14:50 +00:00
|
|
|
for (auto it = d->keyStates.constBegin(); it != d->keyStates.constEnd(); ++it) {
|
|
|
|
if (it.value() == Private::KeyState::Pressed) {
|
2014-09-02 07:34:31 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
uint32_t *k = reinterpret_cast<uint32_t*>(wl_array_add(&keys, sizeof(uint32_t)));
|
|
|
|
*k = it.key();
|
|
|
|
}
|
2014-11-13 12:05:01 +00:00
|
|
|
wl_keyboard_send_enter(d->focusedSurface.keyboard, serial, d->focusedSurface.surface->resource(), &keys);
|
2014-09-02 07:34:31 +00:00
|
|
|
wl_array_release(&keys);
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
d->sendModifiers(d->focusedSurface.keyboard);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
wl_resource *KeyboardInterface::Private::keyboardForSurface(SurfaceInterface *surface) const
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
|
|
|
if (!surface) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
2014-09-18 15:14:50 +00:00
|
|
|
for (auto it = resources.constBegin(); it != resources.constEnd(); ++it) {
|
2014-11-19 15:53:56 +00:00
|
|
|
if ((*it).client == *surface->client()) {
|
2014-09-02 07:34:31 +00:00
|
|
|
return (*it).keyboard;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::keyPressed(quint32 key)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
d->updateKey(key, Private::KeyState::Pressed);
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.keyboard) {
|
|
|
|
wl_keyboard_send_key(d->focusedSurface.keyboard, d->display->nextSerial(), d->eventTime, key, WL_KEYBOARD_KEY_STATE_PRESSED);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::keyReleased(quint32 key)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
d->updateKey(key, Private::KeyState::Released);
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.keyboard) {
|
|
|
|
wl_keyboard_send_key(d->focusedSurface.keyboard, d->display->nextSerial(), d->eventTime, key, WL_KEYBOARD_KEY_STATE_RELEASED);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
void KeyboardInterface::Private::updateKey(quint32 key, KeyState state)
|
2014-09-02 07:34:31 +00:00
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
auto it = keyStates.find(key);
|
|
|
|
if (it == keyStates.end()) {
|
|
|
|
keyStates.insert(key, state);
|
2014-09-02 07:34:31 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
it.value() = state;
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::updateTimestamp(quint32 time)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
d->eventTime = time;
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInterface::updateModifiers(quint32 depressed, quint32 latched, quint32 locked, quint32 group)
|
|
|
|
{
|
2014-09-18 15:14:50 +00:00
|
|
|
d->modifiers.depressed = depressed;
|
|
|
|
d->modifiers.latched = latched;
|
|
|
|
d->modifiers.locked = locked;
|
|
|
|
d->modifiers.group = group;
|
|
|
|
if (d->focusedSurface.surface && d->focusedSurface.keyboard) {
|
|
|
|
d->sendModifiers(d->focusedSurface.keyboard);
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-18 15:14:50 +00:00
|
|
|
SurfaceInterface *KeyboardInterface::focusedSurface() const
|
|
|
|
{
|
|
|
|
return d->focusedSurface.surface;
|
|
|
|
}
|
|
|
|
|
2014-09-02 07:34:31 +00:00
|
|
|
}
|
|
|
|
}
|