/**************************************************************************** ** ** Copyright (C) 2012 Nokia Corporation and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/ ** ** 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 "qpixmap.h" #include #include "qpixmap_raster_p.h" #include "qnativeimage_p.h" #include "qimage_p.h" #include "qpaintengine.h" #include "qbitmap.h" #include "qimage.h" #include #include #include #include #include QT_BEGIN_NAMESPACE QPixmap qt_toRasterPixmap(const QImage &image) { QPlatformPixmap *data = new QRasterPlatformPixmap(image.depth() == 1 ? QPlatformPixmap::BitmapType : QPlatformPixmap::PixmapType); data->fromImage(image, Qt::AutoColor); return QPixmap(data); } QPixmap qt_toRasterPixmap(const QPixmap &pixmap) { if (pixmap.isNull()) return QPixmap(); if (QPixmap(pixmap).data_ptr()->classId() == QPlatformPixmap::RasterClass) return pixmap; return qt_toRasterPixmap(pixmap.toImage()); } QRasterPlatformPixmap::QRasterPlatformPixmap(PixelType type) : QPlatformPixmap(type, RasterClass) { } QRasterPlatformPixmap::~QRasterPlatformPixmap() { } QPlatformPixmap *QRasterPlatformPixmap::createCompatiblePlatformPixmap() const { return new QRasterPlatformPixmap(pixelType()); } void QRasterPlatformPixmap::resize(int width, int height) { QImage::Format format; if (pixelType() == BitmapType) format = QImage::Format_MonoLSB; else format = QNativeImage::systemFormat(); image = QImage(width, height, format); w = width; h = height; d = image.depth(); is_null = (w <= 0 || h <= 0); if (pixelType() == BitmapType && !image.isNull()) { image.setColorCount(2); image.setColor(0, QColor(Qt::color0).rgba()); image.setColor(1, QColor(Qt::color1).rgba()); } setSerialNumber(image.cacheKey() >> 32); } bool QRasterPlatformPixmap::fromData(const uchar *buffer, uint len, const char *format, Qt::ImageConversionFlags flags) { QByteArray a = QByteArray::fromRawData(reinterpret_cast(buffer), len); QBuffer b(&a); b.open(QIODevice::ReadOnly); QImage image = QImageReader(&b, format).read(); if (image.isNull()) return false; createPixmapForImage(image, flags, /* inplace = */true); return !isNull(); } void QRasterPlatformPixmap::fromImage(const QImage &sourceImage, Qt::ImageConversionFlags flags) { Q_UNUSED(flags); QImage image = sourceImage; createPixmapForImage(image, flags, /* inplace = */false); } void QRasterPlatformPixmap::fromImageReader(QImageReader *imageReader, Qt::ImageConversionFlags flags) { Q_UNUSED(flags); QImage image = imageReader->read(); if (image.isNull()) return; createPixmapForImage(image, flags, /* inplace = */true); } // from qbackingstore.cpp extern void qt_scrollRectInImage(QImage &img, const QRect &rect, const QPoint &offset); void QRasterPlatformPixmap::copy(const QPlatformPixmap *data, const QRect &rect) { fromImage(data->toImage(rect).copy(), Qt::NoOpaqueDetection); } bool QRasterPlatformPixmap::scroll(int dx, int dy, const QRect &rect) { if (!image.isNull()) qt_scrollRectInImage(image, rect, QPoint(dx, dy)); return true; } void QRasterPlatformPixmap::fill(const QColor &color) { uint pixel; if (image.depth() == 1) { int gray = qGray(color.rgba()); // Pick the best approximate color in the image's colortable. if (qAbs(qGray(image.color(0)) - gray) < qAbs(qGray(image.color(1)) - gray)) pixel = 0; else pixel = 1; } else if (image.depth() >= 15) { int alpha = color.alpha(); if (alpha != 255) { if (!image.hasAlphaChannel()) { QImage::Format toFormat; #if !(defined(QT_COMPILER_SUPPORTS_NEON) || defined(__SSE2__)) if (image.format() == QImage::Format_RGB16) toFormat = QImage::Format_ARGB8565_Premultiplied; else if (image.format() == QImage::Format_RGB666) toFormat = QImage::Format_ARGB6666_Premultiplied; else if (image.format() == QImage::Format_RGB555) toFormat = QImage::Format_ARGB8555_Premultiplied; else if (image.format() == QImage::Format_RGB444) toFormat = QImage::Format_ARGB4444_Premultiplied; else #endif toFormat = QImage::Format_ARGB32_Premultiplied; if (!image.isNull() && qt_depthForFormat(image.format()) == qt_depthForFormat(toFormat)) { image.detach(); image.d->format = toFormat; } else { image = QImage(image.width(), image.height(), toFormat); } } } pixel = PREMUL(color.rgba()); const QPixelLayout *layout = &qPixelLayouts[image.format()]; layout->convertFromARGB32PM(&pixel, &pixel, 1, layout, 0); } else { pixel = 0; // ### what about 8 bits } image.fill(pixel); } bool QRasterPlatformPixmap::hasAlphaChannel() const { return image.hasAlphaChannel(); } QImage QRasterPlatformPixmap::toImage() const { if (!image.isNull()) { QImageData *data = const_cast(image).data_ptr(); if (data->paintEngine && data->paintEngine->isActive() && data->paintEngine->paintDevice() == &image) { return image.copy(); } } return image; } QImage QRasterPlatformPixmap::toImage(const QRect &rect) const { if (rect.isNull()) return image; QRect clipped = rect.intersected(QRect(0, 0, w, h)); if (d % 8 == 0) return QImage(image.scanLine(clipped.y()) + clipped.x() * (d / 8), clipped.width(), clipped.height(), image.bytesPerLine(), image.format()); else return image.copy(clipped); } QPaintEngine* QRasterPlatformPixmap::paintEngine() const { return image.paintEngine(); } int QRasterPlatformPixmap::metric(QPaintDevice::PaintDeviceMetric metric) const { QImageData *d = image.d; if (!d) return 0; // override the image dpi with the screen dpi when rendering to a pixmap switch (metric) { case QPaintDevice::PdmWidth: return w; case QPaintDevice::PdmHeight: return h; case QPaintDevice::PdmWidthMM: return qRound(d->width * 25.4 / qt_defaultDpiX()); case QPaintDevice::PdmHeightMM: return qRound(d->height * 25.4 / qt_defaultDpiY()); case QPaintDevice::PdmNumColors: return d->colortable.size(); case QPaintDevice::PdmDepth: return this->d; case QPaintDevice::PdmDpiX: // fall-through case QPaintDevice::PdmPhysicalDpiX: return qt_defaultDpiX(); case QPaintDevice::PdmDpiY: // fall-through case QPaintDevice::PdmPhysicalDpiY: return qt_defaultDpiY(); default: qWarning("QRasterPlatformPixmap::metric(): Unhandled metric type %d", metric); break; } return 0; } void QRasterPlatformPixmap::createPixmapForImage(QImage &sourceImage, Qt::ImageConversionFlags flags, bool inPlace) { QImage::Format format; if (flags & Qt::NoFormatConversion) format = sourceImage.format(); else if (pixelType() == BitmapType) { format = QImage::Format_MonoLSB; } else { if (sourceImage.depth() == 1) { format = sourceImage.hasAlphaChannel() ? QImage::Format_ARGB32_Premultiplied : QImage::Format_RGB32; } else { QImage::Format opaqueFormat = QNativeImage::systemFormat(); QImage::Format alphaFormat = QImage::Format_ARGB32_Premultiplied; #if !defined(QT_COMPILER_SUPPORTS_NEON) && !defined(__SSE2__) switch (opaqueFormat) { case QImage::Format_RGB16: alphaFormat = QImage::Format_ARGB8565_Premultiplied; break; default: // We don't care about the others... break; } #endif if (!sourceImage.hasAlphaChannel()) { format = opaqueFormat; } else if ((flags & Qt::NoOpaqueDetection) == 0 && !const_cast(sourceImage).data_ptr()->checkForAlphaPixels()) { format = opaqueFormat; } else { format = alphaFormat; } } } // image has alpha format but is really opaque, so try to do a // more efficient conversion if (format == QImage::Format_RGB32 && (sourceImage.format() == QImage::Format_ARGB32 || sourceImage.format() == QImage::Format_ARGB32_Premultiplied)) { image = sourceImage; if (!inPlace) image.detach(); if (image.d) image.d->format = QImage::Format_RGB32; } else if (inPlace && sourceImage.d->convertInPlace(format, flags)) { image = sourceImage; } else { image = sourceImage.convertToFormat(format); } if (image.d) { w = image.d->width; h = image.d->height; d = image.d->depth; } else { w = h = d = 0; } is_null = (w <= 0 || h <= 0); setSerialNumber(image.cacheKey() >> 32); } QImage* QRasterPlatformPixmap::buffer() { return ℑ } QT_END_NAMESPACE