/**************************************************************************** ** ** Copyright (C) 2011 - 2012 Research In Motion ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtCore module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3.0 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU General Public License version 3.0 requirements will be ** met: http://www.gnu.org/copyleft/gpl.html. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ // #define QBBRASTERWINDOWSURFACE_DEBUG #include "qbbrasterwindowsurface.h" #include "qbbwindow.h" #include #include #include QT_BEGIN_NAMESPACE QBBRasterWindowSurface::QBBRasterWindowSurface(QWidget *window) : QWindowSurface(window), mUseFrontBuffer(false) { #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::QBBRasterWindowSurface - w=" << window; #endif // save platform window associated with widget mPlatformWindow = static_cast(window->platformWindow()); } QBBRasterWindowSurface::~QBBRasterWindowSurface() { #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::~QBBRasterWindowSurface - w=" << window(); #endif } QPaintDevice *QBBRasterWindowSurface::paintDevice() { if (mPlatformWindow->hasBuffers()) return (mUseFrontBuffer) ? mPlatformWindow->frontBuffer().image() : mPlatformWindow->renderBuffer().image(); return 0; } void QBBRasterWindowSurface::flush(QWidget *widget, const QRegion ®ion, const QPoint &offset) { Q_UNUSED(widget); Q_UNUSED(offset); #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::flush - w=" << window(); #endif // visit all pending scroll operations for (int i = mScrollOpList.size() - 1; i >= 0; i--) { // do the scroll operation ScrollOp& op = mScrollOpList[i]; QRegion srcArea = op.totalArea.intersected( op.totalArea.translated(-op.dx, -op.dy) ); mPlatformWindow->scroll(srcArea, op.dx, op.dy); } // clear all pending scroll operations mScrollOpList.clear(); // update the display with newly rendered content mPlatformWindow->post(region); } void QBBRasterWindowSurface::resize(const QSize &size) { #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::resize - w=" << window() << ", s=" << size; #endif // call parent method QWindowSurface::resize(size); // NOTE: defer resizing window buffers until next paint as // resize() can be called multiple times before a paint occurs } bool QBBRasterWindowSurface::scroll(const QRegion &area, int dx, int dy) { #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::scroll - w=" << window(); #endif // calculate entire region affected by scroll operation (src + dst) QRegion totalArea = area.translated(dx, dy); totalArea += area; // visit all pending scroll operations for (int i = mScrollOpList.size() - 1; i >= 0; i--) { ScrollOp& op = mScrollOpList[i]; if (op.totalArea == totalArea) { // same area is being scrolled again - update delta op.dx += dx; op.dy += dy; return true; } else if (op.totalArea.intersects(totalArea)) { // current scroll overlaps previous scroll but is // not equal in area - just paint everything qWarning("QBB: pending scroll operations overlap but not equal"); return false; } } // create new scroll operation mScrollOpList.append( ScrollOp(totalArea, dx, dy) ); return true; } void QBBRasterWindowSurface::beginPaint(const QRegion ®ion) { Q_UNUSED(region); #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::beginPaint - w=" << window(); #endif // resize window buffers if surface resized QSize s = size(); if (s != mPlatformWindow->bufferSize()) { mPlatformWindow->setBufferSize(s); } } void QBBRasterWindowSurface::endPaint(const QRegion ®ion) { Q_UNUSED(region); #if defined(QBBRASTERWINDOWSURFACE_DEBUG) qDebug() << "QBBRasterWindowSurface::endPaint - w=" << window(); #endif } QPixmap QBBRasterWindowSurface::grabWidget(const QWidget *widget, const QRect &rectangle) const { // mUseFrontBuffer is used as a workaround to tell // QBBRasterWindowSurface::paintDevice() to return the // front buffer instead of the renderBuffer. grabWidget() // is the only use case. mUseFrontBuffer = true; QPixmap pixmap = QWindowSurface::grabWidget(widget, rectangle); mUseFrontBuffer = false; return pixmap; } QT_END_NAMESPACE