/******************************************************************** KWin - the KDE window manager This file is part of the KDE project. Copyright (C) 2015 Martin Gräßlin 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 . *********************************************************************/ #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) * screens()->scale(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.sizeInBytes(), buffer.constBits()); } } bool X11WindowedQPainterBackend::usesOverlayWindow() const { return false; } bool X11WindowedQPainterBackend::perScreenRendering() const { return true; } }