374 lines
11 KiB
C++
374 lines
11 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
|
*/
|
|
#include "shadow_interface.h"
|
|
#include "buffer_interface.h"
|
|
#include "display.h"
|
|
#include "surface_interface_p.h"
|
|
|
|
#include <qwayland-server-shadow.h>
|
|
|
|
namespace KWaylandServer
|
|
{
|
|
|
|
class ShadowManagerInterfacePrivate : public QtWaylandServer::org_kde_kwin_shadow_manager
|
|
{
|
|
public:
|
|
ShadowManagerInterfacePrivate(ShadowManagerInterface *_q, Display *display);
|
|
|
|
ShadowManagerInterface *q;
|
|
static const quint32 s_version;
|
|
|
|
protected:
|
|
void org_kde_kwin_shadow_manager_create(Resource *resource, uint32_t id, wl_resource *surface) override;
|
|
void org_kde_kwin_shadow_manager_unset(Resource *resource, wl_resource *surface) override;
|
|
void org_kde_kwin_shadow_manager_destroy(Resource *resource) override;
|
|
};
|
|
|
|
const quint32 ShadowManagerInterfacePrivate::s_version = 2;
|
|
|
|
ShadowManagerInterfacePrivate::ShadowManagerInterfacePrivate(ShadowManagerInterface *_q, Display *display)
|
|
: QtWaylandServer::org_kde_kwin_shadow_manager(*display, s_version)
|
|
, q(_q)
|
|
{
|
|
}
|
|
|
|
void ShadowManagerInterfacePrivate::org_kde_kwin_shadow_manager_destroy(Resource *resource)
|
|
{
|
|
wl_resource_destroy(resource->handle);
|
|
}
|
|
|
|
void ShadowManagerInterfacePrivate::org_kde_kwin_shadow_manager_create(Resource *resource, uint32_t id, wl_resource *surface)
|
|
{
|
|
SurfaceInterface *s = SurfaceInterface::get(surface);
|
|
if (!s) {
|
|
wl_resource_post_error(resource->handle, 0, "Invalid surface");
|
|
return;
|
|
}
|
|
|
|
wl_resource *shadow_resource = wl_resource_create(resource->client(), &org_kde_kwin_shadow_interface, resource->version(), id);
|
|
if (!shadow_resource) {
|
|
wl_client_post_no_memory(resource->client());
|
|
return;
|
|
}
|
|
|
|
auto shadow = new ShadowInterface(shadow_resource);
|
|
|
|
SurfaceInterfacePrivate *surfacePrivate = SurfaceInterfacePrivate::get(s);
|
|
surfacePrivate->setShadow(QPointer<ShadowInterface>(shadow));
|
|
}
|
|
|
|
void ShadowManagerInterfacePrivate::org_kde_kwin_shadow_manager_unset(Resource *resource, wl_resource *surface)
|
|
{
|
|
Q_UNUSED(resource)
|
|
SurfaceInterface *s = SurfaceInterface::get(surface);
|
|
if (!s) {
|
|
wl_resource_post_error(resource->handle, 0, "Invalid surface");
|
|
return;
|
|
}
|
|
SurfaceInterfacePrivate *surfacePrivate = SurfaceInterfacePrivate::get(s);
|
|
surfacePrivate->setShadow(QPointer<ShadowInterface>());
|
|
}
|
|
|
|
ShadowManagerInterface::ShadowManagerInterface(Display *display, QObject *parent)
|
|
: QObject(parent)
|
|
, d(new ShadowManagerInterfacePrivate(this, display))
|
|
{
|
|
}
|
|
|
|
ShadowManagerInterface::~ShadowManagerInterface() = default;
|
|
|
|
class ShadowInterfacePrivate : public QtWaylandServer::org_kde_kwin_shadow
|
|
{
|
|
public:
|
|
ShadowInterfacePrivate(ShadowInterface *_q, wl_resource *resource);
|
|
void commit();
|
|
void attach(State::Flags flag, wl_resource *buffer);
|
|
|
|
struct State {
|
|
enum Flags {
|
|
None = 0,
|
|
LeftBuffer = 1 << 0,
|
|
TopLeftBuffer = 1 << 1,
|
|
TopBuffer = 1 << 2,
|
|
TopRightBuffer = 1 << 3,
|
|
RightBuffer = 1 << 4,
|
|
BottomRightBuffer = 1 << 5,
|
|
BottomBuffer = 1 << 6,
|
|
BottomLeftBuffer = 1 << 7,
|
|
Offset = 1 << 8
|
|
};
|
|
BufferInterface *left = nullptr;
|
|
BufferInterface *topLeft = nullptr;
|
|
BufferInterface *top = nullptr;
|
|
BufferInterface *topRight = nullptr;
|
|
BufferInterface *right = nullptr;
|
|
BufferInterface *bottomRight = nullptr;
|
|
BufferInterface *bottom = nullptr;
|
|
BufferInterface *bottomLeft = nullptr;
|
|
QMarginsF offset;
|
|
Flags flags = Flags::None;
|
|
};
|
|
State current;
|
|
State pending;
|
|
ShadowInterface *q;
|
|
|
|
protected:
|
|
void org_kde_kwin_shadow_destroy_resource(Resource *resource) override;
|
|
void org_kde_kwin_shadow_commit(Resource *resource) override;
|
|
void org_kde_kwin_shadow_attach_left(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_top_left(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_top(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_top_right(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_right(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_bottom_right(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_bottom(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_attach_bottom_left(Resource *resource, wl_resource *buffer) override;
|
|
void org_kde_kwin_shadow_set_left_offset(Resource *resource, wl_fixed_t offset) override;
|
|
void org_kde_kwin_shadow_set_top_offset(Resource *resource, wl_fixed_t offset) override;
|
|
void org_kde_kwin_shadow_set_right_offset(Resource *resource, wl_fixed_t offset) override;
|
|
void org_kde_kwin_shadow_set_bottom_offset(Resource *resource, wl_fixed_t offset) override;
|
|
void org_kde_kwin_shadow_destroy(Resource *resource) override;
|
|
|
|
};
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_commit(Resource *resource)
|
|
{
|
|
Q_UNUSED(resource)
|
|
#define BUFFER( __FLAG__, __PART__ ) \
|
|
if (pending.flags & State::Flags::__FLAG__##Buffer) { \
|
|
if (current.__PART__) { \
|
|
current.__PART__->unref(); \
|
|
} \
|
|
if (pending.__PART__) { \
|
|
pending.__PART__->ref(); \
|
|
} \
|
|
current.__PART__ = pending.__PART__; \
|
|
}
|
|
BUFFER(Left, left)
|
|
BUFFER(TopLeft, topLeft)
|
|
BUFFER(Top, top)
|
|
BUFFER(TopRight, topRight)
|
|
BUFFER(Right, right)
|
|
BUFFER(BottomRight, bottomRight)
|
|
BUFFER(Bottom, bottom)
|
|
BUFFER(BottomLeft, bottomLeft)
|
|
#undef BUFFER
|
|
|
|
if (pending.flags & State::Offset) {
|
|
current.offset = pending.offset;
|
|
}
|
|
pending = State();
|
|
}
|
|
|
|
void ShadowInterfacePrivate::attach(ShadowInterfacePrivate::State::Flags flag, wl_resource *buffer)
|
|
{
|
|
BufferInterface *b = BufferInterface::get(buffer);
|
|
if (b) {
|
|
QObject::connect(b, &BufferInterface::aboutToBeDestroyed, q,
|
|
[this](BufferInterface *buffer) {
|
|
#define PENDING( __PART__ ) \
|
|
if (pending.__PART__ == buffer) { \
|
|
pending.__PART__ = nullptr; \
|
|
}
|
|
PENDING(left)
|
|
PENDING(topLeft)
|
|
PENDING(top)
|
|
PENDING(topRight)
|
|
PENDING(right)
|
|
PENDING(bottomRight)
|
|
PENDING(bottom)
|
|
PENDING(bottomLeft)
|
|
#undef PENDING
|
|
|
|
#define CURRENT( __PART__ ) \
|
|
if (current.__PART__ == buffer) { \
|
|
current.__PART__->unref(); \
|
|
current.__PART__ = nullptr; \
|
|
}
|
|
CURRENT(left)
|
|
CURRENT(topLeft)
|
|
CURRENT(top)
|
|
CURRENT(topRight)
|
|
CURRENT(right)
|
|
CURRENT(bottomRight)
|
|
CURRENT(bottom)
|
|
CURRENT(bottomLeft)
|
|
#undef CURRENT
|
|
}
|
|
);
|
|
}
|
|
switch (flag) {
|
|
case State::LeftBuffer:
|
|
pending.left = b;
|
|
break;
|
|
case State::TopLeftBuffer:
|
|
pending.topLeft = b;
|
|
break;
|
|
case State::TopBuffer:
|
|
pending.top = b;
|
|
break;
|
|
case State::TopRightBuffer:
|
|
pending.topRight = b;
|
|
break;
|
|
case State::RightBuffer:
|
|
pending.right = b;
|
|
break;
|
|
case State::BottomRightBuffer:
|
|
pending.bottomRight = b;
|
|
break;
|
|
case State::BottomBuffer:
|
|
pending.bottom = b;
|
|
break;
|
|
case State::BottomLeftBuffer:
|
|
pending.bottomLeft = b;
|
|
break;
|
|
default:
|
|
Q_UNREACHABLE();
|
|
break;
|
|
}
|
|
pending.flags = State::Flags(pending.flags | flag);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_destroy(Resource *resource)
|
|
{
|
|
wl_resource_destroy(resource->handle);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_destroy_resource(Resource *resource)
|
|
{
|
|
Q_UNUSED(resource)
|
|
delete q;
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_left(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::LeftBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_top_left(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::TopLeftBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_top(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::TopBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_top_right(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::TopRightBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_right(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::RightBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_bottom_right(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::BottomRightBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_bottom(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::BottomBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_attach_bottom_left(Resource *resource, wl_resource *buffer)
|
|
{
|
|
Q_UNUSED(resource)
|
|
attach(State::BottomLeftBuffer, buffer);
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_set_left_offset(Resource *resource, wl_fixed_t offset)
|
|
{
|
|
Q_UNUSED(resource)
|
|
pending.flags = State::Flags(pending.flags | State::Offset);
|
|
pending.offset.setLeft(wl_fixed_to_double(offset));
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_set_top_offset(Resource *resource, wl_fixed_t offset)
|
|
{
|
|
Q_UNUSED(resource)
|
|
pending.flags = State::Flags(pending.flags | State::Offset);
|
|
pending.offset.setTop(wl_fixed_to_double(offset));
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_set_right_offset(Resource *resource, wl_fixed_t offset)
|
|
{
|
|
Q_UNUSED(resource)
|
|
pending.flags = State::Flags(pending.flags | State::Offset);
|
|
pending.offset.setRight(wl_fixed_to_double(offset));
|
|
}
|
|
|
|
void ShadowInterfacePrivate::org_kde_kwin_shadow_set_bottom_offset(Resource *resource, wl_fixed_t offset)
|
|
{
|
|
Q_UNUSED(resource)
|
|
pending.flags = State::Flags(pending.flags | State::Offset);
|
|
pending.offset.setBottom(wl_fixed_to_double(offset));
|
|
}
|
|
|
|
ShadowInterfacePrivate::ShadowInterfacePrivate(ShadowInterface *_q, wl_resource *resource)
|
|
: QtWaylandServer::org_kde_kwin_shadow(resource)
|
|
, q(_q)
|
|
{
|
|
}
|
|
|
|
ShadowInterfacePrivate::~ShadowInterfacePrivate()
|
|
{
|
|
#define CURRENT( __PART__ ) \
|
|
if (current.__PART__) { \
|
|
current.__PART__->unref(); \
|
|
}
|
|
CURRENT(left)
|
|
CURRENT(topLeft)
|
|
CURRENT(top)
|
|
CURRENT(topRight)
|
|
CURRENT(right)
|
|
CURRENT(bottomRight)
|
|
CURRENT(bottom)
|
|
CURRENT(bottomLeft)
|
|
#undef CURRENT
|
|
}
|
|
|
|
ShadowInterface::ShadowInterface(wl_resource *resource)
|
|
: QObject()
|
|
, d(new ShadowInterfacePrivate(this, resource))
|
|
{
|
|
}
|
|
|
|
ShadowInterface::~ShadowInterface() = default;
|
|
|
|
QMarginsF ShadowInterface::offset() const
|
|
{
|
|
return d->current.offset;
|
|
}
|
|
|
|
#define BUFFER( __PART__ ) \
|
|
BufferInterface *ShadowInterface::__PART__() const \
|
|
{ \
|
|
return d->current.__PART__; \
|
|
}
|
|
|
|
BUFFER(left)
|
|
BUFFER(topLeft)
|
|
BUFFER(top)
|
|
BUFFER(topRight)
|
|
BUFFER(right)
|
|
BUFFER(bottomRight)
|
|
BUFFER(bottom)
|
|
BUFFER(bottomLeft)
|
|
|
|
|
|
}
|