2014-08-28 07:52:35 +00:00
|
|
|
/********************************************************************
|
2014-09-17 13:57:56 +00:00
|
|
|
Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
|
2014-08-28 07:52:35 +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-08-28 07:52:35 +00:00
|
|
|
|
2014-09-17 13:57:56 +00:00
|
|
|
This library is distributed in the hope that it will be useful,
|
2014-08-28 07:52:35 +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-08-28 07:52:35 +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-08-28 07:52:35 +00:00
|
|
|
*********************************************************************/
|
|
|
|
#include "surface_interface.h"
|
2014-10-14 12:04:35 +00:00
|
|
|
#include "surface_interface_p.h"
|
2014-08-28 12:22:53 +00:00
|
|
|
#include "buffer_interface.h"
|
2014-11-19 15:53:56 +00:00
|
|
|
#include "clientconnection.h"
|
2014-08-28 07:52:35 +00:00
|
|
|
#include "compositor_interface.h"
|
2014-10-16 12:59:01 +00:00
|
|
|
#include "region_interface.h"
|
2014-10-14 12:04:35 +00:00
|
|
|
#include "subcompositor_interface.h"
|
|
|
|
#include "subsurface_interface_p.h"
|
2014-09-19 05:06:31 +00:00
|
|
|
// Wayland
|
|
|
|
#include <wayland-server.h>
|
2014-10-14 12:04:35 +00:00
|
|
|
// std
|
|
|
|
#include <algorithm>
|
2014-08-28 07:52:35 +00:00
|
|
|
|
2014-09-17 14:10:38 +00:00
|
|
|
namespace KWayland
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-09-17 14:10:38 +00:00
|
|
|
namespace Server
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
|
2014-11-20 15:40:14 +00:00
|
|
|
SurfaceInterface::Private::Private(SurfaceInterface *q, CompositorInterface *c, wl_resource *parentResource)
|
|
|
|
: Resource::Private(q, c, parentResource, &wl_surface_interface, &s_interface)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
SurfaceInterface::Private::~Private()
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
destroy();
|
2014-09-19 05:06:31 +00:00
|
|
|
}
|
|
|
|
|
2016-03-18 08:57:46 +00:00
|
|
|
void SurfaceInterface::Private::addChild(QPointer< SubSurfaceInterface > child)
|
2014-10-14 12:04:35 +00:00
|
|
|
{
|
2016-03-18 08:57:46 +00:00
|
|
|
// protocol is not precise on how to handle the addition of new sub surfaces, this follows Weston's behavior
|
|
|
|
if (subSurface.isNull() || !subSurface->isSynchronized()) {
|
|
|
|
pending.children.append(child);
|
|
|
|
} else {
|
|
|
|
subSurfacePending.children.append(child);
|
|
|
|
}
|
2016-03-21 13:32:31 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
|
|
|
emit q->subSurfaceTreeChanged();
|
|
|
|
QObject::connect(child.data(), &SubSurfaceInterface::positionChanged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::connect(child->surface().data(), &SurfaceInterface::damaged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::connect(child->surface().data(), &SurfaceInterface::unmapped, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::connect(child->surface().data(), &SurfaceInterface::subSurfaceTreeChanged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
2014-10-14 12:04:35 +00:00
|
|
|
}
|
|
|
|
|
2016-03-18 08:57:46 +00:00
|
|
|
void SurfaceInterface::Private::removeChild(QPointer< SubSurfaceInterface > child)
|
2014-10-14 12:04:35 +00:00
|
|
|
{
|
2016-03-18 08:57:46 +00:00
|
|
|
// protocol is not precise on how to handle the addition of new sub surfaces, this follows Weston's behavior
|
|
|
|
if (subSurface.isNull() || !subSurface->isSynchronized()) {
|
|
|
|
pending.children.removeAll(child);
|
|
|
|
} else {
|
|
|
|
subSurfacePending.children.removeAll(child);
|
|
|
|
}
|
2016-03-21 13:32:31 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
|
|
|
emit q->subSurfaceTreeChanged();
|
|
|
|
QObject::disconnect(child.data(), &SubSurfaceInterface::positionChanged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::disconnect(child->surface().data(), &SurfaceInterface::damaged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::disconnect(child->surface().data(), &SurfaceInterface::unmapped, q, &SurfaceInterface::subSurfaceTreeChanged);
|
|
|
|
QObject::disconnect(child->surface().data(), &SurfaceInterface::subSurfaceTreeChanged, q, &SurfaceInterface::subSurfaceTreeChanged);
|
2014-10-14 12:04:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool SurfaceInterface::Private::raiseChild(QPointer<SubSurfaceInterface> subsurface, SurfaceInterface *sibling)
|
|
|
|
{
|
2014-11-14 09:55:06 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
2014-10-14 12:04:35 +00:00
|
|
|
auto it = std::find(pending.children.begin(), pending.children.end(), subsurface);
|
|
|
|
if (it == pending.children.end()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (pending.children.count() == 1) {
|
|
|
|
// nothing to do
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (sibling == q) {
|
|
|
|
// it's to the parent, so needs to become last item
|
|
|
|
pending.children.append(*it);
|
|
|
|
pending.children.erase(it);
|
2016-03-18 08:57:46 +00:00
|
|
|
pending.childrenChanged = true;
|
2014-10-14 12:04:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (!sibling->subSurface()) {
|
|
|
|
// not a sub surface
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
auto siblingIt = std::find(pending.children.begin(), pending.children.end(), sibling->subSurface());
|
|
|
|
if (siblingIt == pending.children.end() || siblingIt == it) {
|
|
|
|
// not a sibling
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
auto value = (*it);
|
|
|
|
pending.children.erase(it);
|
|
|
|
// find the iterator again
|
|
|
|
siblingIt = std::find(pending.children.begin(), pending.children.end(), sibling->subSurface());
|
|
|
|
pending.children.insert(++siblingIt, value);
|
2016-03-18 08:57:46 +00:00
|
|
|
pending.childrenChanged = true;
|
2014-10-14 12:04:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SurfaceInterface::Private::lowerChild(QPointer<SubSurfaceInterface> subsurface, SurfaceInterface *sibling)
|
|
|
|
{
|
2014-11-14 09:55:06 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
2014-10-14 12:04:35 +00:00
|
|
|
auto it = std::find(pending.children.begin(), pending.children.end(), subsurface);
|
|
|
|
if (it == pending.children.end()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (pending.children.count() == 1) {
|
|
|
|
// nothing to do
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (sibling == q) {
|
|
|
|
// it's to the parent, so needs to become first item
|
|
|
|
auto value = *it;
|
|
|
|
pending.children.erase(it);
|
|
|
|
pending.children.prepend(value);
|
2016-03-18 08:57:46 +00:00
|
|
|
pending.childrenChanged = true;
|
2014-10-14 12:04:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (!sibling->subSurface()) {
|
|
|
|
// not a sub surface
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
auto siblingIt = std::find(pending.children.begin(), pending.children.end(), sibling->subSurface());
|
|
|
|
if (siblingIt == pending.children.end() || siblingIt == it) {
|
|
|
|
// not a sibling
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
auto value = (*it);
|
|
|
|
pending.children.erase(it);
|
|
|
|
// find the iterator again
|
|
|
|
siblingIt = std::find(pending.children.begin(), pending.children.end(), sibling->subSurface());
|
|
|
|
pending.children.insert(siblingIt, value);
|
2016-03-18 08:57:46 +00:00
|
|
|
pending.childrenChanged = true;
|
2014-10-14 12:04:35 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-15 09:07:50 +00:00
|
|
|
void SurfaceInterface::Private::setShadow(const QPointer<ShadowInterface> &shadow)
|
|
|
|
{
|
|
|
|
pending.shadow = shadow;
|
|
|
|
pending.shadowIsSet = true;
|
|
|
|
}
|
|
|
|
|
2015-08-26 12:42:58 +00:00
|
|
|
void SurfaceInterface::Private::setBlur(const QPointer<BlurInterface> &blur)
|
|
|
|
{
|
|
|
|
pending.blur = blur;
|
|
|
|
pending.blurIsSet = true;
|
|
|
|
}
|
|
|
|
|
2015-09-09 11:04:11 +00:00
|
|
|
void SurfaceInterface::Private::setSlide(const QPointer<SlideInterface> &slide)
|
|
|
|
{
|
|
|
|
pending.slide = slide;
|
|
|
|
pending.slideIsSet = true;
|
|
|
|
}
|
|
|
|
|
2015-09-02 16:13:25 +00:00
|
|
|
void SurfaceInterface::Private::setContrast(const QPointer<ContrastInterface> &contrast)
|
|
|
|
{
|
|
|
|
pending.contrast = contrast;
|
|
|
|
pending.contrastIsSet = true;
|
|
|
|
}
|
|
|
|
|
2015-09-09 14:39:50 +00:00
|
|
|
#ifndef DOXYGEN_SHOULD_SKIP_THIS
|
2014-09-19 05:06:31 +00:00
|
|
|
const struct wl_surface_interface SurfaceInterface::Private::s_interface = {
|
|
|
|
destroyCallback,
|
|
|
|
attachCallback,
|
|
|
|
damageCallback,
|
|
|
|
frameCallaback,
|
|
|
|
opaqueRegionCallback,
|
|
|
|
inputRegionCallback,
|
|
|
|
commitCallback,
|
|
|
|
bufferTransformCallback,
|
|
|
|
bufferScaleCallback
|
|
|
|
};
|
2015-09-09 14:39:50 +00:00
|
|
|
#endif
|
2014-09-19 05:06:31 +00:00
|
|
|
|
2014-11-20 15:40:14 +00:00
|
|
|
SurfaceInterface::SurfaceInterface(CompositorInterface *parent, wl_resource *parentResource)
|
|
|
|
: Resource(new Private(this, parent, parentResource))
|
2014-09-19 05:06:31 +00:00
|
|
|
{
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
SurfaceInterface::~SurfaceInterface() = default;
|
|
|
|
|
2014-08-28 07:52:35 +00:00
|
|
|
void SurfaceInterface::frameRendered(quint32 msec)
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-08-28 07:52:35 +00:00
|
|
|
// notify all callbacks
|
2015-11-04 13:50:15 +00:00
|
|
|
const bool needsFlush = !d->current.callbacks.isEmpty();
|
2014-09-19 05:06:31 +00:00
|
|
|
while (!d->current.callbacks.isEmpty()) {
|
|
|
|
wl_resource *r = d->current.callbacks.takeFirst();
|
2014-08-28 07:52:35 +00:00
|
|
|
wl_callback_send_done(r, msec);
|
|
|
|
wl_resource_destroy(r);
|
|
|
|
}
|
2016-03-18 08:13:27 +00:00
|
|
|
for (auto it = d->current.children.constBegin(); it != d->current.children.constEnd(); ++it) {
|
|
|
|
const auto &subSurface = *it;
|
|
|
|
if (subSurface.isNull() || subSurface->d_func()->surface.isNull()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
subSurface->d_func()->surface->frameRendered(msec);
|
|
|
|
}
|
2015-11-04 13:50:15 +00:00
|
|
|
if (needsFlush) {
|
|
|
|
client()->flush();
|
|
|
|
}
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::destroy()
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-09-19 05:06:31 +00:00
|
|
|
for (wl_resource *c : current.callbacks) {
|
2014-08-28 07:52:35 +00:00
|
|
|
wl_resource_destroy(c);
|
|
|
|
}
|
2014-09-19 05:06:31 +00:00
|
|
|
for (wl_resource *c : pending.callbacks) {
|
2014-08-28 07:52:35 +00:00
|
|
|
wl_resource_destroy(c);
|
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
for (wl_resource *c : subSurfacePending.callbacks) {
|
|
|
|
wl_resource_destroy(c);
|
|
|
|
}
|
2014-09-19 05:06:31 +00:00
|
|
|
if (current.buffer) {
|
|
|
|
current.buffer->unref();
|
2014-08-28 12:22:53 +00:00
|
|
|
}
|
2014-11-14 08:45:02 +00:00
|
|
|
if (resource) {
|
|
|
|
wl_resource_destroy(resource);
|
|
|
|
resource = nullptr;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-18 08:57:46 +00:00
|
|
|
void SurfaceInterface::Private::swapStates(State *source, State *target, bool emitChanged)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-14 09:55:06 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
2016-03-18 08:57:46 +00:00
|
|
|
const bool bufferChanged = source->bufferIsSet;
|
|
|
|
const bool opaqueRegionChanged = source->opaqueIsSet;
|
|
|
|
const bool inputRegionChanged = source->inputIsSet;
|
|
|
|
const bool scaleFactorChanged = source->scaleIsSet && (target->scale != source->scale);
|
|
|
|
const bool transformFactorChanged = source->transformIsSet && (target->transform != source->transform);
|
|
|
|
const bool shadowChanged = source->shadowIsSet;
|
|
|
|
const bool blurChanged = source->blurIsSet;
|
|
|
|
const bool contrastChanged = source->contrastIsSet;
|
|
|
|
const bool slideChanged = source->slideIsSet;
|
|
|
|
const bool childrenChanged = source->childrenChanged;
|
2015-03-03 09:16:11 +00:00
|
|
|
bool sizeChanged = false;
|
2016-03-18 08:57:46 +00:00
|
|
|
auto buffer = target->buffer;
|
2014-11-28 07:33:32 +00:00
|
|
|
if (bufferChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
// TODO: is the reffing correct for subsurfaces?
|
2015-03-03 09:16:11 +00:00
|
|
|
QSize oldSize;
|
2016-03-18 08:57:46 +00:00
|
|
|
if (target->buffer) {
|
|
|
|
oldSize = target->buffer->size();
|
|
|
|
if (emitChanged) {
|
|
|
|
target->buffer->unref();
|
|
|
|
QObject::disconnect(target->buffer, &BufferInterface::sizeChanged, q, &SurfaceInterface::sizeChanged);
|
|
|
|
} else {
|
|
|
|
delete target->buffer;
|
|
|
|
target->buffer = nullptr;
|
|
|
|
}
|
2014-11-28 07:33:32 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
if (source->buffer) {
|
|
|
|
if (emitChanged) {
|
|
|
|
source->buffer->ref();
|
|
|
|
QObject::connect(source->buffer, &BufferInterface::sizeChanged, q, &SurfaceInterface::sizeChanged);
|
|
|
|
}
|
|
|
|
const QSize newSize = source->buffer->size();
|
2015-03-03 09:16:11 +00:00
|
|
|
sizeChanged = newSize.isValid() && newSize != oldSize;
|
2014-11-28 07:33:32 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
buffer = source->buffer;
|
|
|
|
}
|
|
|
|
// copy values
|
|
|
|
if (bufferChanged) {
|
|
|
|
target->buffer = buffer;
|
|
|
|
target->damage = source->damage;
|
|
|
|
target->bufferIsSet = source->bufferIsSet;
|
|
|
|
}
|
|
|
|
if (childrenChanged) {
|
|
|
|
target->childrenChanged = source->childrenChanged;
|
|
|
|
target->children = source->children;
|
2014-08-28 12:22:53 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
target->callbacks.append(source->callbacks);
|
|
|
|
|
2015-07-15 09:07:50 +00:00
|
|
|
if (shadowChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
target->shadow = source->shadow;
|
|
|
|
target->shadowIsSet = true;
|
2015-07-15 09:07:50 +00:00
|
|
|
}
|
2015-08-26 12:42:58 +00:00
|
|
|
if (blurChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
target->blur = source->blur;
|
|
|
|
target->blurIsSet = true;
|
2015-08-26 12:42:58 +00:00
|
|
|
}
|
2015-09-02 16:13:25 +00:00
|
|
|
if (contrastChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
target->contrast = source->contrast;
|
|
|
|
target->contrastIsSet = true;
|
2015-09-02 16:13:25 +00:00
|
|
|
}
|
2015-09-09 11:04:11 +00:00
|
|
|
if (slideChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
target->slide = source->slide;
|
|
|
|
target->slideIsSet = true;
|
2015-09-09 11:04:11 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
if (inputRegionChanged) {
|
|
|
|
target->input = source->input;
|
|
|
|
target->inputIsInfinite = source->inputIsInfinite;
|
|
|
|
target->inputIsSet = true;
|
|
|
|
}
|
|
|
|
if (opaqueRegionChanged) {
|
|
|
|
target->opaque = source->opaque;
|
|
|
|
target->opaqueIsSet = true;
|
|
|
|
}
|
|
|
|
if (scaleFactorChanged) {
|
|
|
|
target->scale = source->scale;
|
|
|
|
target->scaleIsSet = true;
|
|
|
|
}
|
|
|
|
if (transformFactorChanged) {
|
|
|
|
target->transform = source->transform;
|
|
|
|
target->transformIsSet = true;
|
2014-10-14 12:04:35 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
|
|
|
|
*source = State{};
|
|
|
|
source->children = target->children;
|
2014-08-28 07:52:35 +00:00
|
|
|
if (opaqueRegionChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
emit q->opaqueChanged(target->opaque);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
if (inputRegionChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
emit q->inputChanged(target->input);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
if (scaleFactorChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
emit q->scaleChanged(target->scale);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
if (transformFactorChanged) {
|
2016-03-18 08:57:46 +00:00
|
|
|
emit q->transformChanged(target->transform);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
if (bufferChanged && emitChanged) {
|
|
|
|
if (!target->damage.isEmpty()) {
|
2015-05-26 09:37:00 +00:00
|
|
|
const QRegion windowRegion = QRegion(0, 0, q->size().width(), q->size().height());
|
|
|
|
if (!windowRegion.isEmpty()) {
|
2016-03-18 08:57:46 +00:00
|
|
|
target->damage = windowRegion.intersected(target->damage);
|
|
|
|
if (emitChanged) {
|
|
|
|
emit q->damaged(target->damage);
|
|
|
|
}
|
2015-05-26 09:37:00 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
} else if (!target->buffer && emitChanged) {
|
2014-11-28 07:33:32 +00:00
|
|
|
emit q->unmapped();
|
|
|
|
}
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
if (!emitChanged) {
|
|
|
|
return;
|
|
|
|
}
|
2015-03-03 09:16:11 +00:00
|
|
|
if (sizeChanged) {
|
|
|
|
emit q->sizeChanged();
|
|
|
|
}
|
2015-07-15 09:07:50 +00:00
|
|
|
if (shadowChanged) {
|
|
|
|
emit q->shadowChanged();
|
|
|
|
}
|
2015-08-26 12:42:58 +00:00
|
|
|
if (blurChanged) {
|
|
|
|
emit q->blurChanged();
|
|
|
|
}
|
2015-09-02 16:13:25 +00:00
|
|
|
if (contrastChanged) {
|
|
|
|
emit q->contrastChanged();
|
|
|
|
}
|
2015-09-09 11:04:11 +00:00
|
|
|
if (slideChanged) {
|
|
|
|
emit q->slideOnShowHideChanged();
|
|
|
|
}
|
2016-03-21 13:32:31 +00:00
|
|
|
if (childrenChanged) {
|
|
|
|
emit q->subSurfaceTreeChanged();
|
|
|
|
}
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2016-03-18 08:57:46 +00:00
|
|
|
void SurfaceInterface::Private::commit()
|
|
|
|
{
|
|
|
|
if (!subSurface.isNull() && subSurface->isSynchronized()) {
|
|
|
|
swapStates(&pending, &subSurfacePending, false);
|
|
|
|
} else {
|
|
|
|
swapStates(&pending, ¤t, true);
|
|
|
|
if (!subSurface.isNull()) {
|
|
|
|
subSurface->d_func()->commit();
|
|
|
|
}
|
|
|
|
// commit all subSurfaces to apply position changes
|
|
|
|
// "The cached state is applied to the sub-surface immediately after the parent surface's state is applied"
|
|
|
|
for (auto it = current.children.constBegin(); it != current.children.constEnd(); ++it) {
|
|
|
|
const auto &subSurface = *it;
|
|
|
|
if (subSurface.isNull()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
subSurface->d_func()->commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceInterface::Private::commitSubSurface()
|
|
|
|
{
|
|
|
|
if (subSurface.isNull() || !subSurface->isSynchronized()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
swapStates(&subSurfacePending, ¤t, true);
|
|
|
|
// "The cached state is applied to the sub-surface immediately after the parent surface's state is applied"
|
|
|
|
for (auto it = current.children.constBegin(); it != current.children.constEnd(); ++it) {
|
|
|
|
const auto &subSurface = *it;
|
|
|
|
if (subSurface.isNull() || !subSurface->isSynchronized()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
subSurface->d_func()->commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::damage(const QRect &rect)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-28 07:33:32 +00:00
|
|
|
if (!pending.bufferIsSet || (pending.bufferIsSet && !pending.buffer)) {
|
|
|
|
// TODO: should we send an error?
|
|
|
|
return;
|
|
|
|
}
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.damage = pending.damage.united(rect);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::setScale(qint32 scale)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.scale = scale;
|
2016-03-18 08:57:46 +00:00
|
|
|
pending.scaleIsSet = true;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::setTransform(OutputInterface::Transform transform)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.transform = transform;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::addFrameCallback(uint32_t callback)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-19 18:01:15 +00:00
|
|
|
wl_resource *r = client->createResource(&wl_callback_interface, 1, callback);
|
2014-08-28 07:52:35 +00:00
|
|
|
if (!r) {
|
2014-11-14 08:45:02 +00:00
|
|
|
wl_resource_post_no_memory(resource);
|
2014-08-28 07:52:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
wl_resource_set_implementation(r, nullptr, this, destroyFrameCallback);
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.callbacks << r;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::attachBuffer(wl_resource *buffer, const QPoint &offset)
|
2014-08-28 12:22:53 +00:00
|
|
|
{
|
2014-11-28 07:33:32 +00:00
|
|
|
pending.bufferIsSet = true;
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.offset = offset;
|
|
|
|
if (pending.buffer) {
|
|
|
|
delete pending.buffer;
|
2014-08-28 12:22:53 +00:00
|
|
|
}
|
2014-11-28 07:33:32 +00:00
|
|
|
if (!buffer) {
|
|
|
|
// got a null buffer, deletes content in next frame
|
|
|
|
pending.buffer = nullptr;
|
|
|
|
pending.damage = QRegion();
|
|
|
|
return;
|
|
|
|
}
|
2014-11-14 09:55:06 +00:00
|
|
|
Q_Q(SurfaceInterface);
|
2014-09-19 05:06:31 +00:00
|
|
|
pending.buffer = new BufferInterface(buffer, q);
|
2014-11-07 08:04:19 +00:00
|
|
|
QObject::connect(pending.buffer, &BufferInterface::aboutToBeDestroyed, q,
|
|
|
|
[this](BufferInterface *buffer) {
|
|
|
|
if (pending.buffer == buffer) {
|
|
|
|
pending.buffer = nullptr;
|
|
|
|
}
|
2016-03-18 08:57:46 +00:00
|
|
|
if (subSurfacePending.buffer == buffer) {
|
|
|
|
subSurfacePending.buffer = nullptr;
|
|
|
|
}
|
2014-11-07 08:04:19 +00:00
|
|
|
if (current.buffer == buffer) {
|
|
|
|
current.buffer->unref();
|
|
|
|
current.buffer = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
2014-08-28 12:22:53 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::destroyFrameCallback(wl_resource *r)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-14 09:20:43 +00:00
|
|
|
auto s = cast<Private>(r);
|
2014-09-19 05:06:31 +00:00
|
|
|
s->current.callbacks.removeAll(r);
|
|
|
|
s->pending.callbacks.removeAll(r);
|
2016-03-18 08:57:46 +00:00
|
|
|
s->subSurfacePending.callbacks.removeAll(r);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::destroyCallback(wl_client *client, wl_resource *resource)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2015-04-20 11:56:41 +00:00
|
|
|
wl_resource_destroy(resource);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::attachCallback(wl_client *client, wl_resource *resource, wl_resource *buffer, int32_t sx, int32_t sy)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2014-11-14 09:20:43 +00:00
|
|
|
cast<Private>(resource)->attachBuffer(buffer, QPoint(sx, sy));
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::damageCallback(wl_client *client, wl_resource *resource, int32_t x, int32_t y, int32_t width, int32_t height)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2014-11-14 09:20:43 +00:00
|
|
|
cast<Private>(resource)->damage(QRect(x, y, width, height));
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::frameCallaback(wl_client *client, wl_resource *resource, uint32_t callback)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-14 09:20:43 +00:00
|
|
|
auto s = cast<Private>(resource);
|
2014-11-19 15:53:56 +00:00
|
|
|
Q_ASSERT(client == *s->client);
|
2014-08-28 07:52:35 +00:00
|
|
|
s->addFrameCallback(callback);
|
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::opaqueRegionCallback(wl_client *client, wl_resource *resource, wl_resource *region)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-14 09:20:43 +00:00
|
|
|
auto s = cast<Private>(resource);
|
2014-11-19 15:53:56 +00:00
|
|
|
Q_ASSERT(client == *s->client);
|
2014-10-16 12:59:01 +00:00
|
|
|
auto r = RegionInterface::get(region);
|
|
|
|
s->setOpaque(r ? r->region() : QRegion());
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceInterface::Private::setOpaque(const QRegion ®ion)
|
|
|
|
{
|
|
|
|
pending.opaqueIsSet = true;
|
|
|
|
pending.opaque = region;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::inputRegionCallback(wl_client *client, wl_resource *resource, wl_resource *region)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
2014-11-14 09:20:43 +00:00
|
|
|
auto s = cast<Private>(resource);
|
2014-11-19 15:53:56 +00:00
|
|
|
Q_ASSERT(client == *s->client);
|
2014-10-16 12:59:01 +00:00
|
|
|
auto r = RegionInterface::get(region);
|
|
|
|
s->setInput(r ? r->region() : QRegion(), !r);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SurfaceInterface::Private::setInput(const QRegion ®ion, bool isInfinite)
|
|
|
|
{
|
|
|
|
pending.inputIsSet = true;
|
|
|
|
pending.inputIsInfinite = isInfinite;
|
|
|
|
pending.input = region;
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::commitCallback(wl_client *client, wl_resource *resource)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2014-11-14 09:20:43 +00:00
|
|
|
cast<Private>(resource)->commit();
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::bufferTransformCallback(wl_client *client, wl_resource *resource, int32_t transform)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2014-11-14 09:20:43 +00:00
|
|
|
cast<Private>(resource)->setTransform(OutputInterface::Transform(transform));
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
void SurfaceInterface::Private::bufferScaleCallback(wl_client *client, wl_resource *resource, int32_t scale)
|
2014-08-28 07:52:35 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(client)
|
2014-11-14 09:20:43 +00:00
|
|
|
cast<Private>(resource)->setScale(scale);
|
2014-09-19 05:06:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QRegion SurfaceInterface::damage() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.damage;
|
|
|
|
}
|
|
|
|
|
|
|
|
QRegion SurfaceInterface::opaque() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.opaque;
|
|
|
|
}
|
|
|
|
|
|
|
|
QRegion SurfaceInterface::input() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.input;
|
|
|
|
}
|
|
|
|
|
2014-10-16 12:59:01 +00:00
|
|
|
bool SurfaceInterface::inputIsInfitine() const
|
2015-09-16 14:43:58 +00:00
|
|
|
{
|
|
|
|
return inputIsInfinite();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SurfaceInterface::inputIsInfinite() const
|
2014-10-16 12:59:01 +00:00
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-10-16 12:59:01 +00:00
|
|
|
return d->current.inputIsInfinite;
|
|
|
|
}
|
|
|
|
|
2014-09-19 05:06:31 +00:00
|
|
|
qint32 SurfaceInterface::scale() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.scale;
|
|
|
|
}
|
|
|
|
|
|
|
|
OutputInterface::Transform SurfaceInterface::transform() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.transform;
|
|
|
|
}
|
|
|
|
|
|
|
|
BufferInterface *SurfaceInterface::buffer()
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPoint SurfaceInterface::offset() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-09-19 05:06:31 +00:00
|
|
|
return d->current.offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
SurfaceInterface *SurfaceInterface::get(wl_resource *native)
|
|
|
|
{
|
2014-11-14 14:13:06 +00:00
|
|
|
return Private::get<SurfaceInterface>(native);
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
|
2015-04-02 07:41:48 +00:00
|
|
|
SurfaceInterface *SurfaceInterface::get(quint32 id, const ClientConnection *client)
|
2015-02-09 13:31:20 +00:00
|
|
|
{
|
2015-04-02 07:41:48 +00:00
|
|
|
return Private::get<SurfaceInterface>(id, client);
|
2015-02-09 13:31:20 +00:00
|
|
|
}
|
|
|
|
|
2014-10-14 12:04:35 +00:00
|
|
|
QList< QPointer< SubSurfaceInterface > > SurfaceInterface::childSubSurfaces() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-10-14 12:04:35 +00:00
|
|
|
return d->current.children;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPointer< SubSurfaceInterface > SurfaceInterface::subSurface() const
|
|
|
|
{
|
2014-11-14 08:45:02 +00:00
|
|
|
Q_D();
|
2014-10-14 12:04:35 +00:00
|
|
|
return d->subSurface;
|
|
|
|
}
|
|
|
|
|
2015-03-03 09:16:11 +00:00
|
|
|
QSize SurfaceInterface::size() const
|
|
|
|
{
|
|
|
|
Q_D();
|
|
|
|
// TODO: apply transform and scale to the buffer size
|
|
|
|
if (d->current.buffer) {
|
|
|
|
return d->current.buffer->size();
|
|
|
|
}
|
|
|
|
return QSize();
|
|
|
|
}
|
|
|
|
|
2015-07-15 09:07:50 +00:00
|
|
|
QPointer< ShadowInterface > SurfaceInterface::shadow() const
|
|
|
|
{
|
|
|
|
Q_D();
|
|
|
|
return d->current.shadow;
|
|
|
|
}
|
|
|
|
|
2015-08-26 12:42:58 +00:00
|
|
|
QPointer< BlurInterface > SurfaceInterface::blur() const
|
|
|
|
{
|
|
|
|
Q_D();
|
|
|
|
return d->current.blur;
|
|
|
|
}
|
|
|
|
|
2015-09-02 16:13:25 +00:00
|
|
|
QPointer< ContrastInterface > SurfaceInterface::contrast() const
|
|
|
|
{
|
|
|
|
Q_D();
|
|
|
|
return d->current.contrast;
|
|
|
|
}
|
|
|
|
|
2015-09-09 11:04:11 +00:00
|
|
|
QPointer< SlideInterface > SurfaceInterface::slideOnShowHide() const
|
|
|
|
{
|
|
|
|
Q_D();
|
|
|
|
return d->current.slide;
|
|
|
|
}
|
|
|
|
|
2014-11-14 08:45:02 +00:00
|
|
|
SurfaceInterface::Private *SurfaceInterface::d_func() const
|
|
|
|
{
|
|
|
|
return reinterpret_cast<Private*>(d.data());
|
|
|
|
}
|
|
|
|
|
2014-08-28 07:52:35 +00:00
|
|
|
}
|
|
|
|
}
|