2022-02-05 09:47:00 +00:00
|
|
|
/*
|
|
|
|
SPDX-FileCopyrightText: 2022 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
|
2022-04-12 13:14:23 +00:00
|
|
|
#include "cursordelegate_opengl.h"
|
2022-08-29 07:55:49 +00:00
|
|
|
#include "core/renderlayer.h"
|
|
|
|
#include "core/rendertarget.h"
|
2022-02-05 09:47:00 +00:00
|
|
|
#include "cursor.h"
|
|
|
|
#include "kwingltexture.h"
|
|
|
|
#include "kwinglutils.h"
|
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2022-04-12 13:14:23 +00:00
|
|
|
CursorDelegateOpenGL::~CursorDelegateOpenGL()
|
2022-02-05 09:47:00 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-04-12 13:14:23 +00:00
|
|
|
void CursorDelegateOpenGL::paint(RenderTarget *renderTarget, const QRegion ®ion)
|
2022-02-05 09:47:00 +00:00
|
|
|
{
|
2022-04-12 13:14:23 +00:00
|
|
|
if (!region.intersects(layer()->mapToGlobal(layer()->rect()))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-05 09:47:00 +00:00
|
|
|
auto allocateTexture = [this]() {
|
|
|
|
const QImage img = Cursors::self()->currentCursor()->image();
|
|
|
|
if (img.isNull()) {
|
|
|
|
m_cursorTextureDirty = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_cursorTexture.reset(new GLTexture(img));
|
|
|
|
m_cursorTexture->setWrapMode(GL_CLAMP_TO_EDGE);
|
|
|
|
m_cursorTextureDirty = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
// lazy init texture cursor only in case we need software rendering
|
|
|
|
if (!m_cursorTexture) {
|
|
|
|
allocateTexture();
|
|
|
|
|
|
|
|
// handle shape update on case cursor image changed
|
|
|
|
connect(Cursors::self(), &Cursors::currentCursorChanged, this, [this]() {
|
|
|
|
m_cursorTextureDirty = true;
|
|
|
|
});
|
|
|
|
} else if (m_cursorTextureDirty) {
|
|
|
|
const QImage image = Cursors::self()->currentCursor()->image();
|
|
|
|
if (image.size() == m_cursorTexture->size()) {
|
|
|
|
m_cursorTexture->update(image);
|
|
|
|
m_cursorTextureDirty = false;
|
|
|
|
} else {
|
|
|
|
allocateTexture();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-12 13:14:23 +00:00
|
|
|
const QRect cursorRect = layer()->mapToGlobal(layer()->rect());
|
2022-08-18 10:16:53 +00:00
|
|
|
const qreal scale = renderTarget->devicePixelRatio();
|
2022-02-05 09:47:00 +00:00
|
|
|
|
|
|
|
QMatrix4x4 mvp;
|
2022-08-18 10:16:53 +00:00
|
|
|
mvp.ortho(QRect(QPoint(0, 0), renderTarget->size()));
|
|
|
|
mvp.translate(cursorRect.x() * scale, cursorRect.y() * scale);
|
2022-02-05 09:47:00 +00:00
|
|
|
|
2022-12-20 15:48:05 +00:00
|
|
|
GLFramebuffer *fbo = std::get<GLFramebuffer *>(renderTarget->nativeHandle());
|
|
|
|
GLFramebuffer::pushFramebuffer(fbo);
|
|
|
|
|
2022-02-05 09:47:00 +00:00
|
|
|
// Don't need to call GLVertexBuffer::beginFrame() and GLVertexBuffer::endOfFrame() because
|
|
|
|
// the GLVertexBuffer::streamingBuffer() is not being used when painting cursor.
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
|
|
|
|
m_cursorTexture->bind();
|
|
|
|
ShaderBinder binder(ShaderTrait::MapTexture);
|
|
|
|
binder.shader()->setUniform(GLShader::ModelViewProjectionMatrix, mvp);
|
2022-08-18 10:16:53 +00:00
|
|
|
m_cursorTexture->render(region, QRect(0, 0, cursorRect.width(), cursorRect.height()), scale);
|
2022-02-05 09:47:00 +00:00
|
|
|
m_cursorTexture->unbind();
|
|
|
|
glDisable(GL_BLEND);
|
2022-12-20 15:48:05 +00:00
|
|
|
|
|
|
|
GLFramebuffer::popFramebuffer();
|
2022-02-05 09:47:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace KWin
|