340 lines
9.7 KiB
C++
340 lines
9.7 KiB
C++
/*
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
SPDX-FileCopyrightText: 2006 Lubos Lunak <l.lunak@kde.org>
|
|
SPDX-FileCopyrightText: 2009, 2010, 2011 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
#ifndef KWIN_SCENE_OPENGL_H
|
|
#define KWIN_SCENE_OPENGL_H
|
|
|
|
#include "scene.h"
|
|
#include "shadow.h"
|
|
|
|
#include "kwinglutils.h"
|
|
|
|
#include "decorations/decorationrenderer.h"
|
|
#include "platformsupport/scenes/opengl/backend.h"
|
|
|
|
namespace KWin
|
|
{
|
|
class LanczosFilter;
|
|
class OpenGLBackend;
|
|
class SyncManager;
|
|
class SyncObject;
|
|
|
|
class KWIN_EXPORT SceneOpenGL
|
|
: public Scene
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
class EffectFrame;
|
|
~SceneOpenGL() override;
|
|
bool initFailed() const override;
|
|
bool hasPendingFlush() const override;
|
|
qint64 paint(const QRegion &damage, const QList<Toplevel *> &windows) override;
|
|
Scene::EffectFrame *createEffectFrame(EffectFrameImpl *frame) override;
|
|
Shadow *createShadow(Toplevel *toplevel) override;
|
|
void screenGeometryChanged(const QSize &size) override;
|
|
OverlayWindow *overlayWindow() const override;
|
|
bool usesOverlayWindow() const override;
|
|
bool blocksForRetrace() const override;
|
|
bool syncsToVBlank() const override;
|
|
bool makeOpenGLContextCurrent() override;
|
|
void doneOpenGLContextCurrent() override;
|
|
bool supportsSurfacelessContext() const override;
|
|
bool supportsNativeFence() const override;
|
|
Decoration::Renderer *createDecorationRenderer(Decoration::DecoratedClientImpl *impl) override;
|
|
void triggerFence() override;
|
|
virtual QMatrix4x4 projectionMatrix() const = 0;
|
|
bool animationsSupported() const override;
|
|
|
|
void insertWait();
|
|
|
|
void idle() override;
|
|
|
|
bool debug() const { return m_debug; }
|
|
void initDebugOutput();
|
|
|
|
/**
|
|
* @brief Factory method to create a backend specific texture.
|
|
*
|
|
* @return :SceneOpenGL::Texture*
|
|
*/
|
|
SceneOpenGLTexture *createTexture();
|
|
|
|
OpenGLBackend *backend() const {
|
|
return m_backend;
|
|
}
|
|
|
|
QVector<QByteArray> openGLPlatformInterfaceExtensions() const override;
|
|
QSharedPointer<GLTexture> textureForOutput(AbstractOutput *output) const override;
|
|
|
|
static SceneOpenGL *createScene(QObject *parent);
|
|
|
|
protected:
|
|
SceneOpenGL(OpenGLBackend *backend, QObject *parent = nullptr);
|
|
void paintBackground(const QRegion ®ion) override;
|
|
void aboutToStartPainting(int screenId, const QRegion &damage) override;
|
|
void extendPaintRegion(QRegion ®ion, bool opaqueFullscreen) override;
|
|
QMatrix4x4 transformation(int mask, const ScreenPaintData &data) const;
|
|
void paintDesktop(int desktop, int mask, const QRegion ®ion, ScreenPaintData &data) override;
|
|
void paintEffectQuickView(EffectQuickView *w) override;
|
|
|
|
void handleGraphicsReset(GLenum status);
|
|
|
|
virtual void doPaintBackground(const QVector<float> &vertices) = 0;
|
|
virtual void updateProjectionMatrix() = 0;
|
|
|
|
protected:
|
|
bool init_ok;
|
|
private:
|
|
bool viewportLimitsMatched(const QSize &size) const;
|
|
|
|
private:
|
|
bool m_debug;
|
|
OpenGLBackend *m_backend;
|
|
SyncManager *m_syncManager;
|
|
SyncObject *m_currentFence;
|
|
};
|
|
|
|
class SceneOpenGL2 : public SceneOpenGL
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
explicit SceneOpenGL2(OpenGLBackend *backend, QObject *parent = nullptr);
|
|
~SceneOpenGL2() override;
|
|
CompositingType compositingType() const override {
|
|
return OpenGL2Compositing;
|
|
}
|
|
|
|
static bool supported(OpenGLBackend *backend);
|
|
|
|
QMatrix4x4 projectionMatrix() const override { return m_projectionMatrix; }
|
|
QMatrix4x4 screenProjectionMatrix() const override { return m_screenProjectionMatrix; }
|
|
|
|
protected:
|
|
void paintSimpleScreen(int mask, const QRegion ®ion) override;
|
|
void paintGenericScreen(int mask, const ScreenPaintData &data) override;
|
|
void doPaintBackground(const QVector< float >& vertices) override;
|
|
Scene::Window *createWindow(Toplevel *t) override;
|
|
void finalDrawWindow(EffectWindowImpl* w, int mask, const QRegion ®ion, WindowPaintData& data) override;
|
|
void updateProjectionMatrix() override;
|
|
void paintCursor(const QRegion ®ion) override;
|
|
|
|
private:
|
|
void performPaintWindow(EffectWindowImpl* w, int mask, const QRegion ®ion, WindowPaintData& data);
|
|
QMatrix4x4 createProjectionMatrix() const;
|
|
|
|
private:
|
|
LanczosFilter *m_lanczosFilter;
|
|
QScopedPointer<GLTexture> m_cursorTexture;
|
|
QMatrix4x4 m_projectionMatrix;
|
|
QMatrix4x4 m_screenProjectionMatrix;
|
|
GLuint vao;
|
|
};
|
|
|
|
class OpenGLWindowPixmap;
|
|
|
|
class OpenGLWindow final : public Scene::Window
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
enum Leaf { ShadowLeaf, DecorationLeaf, ContentLeaf, PreviousContentLeaf };
|
|
|
|
struct RenderNode
|
|
{
|
|
RenderNode()
|
|
: texture(nullptr)
|
|
, firstVertex(0)
|
|
, vertexCount(0)
|
|
, opacity(1.0)
|
|
, hasAlpha(false)
|
|
, coordinateType(UnnormalizedCoordinates)
|
|
{
|
|
}
|
|
|
|
GLTexture *texture;
|
|
WindowQuadList quads;
|
|
int firstVertex;
|
|
int vertexCount;
|
|
float opacity;
|
|
bool hasAlpha;
|
|
TextureCoordinateType coordinateType;
|
|
Leaf leafType;
|
|
};
|
|
|
|
struct RenderContext
|
|
{
|
|
QVector<RenderNode> renderNodes;
|
|
int shadowOffset = 0;
|
|
int decorationOffset = 0;
|
|
int contentOffset = 0;
|
|
int previousContentOffset = 0;
|
|
int quadCount = 0;
|
|
};
|
|
|
|
OpenGLWindow(Toplevel *toplevel, SceneOpenGL *scene);
|
|
~OpenGLWindow() override;
|
|
|
|
WindowPixmap *createWindowPixmap() override;
|
|
void performPaint(int mask, const QRegion ®ion, const WindowPaintData &data) override;
|
|
QSharedPointer<GLTexture> windowTexture() override;
|
|
|
|
private:
|
|
QMatrix4x4 transformation(int mask, const WindowPaintData &data) const;
|
|
GLTexture *getDecorationTexture() const;
|
|
QMatrix4x4 modelViewProjectionMatrix(int mask, const WindowPaintData &data) const;
|
|
QVector4D modulate(float opacity, float brightness) const;
|
|
void setBlendEnabled(bool enabled);
|
|
void initializeRenderContext(RenderContext &context, const WindowPaintData &data);
|
|
bool beginRenderWindow(int mask, const QRegion ®ion, WindowPaintData &data);
|
|
void endRenderWindow();
|
|
bool bindTexture();
|
|
|
|
SceneOpenGL *m_scene;
|
|
bool m_hardwareClipping = false;
|
|
bool m_blendingEnabled = false;
|
|
};
|
|
|
|
class OpenGLWindowPixmap : public WindowPixmap
|
|
{
|
|
public:
|
|
explicit OpenGLWindowPixmap(Scene::Window *window, SceneOpenGL *scene);
|
|
~OpenGLWindowPixmap() override;
|
|
SceneOpenGLTexture *texture() const;
|
|
bool bind();
|
|
bool isValid() const override;
|
|
protected:
|
|
WindowPixmap *createChild(KWaylandServer::SubSurfaceInterface *subSurface) override;
|
|
private:
|
|
explicit OpenGLWindowPixmap(KWaylandServer::SubSurfaceInterface *subSurface, WindowPixmap *parent, SceneOpenGL *scene);
|
|
QScopedPointer<SceneOpenGLTexture> m_texture;
|
|
SceneOpenGL *m_scene;
|
|
};
|
|
|
|
class SceneOpenGL::EffectFrame
|
|
: public Scene::EffectFrame
|
|
{
|
|
public:
|
|
EffectFrame(EffectFrameImpl* frame, SceneOpenGL *scene);
|
|
~EffectFrame() override;
|
|
|
|
void free() override;
|
|
void freeIconFrame() override;
|
|
void freeTextFrame() override;
|
|
void freeSelection() override;
|
|
|
|
void render(const QRegion ®ion, double opacity, double frameOpacity) override;
|
|
|
|
void crossFadeIcon() override;
|
|
void crossFadeText() override;
|
|
|
|
static void cleanup();
|
|
|
|
private:
|
|
void updateTexture();
|
|
void updateTextTexture();
|
|
|
|
GLTexture *m_texture;
|
|
GLTexture *m_textTexture;
|
|
GLTexture *m_oldTextTexture;
|
|
QPixmap *m_textPixmap; // need to keep the pixmap around to workaround some driver problems
|
|
GLTexture *m_iconTexture;
|
|
GLTexture *m_oldIconTexture;
|
|
GLTexture *m_selectionTexture;
|
|
GLVertexBuffer *m_unstyledVBO;
|
|
SceneOpenGL *m_scene;
|
|
|
|
static GLTexture* m_unstyledTexture;
|
|
static QPixmap* m_unstyledPixmap; // need to keep the pixmap around to workaround some driver problems
|
|
static void updateUnstyledTexture(); // Update OpenGL unstyled frame texture
|
|
};
|
|
|
|
/**
|
|
* @short OpenGL implementation of Shadow.
|
|
*
|
|
* This class extends Shadow by the Elements required for OpenGL rendering.
|
|
* @author Martin Gräßlin <mgraesslin@kde.org>
|
|
*/
|
|
class SceneOpenGLShadow
|
|
: public Shadow
|
|
{
|
|
public:
|
|
explicit SceneOpenGLShadow(Toplevel *toplevel);
|
|
~SceneOpenGLShadow() override;
|
|
|
|
GLTexture *shadowTexture() {
|
|
return m_texture.data();
|
|
}
|
|
protected:
|
|
void buildQuads() override;
|
|
bool prepareBackend() override;
|
|
private:
|
|
QSharedPointer<GLTexture> m_texture;
|
|
};
|
|
|
|
class SceneOpenGLDecorationRenderer : public Decoration::Renderer
|
|
{
|
|
Q_OBJECT
|
|
public:
|
|
enum class DecorationPart : int {
|
|
Left,
|
|
Top,
|
|
Right,
|
|
Bottom,
|
|
Count
|
|
};
|
|
explicit SceneOpenGLDecorationRenderer(Decoration::DecoratedClientImpl *client);
|
|
~SceneOpenGLDecorationRenderer() override;
|
|
|
|
void render() override;
|
|
void reparent(Deleted *deleted) override;
|
|
|
|
GLTexture *texture() {
|
|
return m_texture.data();
|
|
}
|
|
GLTexture *texture() const {
|
|
return m_texture.data();
|
|
}
|
|
|
|
private:
|
|
void resizeTexture();
|
|
QScopedPointer<GLTexture> m_texture;
|
|
};
|
|
|
|
inline bool SceneOpenGL::hasPendingFlush() const
|
|
{
|
|
return m_backend->hasPendingFlush();
|
|
}
|
|
|
|
inline bool SceneOpenGL::usesOverlayWindow() const
|
|
{
|
|
return m_backend->usesOverlayWindow();
|
|
}
|
|
|
|
inline SceneOpenGLTexture* OpenGLWindowPixmap::texture() const
|
|
{
|
|
return m_texture.data();
|
|
}
|
|
|
|
class KWIN_EXPORT OpenGLFactory : public SceneFactory
|
|
{
|
|
Q_OBJECT
|
|
Q_INTERFACES(KWin::SceneFactory)
|
|
Q_PLUGIN_METADATA(IID "org.kde.kwin.Scene" FILE "opengl.json")
|
|
|
|
public:
|
|
explicit OpenGLFactory(QObject *parent = nullptr);
|
|
~OpenGLFactory() override;
|
|
|
|
Scene *create(QObject *parent = nullptr) const override;
|
|
};
|
|
|
|
} // namespace
|
|
|
|
#endif
|