/* SPDX-FileCopyrightText: 2015 Martin Gräßlin 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 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(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()); } 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) }