/**************************************************************************** ** ** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies). ** All rights reserved. ** Contact: Nokia Corporation (qt-info@nokia.com) ** ** This file is part of the QtGui module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** GNU Lesser General Public License Usage ** 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, Nokia gives you certain additional ** rights. These rights are described in the Nokia 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. ** ** Other Usage ** Alternatively, this file may be used in accordance with the terms and ** conditions contained in a signed written agreement between you and Nokia. ** ** ** ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include // for Q_WS_WIN define (non-PCH) #include #include #include #include #include #include #include #ifdef QT_GRAPHICSSYSTEM_RUNTIME #include #endif QT_BEGIN_NAMESPACE struct QS60WindowSurfacePrivate { QPixmap device; QList bufferImages; }; TDisplayMode displayMode(bool opaque) { TDisplayMode mode = S60->screenDevice()->DisplayMode(); if (opaque) { mode = EColor16MU; } else { if (QSysInfo::symbianVersion() >= QSysInfo::SV_SF_3) mode = Q_SYMBIAN_ECOLOR16MAP; // Symbian^3 WServ has support for ARGB32_PRE else mode = EColor16MA; // Symbian prior to Symbian^3 sw accelerates EColor16MA } return mode; } bool blitWriteAlpha(QWidgetPrivate *widgetPrivate) { QWExtra *extra = widgetPrivate->extraData(); return extra ? extra->nativePaintMode == QWExtra::BlitWriteAlpha : false; } QS60WindowSurface::QS60WindowSurface(QWidget* widget) : QWindowSurface(widget), d_ptr(new QS60WindowSurfacePrivate) { QWidgetPrivate *widgetPrivate = qt_widget_private(widget); const bool opaque = widgetPrivate->isOpaque && !blitWriteAlpha(widgetPrivate); TDisplayMode mode = displayMode(opaque); // We create empty CFbsBitmap here -> it will be resized in setGeometry CFbsBitmap *bitmap = new CFbsBitmap; // CBase derived object needs check on new Q_CHECK_PTR(bitmap); qt_symbian_throwIfError( bitmap->Create( TSize(0, 0), mode ) ); QSymbianRasterPixmapData *data = new QSymbianRasterPixmapData(QPixmapData::PixmapType); if (data) { data->fromSymbianBitmap(bitmap, true); d_ptr->device = QPixmap(data); } } QS60WindowSurface::~QS60WindowSurface() { #if defined(QT_GRAPHICSSYSTEM_RUNTIME) && defined(Q_SYMBIAN_SUPPORTS_SURFACES) if(QApplicationPrivate::runtime_graphics_system) { QRuntimeGraphicsSystem *runtimeGraphicsSystem = static_cast(QApplicationPrivate::graphics_system); if(runtimeGraphicsSystem->graphicsSystemName() == QLatin1String("openvg")) { // Graphics system has been switched from raster to openvg. // Issue empty redraw to clear the UI surface QWidget *w = window(); if (w->testAttribute(Qt::WA_WState_Created)) { RWindow *const window = static_cast(w->winId()->DrawableWindow()); window->BeginRedraw(); window->EndRedraw(); } } } #endif delete d_ptr; } void QS60WindowSurface::beginPaint(const QRegion &rgn) { #ifdef Q_SYMBIAN_SUPPORTS_SURFACES S60->wsSession().Finish(); #endif QWidgetPrivate *windowPrivate = qt_widget_private(window()); if (!windowPrivate->isOpaque || blitWriteAlpha(windowPrivate)) { QSymbianRasterPixmapData *pixmapData = static_cast(d_ptr->device.data_ptr().data()); TDisplayMode mode = displayMode(false); if (pixmapData->cfbsBitmap->DisplayMode() != mode) pixmapData->convertToDisplayMode(mode); pixmapData->beginDataAccess(); if (!windowPrivate->isOpaque) { QPainter p(&pixmapData->image); p.setCompositionMode(QPainter::CompositionMode_Source); const QVector rects = rgn.rects(); const QColor blank = Qt::transparent; for (QVector::const_iterator it = rects.begin(); it != rects.end(); ++it) { p.fillRect(*it, blank); } } pixmapData->endDataAccess(); } } void QS60WindowSurface::endPaint(const QRegion &) { qDeleteAll(d_ptr->bufferImages); d_ptr->bufferImages.clear(); } QImage* QS60WindowSurface::buffer(const QWidget *widget) { if (widget->window() != window()) return 0; QPaintDevice *pdev = paintDevice(); if (!pdev) return 0; const QPoint off = offset(widget); QImage *img = &(static_cast(d_ptr->device.data_ptr().data())->image); QRect rect(off, widget->size()); rect &= QRect(QPoint(), img->size()); if (rect.isEmpty()) return 0; img = new QImage(img->scanLine(rect.y()) + rect.x() * img->depth() / 8, rect.width(), rect.height(), img->bytesPerLine(), img->format()); d_ptr->bufferImages.append(img); return img; } void QS60WindowSurface::flush(QWidget *widget, const QRegion ®ion, const QPoint &) { QWidget *window = widget->window(); Q_ASSERT(window); QTLWExtra *topExtra = window->d_func()->maybeTopData(); Q_ASSERT(topExtra); QRect qr = region.boundingRect(); if (!topExtra->inExpose) { topExtra->inExpose = true; // Prevent DrawNow() from calling syncBackingStore() again TRect tr = qt_QRect2TRect(qr); widget->winId()->DrawNow(tr); topExtra->inExpose = false; } else { // This handles the case when syncBackingStore updates content outside of the // original drawing rectangle. This might happen if there are pending update() // events at the same time as we get a Draw() from Symbian. QRect drawRect = qt_TRect2QRect(widget->winId()->DrawableWindow()->GetDrawRect()); if (!drawRect.contains(qr)) widget->winId()->DrawDeferred(); } } bool QS60WindowSurface::scroll(const QRegion &area, int dx, int dy) { QRect rect = area.boundingRect(); if (dx == 0 && dy == 0) return false; if (d_ptr->device.isNull()) return false; QSymbianRasterPixmapData *data = static_cast(d_ptr->device.data_ptr().data()); data->scroll(dx, dy, rect); return true; } QPaintDevice* QS60WindowSurface::paintDevice() { return &d_ptr->device; } void QS60WindowSurface::setGeometry(const QRect& rect) { if (rect == geometry()) return; QSymbianRasterPixmapData *data = static_cast(d_ptr->device.data_ptr().data()); data->resize(rect.width(), rect.height()); QWindowSurface::setGeometry(rect); } QWindowSurface::WindowSurfaceFeatures QS60WindowSurface::features() const { return QWindowSurface::AllFeatures; } CFbsBitmap* QS60WindowSurface::symbianBitmap() const { QSymbianRasterPixmapData *data = static_cast(d_ptr->device.data_ptr().data()); return data->cfbsBitmap; } QT_END_NAMESPACE