kwin/src/wayland/surface_interface.h
Roman Gilg 96967485ba [server] Respect input region of sub-surfaces on pointer surface focus
Summary:
KWayland takes always the top-most child surface at a given position for its
pointer input. But if a sub-surface sets its input region, it should not select
this one when the position is out of its input region, but rather try the
surface below.

Test Plan:
My testing was only on my Xwayland branch. Supposed to also fix a problem
with Firefox native Wayland port.

Reviewers: #frameworks, graesslin, davidedmundson

Reviewed By: davidedmundson

Subscribers: davidedmundson, zzag, kde-frameworks-devel, graesslin, plasma-devel

Tags: #frameworks, #plasma_on_wayland

Differential Revision: https://phabricator.kde.org/D7038
2018-10-24 13:57:48 +02:00

361 lines
12 KiB
C++

/********************************************************************
Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
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.
This library 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
Lesser General Public License for more details.
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/>.
*********************************************************************/
#ifndef WAYLAND_SERVER_SURFACE_INTERFACE_H
#define WAYLAND_SERVER_SURFACE_INTERFACE_H
#include "resource.h"
#include "output_interface.h"
#include <QObject>
#include <QPointer>
#include <QRegion>
#include <KWayland/Server/kwaylandserver_export.h>
namespace KWayland
{
namespace Server
{
class BlurManagerInterface;
class BlurInterface;
class BufferInterface;
class ConfinedPointerInterface;
class ContrastInterface;
class ContrastManagerInterface;
class CompositorInterface;
class IdleInhibitManagerUnstableV1Interface;
class LockedPointerInterface;
class PointerConstraintsUnstableV1Interface;
class ShadowManagerInterface;
class ShadowInterface;
class SlideInterface;
class SubSurfaceInterface;
/**
* @brief Resource representing a wl_surface.
*
* The SurfaceInterface gets created by the CompositorInterface. A SurfaceInterface normally
* takes up a role by being "attached" to either a ShellSurfaceInterface, a SubSurfaceInterface
* or a Cursor.
*
* The implementation of the SurfaceInterface does not only wrap the features exposed by wl_surface,
* but goes further by integrating the information added to a SurfaceInterface by other interfaces.
* This should make interacting from the server easier, it only needs to monitor the SurfaceInterface
* and does not need to track each specific interface.
*
* The SurfaceInterface takes care of reference/unreferencing the BufferInterface attached to it.
* As long as a BufferInterface is attached, the released signal won't be sent. If the BufferInterface
* is no longer needed by the SurfaceInterface, it will get unreferenced and might be automatically
* deleted (if it's no longer referenced).
*
* @see CompositorInterface
* @see BufferInterface
* @see SubSurfaceInterface
* @see BlurInterface
* @see ContrastInterface
* @see ShadowInterface
* @see SlideInterface
**/
class KWAYLANDSERVER_EXPORT SurfaceInterface : public Resource
{
Q_OBJECT
/**
* The current damage region.
**/
Q_PROPERTY(QRegion damage READ damage NOTIFY damaged)
/**
* The opaque region for a translucent buffer.
**/
Q_PROPERTY(QRegion opaque READ opaque NOTIFY opaqueChanged)
/**
* The current input region.
**/
Q_PROPERTY(QRegion input READ input NOTIFY inputChanged)
Q_PROPERTY(qint32 scale READ scale NOTIFY scaleChanged)
Q_PROPERTY(KWayland::Server::OutputInterface::Transform transform READ transform NOTIFY transformChanged)
Q_PROPERTY(QSize size READ size NOTIFY sizeChanged)
public:
virtual ~SurfaceInterface();
void frameRendered(quint32 msec);
QRegion damage() const;
QRegion opaque() const;
QRegion input() const;
/**
* Use Surface::inputIsInfinite instead.
* @deprecated
* @see inputIsInfinite
*/
bool inputIsInfitine() const;
/**
* Replaces Surface::inputIsInfitine instead.
* @since 5.5
*/
bool inputIsInfinite() const;
qint32 scale() const;
OutputInterface::Transform transform() const;
/**
* @returns the current BufferInterface, might be @c nullptr.
**/
BufferInterface *buffer();
QPoint offset() const;
/**
* The size of the Surface in global compositor space.
* @see For buffer size use BufferInterface::size
* from SurfaceInterface::buffer
* @since 5.3
**/
QSize size() const;
/**
* @returns The SubSurface for this Surface in case there is one.
**/
QPointer<SubSurfaceInterface> subSurface() const;
/**
* @returns Children in stacking order from bottom (first) to top (last).
**/
QList<QPointer<SubSurfaceInterface>> childSubSurfaces() const;
/**
* @returns The Shadow for this Surface.
* @since 5.4
**/
QPointer<ShadowInterface> shadow() const;
/**
* @returns The Blur for this Surface.
* @since 5.5
**/
QPointer<BlurInterface> blur() const;
/**
* @returns The Slide for this Surface.
* @since 5.5
**/
QPointer<SlideInterface> slideOnShowHide() const;
/**
* @returns The Contrast for this Surface.
* @since 5.5
**/
QPointer<ContrastInterface> contrast() const;
/**
* Whether the SurfaceInterface is currently considered to be mapped.
* A SurfaceInterface is mapped if it has a non-null BufferInterface attached.
* If the SurfaceInterface references a SubSurfaceInterface it is only considered
* mapped if it has a BufferInterface attached and the parent SurfaceInterface is mapped.
*
* @returns Whether the SurfaceInterface is currently mapped
* @since 5.22
**/
bool isMapped() const;
/**
* Returns the tracked damage since the last call to {@link resetTrackedDamage}.
* In contrast to {@link damage} this method does not reset the damage when
* a new BufferInterface gets committed. This allows a compositor to properly
* track the damage over multiple commits even if it didn't render each new
* BufferInterface.
*
* The damage gets reset whenever {@link resetTrackedDamage} is called.
* This allows a compositor to properly track the change in its rendering scene
* for this SurfaceInterface. After it updates its internal state (e.g. by creating
* an OpenGL texture from the BufferInterface) it can invoke {@link resetTrackedDamage}
* and the damage tracker will start to track further damage changes.
*
* @returns Combined damage since last call to resetTrackedDamage
* @see damage
* @see resetTrackedDamage
* @since 5.22
**/
QRegion trackedDamage() const;
/**
* Reset the damage tracking. The compositor should invoke this method once it updated
* it's internal state and processed the current damage.
* @see trackedDamage
* @since 5.22
**/
void resetTrackedDamage();
/**
* Finds the SurfaceInterface at the given @p position in surface-local coordinates.
* This can be either a descendant SurfaceInterface honoring the stacking order or
* the SurfaceInterface itself if its geometry contains the given @p position.
*
* If no such SurfaceInterface is found, e.g. because the SurfaceInterface is unmapped,
* @c nullptr is returned.
*
* @param position The position in surface-local coordinates
* @returns Child surface at the given @p position or surface itself at the position, might be @c nullptr
* @since 5.22
**/
SurfaceInterface *surfaceAt(const QPointF &position);
/**
* Finds the input receiving SurfaceInterface at the given @p position in surface-local coordinates.
* This can be either a descendant SurfaceInterface honoring the stacking order or
* the SurfaceInterface itself if its geometry contains the given @p position.
*
* If no such SurfaceInterface is found, e.g. because the SurfaceInterface is unmapped or there is no
* input region containing the position,
* @c nullptr is returned.
*
* @param position The position in surface-local coordinates
* @returns Input receiving child surface at the given @p position or surface itself at the position, might be @c nullptr
* @since 5.52
**/
SurfaceInterface *inputSurfaceAt(const QPointF &position);
/**
* Sets the @p outputs this SurfaceInterface overlaps with, may be empty.
*
* The compositor should update whenever the SurfaceInterface becomes visible on
* an OutputInterface by e.g. getting (un)mapped, resized, moved, etc.
*
* @see outputs
* @since 5.27
**/
void setOutputs(const QVector<OutputInterface *> &outputs);
/**
* @returns All OutputInterfaces the SurfaceInterface is on.
* @see setOutputs
* @since 5.27
**/
QVector<OutputInterface *> outputs() const;
/**
* Pointer confinement installed on this SurfaceInterface.
* @see pointerConstraintsChanged
* @since 5.29
**/
QPointer<ConfinedPointerInterface> confinedPointer() const;
/**
* Pointer lock installed on this SurfaceInterface.
* @see pointerConstraintsChanged
* @since 5.29
**/
QPointer<LockedPointerInterface> lockedPointer() const;
/**
* @returns Whether this SurfaceInterface wants idle to be inhibited on the Output it is shown
* @see inhibitsIdleChanged
* @since 5.41
**/
bool inhibitsIdle() const;
/**
* @returns The SurfaceInterface for the @p native resource.
**/
static SurfaceInterface *get(wl_resource *native);
/**
* @returns The SurfaceInterface with given @p id for @p client, if it exists, otherwise @c nullptr.
* @since 5.3
**/
static SurfaceInterface *get(quint32 id, const ClientConnection *client);
Q_SIGNALS:
/**
* Emitted whenever the SurfaceInterface got damaged.
* The signal is only emitted during the commit of state.
* A damage means that a new BufferInterface got attached.
*
* @see buffer
* @see damage
**/
void damaged(const QRegion&);
void opaqueChanged(const QRegion&);
void inputChanged(const QRegion&);
void scaleChanged(qint32);
void transformChanged(KWayland::Server::OutputInterface::Transform);
/**
* Emitted when the Surface removes its content
**/
void unmapped();
/**
* @since 5.3
**/
void sizeChanged();
/**
* @since 5.4
**/
void shadowChanged();
/**
* @since 5.5
**/
void blurChanged();
/**
* @since 5.5
**/
void slideOnShowHideChanged();
/**
* @since 5.5
**/
void contrastChanged();
/**
* Emitted whenever the tree of sub-surfaces changes in a way which requires a repaint.
* @since 5.22
**/
void subSurfaceTreeChanged();
/**
* Emitted whenever a pointer constraint get (un)installed on this SurfaceInterface.
*
* The pointer constraint does not get activated, the compositor needs to activate
* the lock/confinement.
*
* @see confinedPointer
* @see lockedPointer
* @since 5.29
**/
void pointerConstraintsChanged();
/**
* Emitted whenever the SurfaceInterface starts/ends to inhibit idle.
* @see inhibitsIdle
* @since 5.41
**/
void inhibitsIdleChanged();
private:
friend class CompositorInterface;
friend class SubSurfaceInterface;
friend class ShadowManagerInterface;
friend class BlurManagerInterface;
friend class SlideManagerInterface;
friend class ContrastManagerInterface;
friend class IdleInhibitManagerUnstableV1Interface;
friend class PointerConstraintsUnstableV1Interface;
explicit SurfaceInterface(CompositorInterface *parent, wl_resource *parentResource);
class Private;
Private *d_func() const;
};
}
}
Q_DECLARE_METATYPE(KWayland::Server::SurfaceInterface*)
#endif