542 lines
17 KiB
C++
542 lines
17 KiB
C++
/*
|
|
SPDX-FileCopyrightText: 2014 Martin Gräßlin <mgraesslin@kde.org>
|
|
SPDX-FileCopyrightText: 2020 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
|
|
|
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
|
*/
|
|
#pragma once
|
|
|
|
#include "core/colorspace.h"
|
|
#include "core/output.h"
|
|
#include "core/renderbackend.h"
|
|
|
|
#include <QMatrix4x4>
|
|
#include <QObject>
|
|
#include <QRegion>
|
|
|
|
struct wl_resource;
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
class GraphicsBuffer;
|
|
class BlurInterface;
|
|
class ClientConnection;
|
|
class ConfinedPointerV1Interface;
|
|
class ContrastInterface;
|
|
class CompositorInterface;
|
|
class LinuxDmaBufV1Feedback;
|
|
class LockedPointerV1Interface;
|
|
class OutputInterface;
|
|
class ShadowInterface;
|
|
class SlideInterface;
|
|
class SubSurfaceInterface;
|
|
class SurfaceInterfacePrivate;
|
|
class Transaction;
|
|
|
|
enum class PresentationHint {
|
|
VSync,
|
|
Async
|
|
};
|
|
|
|
/**
|
|
* The SurfaceRole class represents a role assigned to a wayland surface.
|
|
*/
|
|
class KWIN_EXPORT SurfaceRole
|
|
{
|
|
public:
|
|
explicit SurfaceRole(const QByteArray &name);
|
|
|
|
/**
|
|
* The human readable name of the surface role.
|
|
*/
|
|
QByteArray name() const;
|
|
|
|
private:
|
|
QByteArray m_name;
|
|
};
|
|
|
|
/**
|
|
* @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 GraphicsBuffer attached to it.
|
|
* As long as a GraphicsBuffer is attached, the released signal won't be sent. If the GraphicsBuffer
|
|
* 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 GraphicsBuffer
|
|
* @see SubSurfaceInterface
|
|
* @see BlurInterface
|
|
* @see ContrastInterface
|
|
* @see ShadowInterface
|
|
* @see SlideInterface
|
|
* @see LinuxDmaBufV1Feedback
|
|
*/
|
|
class KWIN_EXPORT SurfaceInterface : public QObject
|
|
{
|
|
Q_OBJECT
|
|
/**
|
|
* 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(QSizeF size READ size NOTIFY sizeChanged)
|
|
public:
|
|
explicit SurfaceInterface(CompositorInterface *compositor, wl_resource *resource);
|
|
~SurfaceInterface() override;
|
|
|
|
/**
|
|
* Returns the object id for this Wayland surface.
|
|
*/
|
|
uint32_t id() const;
|
|
/**
|
|
* Returns the Wayland client that owns this SurfaceInterface.
|
|
*/
|
|
ClientConnection *client() const;
|
|
/**
|
|
* Returns the Wayland resource corresponding to this SurfaceInterface.
|
|
*/
|
|
wl_resource *resource() const;
|
|
/**
|
|
* Returns the compositor for this SurfaceInterface.
|
|
*/
|
|
CompositorInterface *compositor() const;
|
|
|
|
/**
|
|
* Returns the role of this surface, or @c null if no role has been assigned to the surface.
|
|
*
|
|
* Once a role is given to the surface, it is permanent.
|
|
*/
|
|
SurfaceRole *role() const;
|
|
void setRole(SurfaceRole *role);
|
|
|
|
/**
|
|
* Maps the specified @a point from the surface-local coordinates to buffer pixel coordinates.
|
|
*
|
|
* Note that there is no direct connection between points in the surface-local coordinates
|
|
* and points in the buffer pixel coordinates. In order to map points between the two spaces,
|
|
* one has to use mapToBuffer() and mapFromBuffer().
|
|
*
|
|
* The returned value will become invalid when the surfaceToBufferMatrixChanged() signal is emitted.
|
|
*
|
|
* @see surfaceToBufferMatrix(), surfaceToBufferMatrixChanged()
|
|
*/
|
|
QPointF mapToBuffer(const QPointF &point) const;
|
|
/**
|
|
* Maps the specified @a region from the surface-local coordinates to buffer pixel coordinates.
|
|
*
|
|
* Note that there is no direct connection between regions in the surface-local coordinates
|
|
* and regions in the buffer pixel coordinates. In order to map regions between the two spaces,
|
|
* one has to use mapToBuffer() and mapFromBuffer().
|
|
*
|
|
* The returned value will become invalid when the surfaceToBufferMatrixChanged() signal is emitted.
|
|
*
|
|
* @see surfaceToBufferMatrix(), surfaceToBufferMatrixChanged()
|
|
*/
|
|
QRegion mapToBuffer(const QRegion ®ion) const;
|
|
/**
|
|
* Returns the projection matrix from the surface-local coordinates to buffer coordinates.
|
|
*
|
|
* @see surfaceToBufferMatrixChanged()
|
|
*/
|
|
QMatrix4x4 surfaceToBufferMatrix() const;
|
|
|
|
/**
|
|
* Maps the specified @a point in this surface's coordinate system to the equivalent point
|
|
* within the @a child's coordinate system, and returns the mapped point.
|
|
*
|
|
* If this surface is not an ancestor of the @a child, a null point is returned.
|
|
*/
|
|
QPointF mapToChild(SurfaceInterface *child, const QPointF &point) const;
|
|
|
|
void frameRendered(quint32 msec);
|
|
bool hasFrameCallbacks() const;
|
|
|
|
std::unique_ptr<PresentationFeedback> takePresentationFeedback(Output *output);
|
|
|
|
QRegion opaque() const;
|
|
QRegion input() const;
|
|
QRegion bufferDamage() const;
|
|
QRectF bufferSourceBox() const;
|
|
/**
|
|
* Returns the buffer transform that had been applied to the buffer to compensate for
|
|
* output rotation.
|
|
*
|
|
* If the surface is on an output that is rotated 90 degrees clockwise, the buffer will
|
|
* be rotated 90 degrees counter clockwise.
|
|
*/
|
|
OutputTransform bufferTransform() const;
|
|
/**
|
|
* @returns the current GraphicsBuffer, might be @c nullptr.
|
|
*/
|
|
GraphicsBuffer *buffer() const;
|
|
QPoint offset() const;
|
|
/**
|
|
* Returns the current size of the surface, in surface coordinates.
|
|
*
|
|
* Note that there is no direct relationship between the surface size and the buffer size.
|
|
* In order to determine the size of the currently attached buffer, use buffer()->size().
|
|
*/
|
|
QSizeF size() const;
|
|
/**
|
|
* Returns the rectangle that bounds this surface and all of its sub-surfaces.
|
|
*
|
|
* QPoint(0, 0) corresponds to the upper left corner of this surface.
|
|
*/
|
|
QRectF boundingRect() const;
|
|
/**
|
|
* Returns the size of the attached buffer, in device pixels.
|
|
*
|
|
* If no buffer is attached to this surface, an invalid QSize will be returned.
|
|
*/
|
|
QSize bufferSize() const;
|
|
|
|
/**
|
|
* @returns The SubSurface for this Surface in case there is one.
|
|
*/
|
|
SubSurfaceInterface *subSurface() const;
|
|
/**
|
|
* Returns the sub-surfaces that are below this surface. The sub-surfaces are sorted
|
|
* from bottom to top.
|
|
*/
|
|
QList<SubSurfaceInterface *> below() const;
|
|
/**
|
|
* Returns the sub-surfaces that are above this surface. The sub-surfaces are sorted
|
|
* from bottom to top.
|
|
*/
|
|
QList<SubSurfaceInterface *> above() const;
|
|
|
|
/**
|
|
* @returns The Shadow for this Surface.
|
|
*/
|
|
ShadowInterface *shadow() const;
|
|
|
|
/**
|
|
* @returns The Blur for this Surface.
|
|
*/
|
|
BlurInterface *blur() const;
|
|
|
|
/**
|
|
* @returns The Slide for this Surface.
|
|
*/
|
|
SlideInterface *slideOnShowHide() const;
|
|
|
|
/**
|
|
* @returns The Contrast for this Surface.
|
|
*/
|
|
ContrastInterface *contrast() const;
|
|
|
|
/**
|
|
* Whether the SurfaceInterface is currently considered to be mapped.
|
|
* A SurfaceInterface is mapped if it has a non-null GraphicsBuffer attached.
|
|
* If the SurfaceInterface references a SubSurfaceInterface it is only considered
|
|
* mapped if it has a GraphicsBuffer attached and the parent SurfaceInterface is mapped.
|
|
*
|
|
* @returns Whether the SurfaceInterface is currently mapped
|
|
*/
|
|
bool isMapped() const;
|
|
|
|
/**
|
|
* 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
|
|
*/
|
|
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
|
|
*/
|
|
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
|
|
*/
|
|
void setOutputs(const QList<OutputInterface *> &outputs, OutputInterface *primaryOutput);
|
|
|
|
/**
|
|
* @returns All OutputInterfaces the SurfaceInterface is on.
|
|
* @see setOutputs
|
|
*/
|
|
QList<OutputInterface *> outputs() const;
|
|
|
|
/**
|
|
* Pointer confinement installed on this SurfaceInterface.
|
|
* @see pointerConstraintsChanged
|
|
*/
|
|
ConfinedPointerV1Interface *confinedPointer() const;
|
|
|
|
/**
|
|
* Pointer lock installed on this SurfaceInterface.
|
|
* @see pointerConstraintsChanged
|
|
*/
|
|
LockedPointerV1Interface *lockedPointer() const;
|
|
|
|
/**
|
|
* @returns Whether this SurfaceInterface wants idle to be inhibited on the Output it is shown
|
|
* @see inhibitsIdleChanged
|
|
*/
|
|
bool inhibitsIdle() const;
|
|
|
|
/**
|
|
* dmabuf feedback installed on this SurfaceInterface
|
|
*/
|
|
LinuxDmaBufV1Feedback *dmabufFeedbackV1() const;
|
|
|
|
/**
|
|
* @returns the current content type of this surface
|
|
*/
|
|
ContentType contentType() 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.
|
|
*/
|
|
static SurfaceInterface *get(quint32 id, const ClientConnection *client);
|
|
|
|
/**
|
|
* @see ClientConnection::setScaleOverride
|
|
*/
|
|
qreal scaleOverride() const;
|
|
/**
|
|
* Convert a co-ordinate from kwin logical space to surface logical space
|
|
* @internal
|
|
*/
|
|
QPoint toSurfaceLocal(const QPoint &point) const;
|
|
/**
|
|
* Convert a co-ordinate from kwin logical space to surface logical space
|
|
* @internal
|
|
*/
|
|
QPointF toSurfaceLocal(const QPointF &point) const;
|
|
|
|
/**
|
|
* @returns if the client thinks the content of this surface is suitable for presentation with tearing
|
|
*/
|
|
PresentationHint presentationHint() const;
|
|
|
|
/**
|
|
* Sets a preferred buffer scale that clients should provide buffers in
|
|
* @param scale
|
|
*/
|
|
void setPreferredBufferScale(qreal scale);
|
|
|
|
/**
|
|
* Sets the preferred buffer transform for this surface.
|
|
*
|
|
* This indicates to the client the preferred buffer transform to use when
|
|
* attaching buffers to this surface.
|
|
*/
|
|
void setPreferredBufferTransform(OutputTransform transform);
|
|
|
|
/**
|
|
* The first committed transaction that is scheduled to be applied to this surface.
|
|
*/
|
|
Transaction *firstTransaction() const;
|
|
void setFirstTransaction(Transaction *transaction);
|
|
|
|
/**
|
|
* The last committed transaction that is scheduled to be applied to this surface.
|
|
*/
|
|
Transaction *lastTransaction() const;
|
|
void setLastTransaction(Transaction *transaction);
|
|
|
|
const ColorDescription &colorDescription() const;
|
|
|
|
void setPreferredColorDescription(const ColorDescription &descr);
|
|
|
|
/**
|
|
* Traverses the surface sub-tree with this surface as the root.
|
|
*/
|
|
void traverseTree(std::function<void(SurfaceInterface *surface)> callback);
|
|
|
|
Q_SIGNALS:
|
|
/**
|
|
* This signal is emitted when the underlying wl_surface resource is about to be freed.
|
|
*
|
|
* The unbound() signal is emitted either when the client that owns the surface has been
|
|
* destroyed or if the surface has been destroyed due to a destructor request.
|
|
*
|
|
* The SurfaceInterface object and the associated wl_surface resource are valid when this
|
|
* signal is emitted.
|
|
*/
|
|
void aboutToBeDestroyed();
|
|
/**
|
|
* This signal is emitted when the projection matrix from the surface-local coordinate space
|
|
* to the buffer coordinate space has been changed.
|
|
*
|
|
* Note that the compositor will most likely need to re-compute the texture coordinates after
|
|
* the surface-to-buffer matrix has been changed.
|
|
*/
|
|
void surfaceToBufferMatrixChanged();
|
|
/**
|
|
* Emitted whenever the SurfaceInterface got damaged.
|
|
* The signal is only emitted during the commit of state.
|
|
* A damage means that a new GraphicsBuffer got attached.
|
|
*
|
|
* @see buffer
|
|
* @see damage
|
|
*/
|
|
void damaged(const QRegion &);
|
|
void opaqueChanged(const QRegion &);
|
|
void inputChanged(const QRegion &);
|
|
/**
|
|
* This signal is emitted when the buffer transform has changed.
|
|
*/
|
|
void bufferTransformChanged(KWin::OutputTransform);
|
|
void bufferSourceBoxChanged();
|
|
/**
|
|
* This signal is emitted when the size of the attached buffer has changed.
|
|
*/
|
|
void bufferSizeChanged();
|
|
/**
|
|
* Emitted when the Surface becomes visible, i.e. a non-null buffer has been attached.
|
|
*/
|
|
void mapped();
|
|
/**
|
|
* Emitted when the Surface removes its content
|
|
*/
|
|
void unmapped();
|
|
/**
|
|
* This signal is emitted when the surface size has changed.
|
|
*/
|
|
void sizeChanged();
|
|
void shadowChanged();
|
|
void blurChanged();
|
|
void slideOnShowHideChanged();
|
|
void contrastChanged();
|
|
/**
|
|
* Emitted whenever a new child sub-surface @p subSurface is added.
|
|
*/
|
|
void childSubSurfaceAdded(SubSurfaceInterface *subSurface);
|
|
/**
|
|
* Emitted whenver the child sub-surface @p subSurface is removed.
|
|
*/
|
|
void childSubSurfaceRemoved(SubSurfaceInterface *subSurface);
|
|
/**
|
|
* This signal is emitted when the list of child subsurfaces changes.
|
|
*/
|
|
void childSubSurfacesChanged();
|
|
|
|
/**
|
|
* 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
|
|
*/
|
|
void pointerConstraintsChanged();
|
|
|
|
/**
|
|
* Emitted whenever the SurfaceInterface starts/ends to inhibit idle.
|
|
* @see inhibitsIdle
|
|
*/
|
|
void inhibitsIdleChanged();
|
|
|
|
void colorDescriptionChanged();
|
|
|
|
/**
|
|
* Emitted when the Surface has been committed.
|
|
*
|
|
* This signal is emitted after all the relevant damage and xyzChanged signals
|
|
* for this commit are emitted.
|
|
*/
|
|
void committed();
|
|
|
|
/**
|
|
* This signal is emitted when a surface commit with the specified \a serial has been cached
|
|
* to be applied later.
|
|
*/
|
|
void stateStashed(quint32 serial);
|
|
|
|
/**
|
|
* This signal is emitted when the state in a surface commit with the specified \a serial
|
|
* has been applied.
|
|
*/
|
|
void stateApplied(quint32 serial);
|
|
|
|
private:
|
|
std::unique_ptr<SurfaceInterfacePrivate> d;
|
|
friend class SurfaceInterfacePrivate;
|
|
};
|
|
|
|
/**
|
|
* The SurfaceExtension class is the base class for wl_surface extensions. The SurfaceExtension
|
|
* helps with managing extension state and keeping it in sync with the surface state.
|
|
*/
|
|
template<typename Commit>
|
|
class SurfaceExtension : public QObject
|
|
{
|
|
public:
|
|
explicit SurfaceExtension(SurfaceInterface *surface)
|
|
{
|
|
connect(surface, &SurfaceInterface::stateStashed, this, &SurfaceExtension::stashState);
|
|
connect(surface, &SurfaceInterface::stateApplied, this, &SurfaceExtension::applyState);
|
|
}
|
|
|
|
virtual void apply(Commit *commit) = 0;
|
|
|
|
Commit pending;
|
|
QMap<quint32, Commit> stashed;
|
|
|
|
private:
|
|
void stashState(quint32 serial)
|
|
{
|
|
Commit stash = std::exchange(pending, Commit{});
|
|
stashed.insert(serial, stash);
|
|
}
|
|
|
|
void applyState(quint32 serial)
|
|
{
|
|
if (!stashed.isEmpty()) {
|
|
if (stashed.firstKey() == serial) {
|
|
Commit stash = stashed.take(serial);
|
|
apply(&stash);
|
|
}
|
|
return;
|
|
}
|
|
|
|
apply(&pending);
|
|
pending = Commit{};
|
|
}
|
|
};
|
|
|
|
} // namespace KWin
|
|
|
|
Q_DECLARE_METATYPE(KWin::SurfaceInterface *)
|