kwin/plugins/platforms/x11/windowed/scene_qpainter_x11_backend.cpp

105 lines
3 KiB
C++
Raw Normal View History

/********************************************************************
KWin - the KDE window manager
This file is part of the KDE project.
Copyright (C) 2015 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/>.
*********************************************************************/
#include "scene_qpainter_x11_backend.h"
#include "x11windowed_backend.h"
#include "screens.h"
namespace KWin
{
X11WindowedQPainterBackend::X11WindowedQPainterBackend(X11WindowedBackend *backend)
: QPainterBackend()
, m_backend(backend)
{
connect(screens(), &Screens::changed, this, &X11WindowedQPainterBackend::createOutputs);
createOutputs();
}
X11WindowedQPainterBackend::~X11WindowedQPainterBackend()
{
qDeleteAll(m_outputs);
if (m_gc) {
xcb_free_gc(m_backend->connection(), m_gc);
}
}
void X11WindowedQPainterBackend::createOutputs()
{
qDeleteAll(m_outputs);
m_outputs.clear();
for (int i = 0; i < screens()->count(); ++i) {
Output *output = new Output;
output->window = m_backend->windowForScreen(i);
output->buffer = QImage(screens()->size(i), QImage::Format_RGB32);
output->buffer.fill(Qt::black);
m_outputs << output;
}
m_needsFullRepaint = true;
}
QImage *X11WindowedQPainterBackend::buffer()
{
return bufferForScreen(0);
}
QImage *X11WindowedQPainterBackend::bufferForScreen(int screen)
{
return &m_outputs.at(screen)->buffer;
}
bool X11WindowedQPainterBackend::needsFullRepaint() const
{
return m_needsFullRepaint;
}
void X11WindowedQPainterBackend::prepareRenderingFrame()
{
}
void X11WindowedQPainterBackend::present(int mask, const QRegion &damage)
{
Q_UNUSED(mask)
Q_UNUSED(damage)
xcb_connection_t *c = m_backend->connection();
const xcb_window_t window = m_backend->window();
if (m_gc == XCB_NONE) {
m_gc = xcb_generate_id(c);
xcb_create_gc(c, m_gc, window, 0, nullptr);
}
for (auto it = m_outputs.constBegin(); it != m_outputs.constEnd(); ++it) {
// TODO: only update changes?
const QImage &buffer = (*it)->buffer;
xcb_put_image(c, XCB_IMAGE_FORMAT_Z_PIXMAP, (*it)->window, m_gc,
buffer.width(), buffer.height(), 0, 0, 0, 24,
buffer.byteCount(), buffer.constBits());
}
}
bool X11WindowedQPainterBackend::usesOverlayWindow() const
{
return false;
}
bool X11WindowedQPainterBackend::perScreenRendering() const
{
return true;
}
}