355 lines
10 KiB
C
355 lines
10 KiB
C
|
/********************************************************************
|
||
|
KWin - the KDE window manager
|
||
|
This file is part of the KDE project.
|
||
|
|
||
|
Copyright (C) 2006 Lubos Lunak <l.lunak@kde.org>
|
||
|
Copyright (C) 2009, 2010, 2011 Martin Gräßlin <mgraesslin@kde.org>
|
||
|
|
||
|
This program is free software; you can redistribute it and/or modify
|
||
|
it under the terms of the GNU General Public License as published by
|
||
|
the Free Software Foundation; either version 2 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
This program 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 General Public License for more details.
|
||
|
|
||
|
You should have received a copy of the GNU General Public License
|
||
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*********************************************************************/
|
||
|
|
||
|
#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;
|
||
|
class Window;
|
||
|
virtual ~SceneOpenGL();
|
||
|
virtual bool initFailed() const;
|
||
|
virtual bool hasPendingFlush() const;
|
||
|
virtual qint64 paint(QRegion damage, ToplevelList windows);
|
||
|
virtual Scene::EffectFrame *createEffectFrame(EffectFrameImpl *frame);
|
||
|
virtual Shadow *createShadow(Toplevel *toplevel);
|
||
|
virtual void screenGeometryChanged(const QSize &size);
|
||
|
virtual OverlayWindow *overlayWindow();
|
||
|
virtual bool usesOverlayWindow() const;
|
||
|
virtual bool blocksForRetrace() const;
|
||
|
virtual bool syncsToVBlank() const;
|
||
|
virtual bool makeOpenGLContextCurrent() override;
|
||
|
virtual void doneOpenGLContextCurrent() override;
|
||
|
Decoration::Renderer *createDecorationRenderer(Decoration::DecoratedClientImpl *impl) override;
|
||
|
virtual void triggerFence() override;
|
||
|
virtual QMatrix4x4 projectionMatrix() const = 0;
|
||
|
bool animationsSupported() const override;
|
||
|
|
||
|
void insertWait();
|
||
|
|
||
|
void idle();
|
||
|
|
||
|
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;
|
||
|
|
||
|
static SceneOpenGL *createScene(QObject *parent);
|
||
|
|
||
|
protected:
|
||
|
SceneOpenGL(OpenGLBackend *backend, QObject *parent = nullptr);
|
||
|
virtual void paintBackground(QRegion region);
|
||
|
virtual void extendPaintRegion(QRegion ®ion, bool opaqueFullscreen);
|
||
|
QMatrix4x4 transformation(int mask, const ScreenPaintData &data) const;
|
||
|
virtual void paintDesktop(int desktop, int mask, const QRegion ®ion, ScreenPaintData &data);
|
||
|
|
||
|
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);
|
||
|
virtual ~SceneOpenGL2();
|
||
|
virtual CompositingType compositingType() const {
|
||
|
return OpenGL2Compositing;
|
||
|
}
|
||
|
|
||
|
static bool supported(OpenGLBackend *backend);
|
||
|
|
||
|
QMatrix4x4 projectionMatrix() const override { return m_projectionMatrix; }
|
||
|
QMatrix4x4 screenProjectionMatrix() const override { return m_screenProjectionMatrix; }
|
||
|
|
||
|
protected:
|
||
|
virtual void paintSimpleScreen(int mask, QRegion region);
|
||
|
virtual void paintGenericScreen(int mask, ScreenPaintData data);
|
||
|
virtual void doPaintBackground(const QVector< float >& vertices);
|
||
|
virtual Scene::Window *createWindow(Toplevel *t);
|
||
|
virtual void finalDrawWindow(EffectWindowImpl* w, int mask, QRegion region, WindowPaintData& data);
|
||
|
virtual void updateProjectionMatrix() override;
|
||
|
void paintCursor() override;
|
||
|
|
||
|
private:
|
||
|
void performPaintWindow(EffectWindowImpl* w, int mask, QRegion region, WindowPaintData& data);
|
||
|
QMatrix4x4 createProjectionMatrix() const;
|
||
|
|
||
|
private:
|
||
|
LanczosFilter *m_lanczosFilter;
|
||
|
QScopedPointer<GLTexture> m_cursorTexture;
|
||
|
QMatrix4x4 m_projectionMatrix;
|
||
|
QMatrix4x4 m_screenProjectionMatrix;
|
||
|
GLuint vao;
|
||
|
};
|
||
|
|
||
|
class SceneOpenGL::Window
|
||
|
: public Scene::Window
|
||
|
{
|
||
|
public:
|
||
|
virtual ~Window();
|
||
|
bool beginRenderWindow(int mask, const QRegion ®ion, WindowPaintData &data);
|
||
|
virtual void performPaint(int mask, QRegion region, WindowPaintData data) = 0;
|
||
|
void endRenderWindow();
|
||
|
bool bindTexture();
|
||
|
void setScene(SceneOpenGL *scene) {
|
||
|
m_scene = scene;
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
virtual WindowPixmap* createWindowPixmap();
|
||
|
Window(Toplevel* c);
|
||
|
enum TextureType {
|
||
|
Content,
|
||
|
Decoration,
|
||
|
Shadow
|
||
|
};
|
||
|
|
||
|
QMatrix4x4 transformation(int mask, const WindowPaintData &data) const;
|
||
|
GLTexture *getDecorationTexture() const;
|
||
|
|
||
|
protected:
|
||
|
SceneOpenGL *m_scene;
|
||
|
bool m_hardwareClipping;
|
||
|
};
|
||
|
|
||
|
class SceneOpenGL2Window : public SceneOpenGL::Window
|
||
|
{
|
||
|
public:
|
||
|
enum Leaf { ShadowLeaf = 0, DecorationLeaf, ContentLeaf, PreviousContentLeaf, LeafCount };
|
||
|
|
||
|
struct LeafNode
|
||
|
{
|
||
|
LeafNode()
|
||
|
: texture(0),
|
||
|
firstVertex(0),
|
||
|
vertexCount(0),
|
||
|
opacity(1.0),
|
||
|
hasAlpha(false),
|
||
|
coordinateType(UnnormalizedCoordinates)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
GLTexture *texture;
|
||
|
int firstVertex;
|
||
|
int vertexCount;
|
||
|
float opacity;
|
||
|
bool hasAlpha;
|
||
|
TextureCoordinateType coordinateType;
|
||
|
};
|
||
|
|
||
|
explicit SceneOpenGL2Window(Toplevel *c);
|
||
|
virtual ~SceneOpenGL2Window();
|
||
|
|
||
|
protected:
|
||
|
QMatrix4x4 modelViewProjectionMatrix(int mask, const WindowPaintData &data) const;
|
||
|
QVector4D modulate(float opacity, float brightness) const;
|
||
|
void setBlendEnabled(bool enabled);
|
||
|
void setupLeafNodes(LeafNode *nodes, const WindowQuadList *quads, const WindowPaintData &data);
|
||
|
virtual void performPaint(int mask, QRegion region, WindowPaintData data);
|
||
|
|
||
|
private:
|
||
|
/**
|
||
|
* Whether prepareStates enabled blending and restore states should disable again.
|
||
|
**/
|
||
|
bool m_blendingEnabled;
|
||
|
};
|
||
|
|
||
|
class OpenGLWindowPixmap : public WindowPixmap
|
||
|
{
|
||
|
public:
|
||
|
explicit OpenGLWindowPixmap(Scene::Window *window, SceneOpenGL *scene);
|
||
|
virtual ~OpenGLWindowPixmap();
|
||
|
SceneOpenGLTexture *texture() const;
|
||
|
bool bind();
|
||
|
bool isValid() const override;
|
||
|
protected:
|
||
|
WindowPixmap *createChild(const QPointer<KWayland::Server::SubSurfaceInterface> &subSurface) override;
|
||
|
private:
|
||
|
explicit OpenGLWindowPixmap(const QPointer<KWayland::Server::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);
|
||
|
virtual ~EffectFrame();
|
||
|
|
||
|
virtual void free();
|
||
|
virtual void freeIconFrame();
|
||
|
virtual void freeTextFrame();
|
||
|
virtual void freeSelection();
|
||
|
|
||
|
virtual void render(QRegion region, double opacity, double frameOpacity);
|
||
|
|
||
|
virtual void crossFadeIcon();
|
||
|
virtual void crossFadeText();
|
||
|
|
||
|
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);
|
||
|
virtual ~SceneOpenGLShadow();
|
||
|
|
||
|
GLTexture *shadowTexture() {
|
||
|
return m_texture.data();
|
||
|
}
|
||
|
protected:
|
||
|
virtual void buildQuads();
|
||
|
virtual bool prepareBackend();
|
||
|
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);
|
||
|
virtual ~SceneOpenGLDecorationRenderer();
|
||
|
|
||
|
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
|