qtimageformats/tests/auto/tiff/tst_qtiff.cpp

524 lines
20 KiB
C++

/****************************************************************************
**
** Copyright (C) 2012 Digia Plc and/or its subsidiary(-ies).
** Contact: http://www.qt-project.org/legal
**
** This file is part of the QtAddOn.ImageFormats 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$
**
****************************************************************************/
#include <QtTest/QtTest>
#include <QtGui/QtGui>
Q_DECLARE_METATYPE(QImage::Format)
Q_DECLARE_METATYPE(QImageWriter::ImageWriterError)
typedef QList<int> QIntList;
Q_DECLARE_METATYPE(QIntList)
class tst_qtiff: public QObject
{
Q_OBJECT
private slots:
void initTestCase();
void formatHandler();
void readImage_data();
void readImage();
void readCorruptImage_data();
void readCorruptImage();
void tiffCompression_data();
void tiffCompression();
void tiffEndianness();
void tiffOrientation_data();
void tiffOrientation();
void tiffGrayscale();
void dotsPerMeter_data();
void dotsPerMeter();
void physicalDpi_data();
void physicalDpi();
void writeImage_data();
void writeImage();
void readWriteNonDestructive_data();
void readWriteNonDestructive();
void largeTiff();
void supportsOption_data();
void supportsOption();
void resolution_data();
void resolution();
private:
QString prefix;
};
void tst_qtiff::initTestCase()
{
prefix = ":/tiff/";
}
void tst_qtiff::formatHandler()
{
QString testFormat = "TIFF";
QString testFile = prefix + "image.tif";
QList<QByteArray> formats = QImageReader::supportedImageFormats();
bool formatSupported = false;
for (QList<QByteArray>::Iterator it = formats.begin(); it != formats.end(); ++it) {
if (*it == testFormat.toLower()) {
formatSupported = true;
break;
}
}
QVERIFY(formatSupported);
QCOMPARE(QImageReader::imageFormat(testFile), testFormat.toLatin1().toLower());
}
void tst_qtiff::readImage_data()
{
QTest::addColumn<QString>("fileName");
QTest::addColumn<QSize>("size");
QTest::newRow("grayscale-ref") << QString("grayscale-ref.tif") << QSize(320, 200);
QTest::newRow("grayscale") << QString("grayscale.tif") << QSize(320, 200);
QTest::newRow("image_100dpi") << QString("image_100dpi.tif") << QSize(22, 22);
QTest::newRow("image") << QString("image.tif") << QSize(22, 22);
QTest::newRow("indexed_orientation_1") << QString("indexed_orientation_1.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_2") << QString("indexed_orientation_2.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_3") << QString("indexed_orientation_3.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_4") << QString("indexed_orientation_4.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_5") << QString("indexed_orientation_5.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_6") << QString("indexed_orientation_6.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_7") << QString("indexed_orientation_7.tiff") << QSize(64, 64);
QTest::newRow("indexed_orientation_8") << QString("indexed_orientation_8.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_1") << QString("mono_orientation_1.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_2") << QString("mono_orientation_2.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_3") << QString("mono_orientation_3.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_4") << QString("mono_orientation_4.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_5") << QString("mono_orientation_5.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_6") << QString("mono_orientation_6.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_7") << QString("mono_orientation_7.tiff") << QSize(64, 64);
QTest::newRow("mono_orientation_8") << QString("mono_orientation_8.tiff") << QSize(64, 64);
QTest::newRow("original_indexed") << QString("original_indexed.tiff") << QSize(64, 64);
QTest::newRow("original_mono") << QString("original_mono.tiff") << QSize(64, 64);
QTest::newRow("original_rgb") << QString("original_rgb.tiff") << QSize(64, 64);
QTest::newRow("rgba_adobedeflate_littleendian") << QString("rgba_adobedeflate_littleendian.tif") << QSize(200, 200);
QTest::newRow("rgba_lzw_littleendian") << QString("rgba_lzw_littleendian.tif") << QSize(200, 200);
QTest::newRow("rgba_nocompression_bigendian") << QString("rgba_nocompression_bigendian.tif") << QSize(200, 200);
QTest::newRow("rgba_nocompression_littleendian") << QString("rgba_nocompression_littleendian.tif") << QSize(200, 200);
QTest::newRow("rgba_packbits_littleendian") << QString("rgba_packbits_littleendian.tif") << QSize(200, 200);
QTest::newRow("rgba_zipdeflate_littleendian") << QString("rgba_zipdeflate_littleendian.tif") << QSize(200, 200);
QTest::newRow("rgb_orientation_1") << QString("rgb_orientation_1.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_2") << QString("rgb_orientation_2.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_3") << QString("rgb_orientation_3.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_4") << QString("rgb_orientation_4.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_5") << QString("rgb_orientation_5.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_6") << QString("rgb_orientation_6.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_7") << QString("rgb_orientation_7.tiff") << QSize(64, 64);
QTest::newRow("rgb_orientation_8") << QString("rgb_orientation_8.tiff") << QSize(64, 64);
QTest::newRow("teapot") << QString("teapot.tiff") << QSize(256, 256);
}
void tst_qtiff::readImage()
{
QFETCH(QString, fileName);
QFETCH(QSize, size);
QString path = prefix + fileName;
QImageReader reader(path);
QVERIFY(reader.canRead());
QImage image = reader.read();
QVERIFY(!image.isNull());
QCOMPARE(image.size(), size);
}
void tst_qtiff::readCorruptImage_data()
{
QTest::addColumn<QString>("fileName");
QTest::addColumn<QString>("message");
QTest::newRow("corrupt tiff") << QString("corrupt-data.tif") << QString();
}
void tst_qtiff::readCorruptImage()
{
QFETCH(QString, fileName);
QFETCH(QString, message);
QString path = prefix + fileName;
QImageReader reader(path);
if (!message.isEmpty())
QTest::ignoreMessage(QtWarningMsg, message.toLatin1());
QVERIFY(reader.canRead());
QImage image = reader.read();
QVERIFY(image.isNull());
}
void tst_qtiff::tiffCompression_data()
{
QTest::addColumn<QString>("uncompressedFile");
QTest::addColumn<QString>("compressedFile");
QTest::newRow("TIFF: adobedeflate") << "rgba_nocompression_littleendian.tif"
<< "rgba_adobedeflate_littleendian.tif";
QTest::newRow("TIFF: lzw") << "rgba_nocompression_littleendian.tif"
<< "rgba_lzw_littleendian.tif";
QTest::newRow("TIFF: packbits") << "rgba_nocompression_littleendian.tif"
<< "rgba_packbits_littleendian.tif";
QTest::newRow("TIFF: zipdeflate") << "rgba_nocompression_littleendian.tif"
<< "rgba_zipdeflate_littleendian.tif";
}
void tst_qtiff::tiffCompression()
{
QFETCH(QString, uncompressedFile);
QFETCH(QString, compressedFile);
QImage uncompressedImage(prefix + uncompressedFile);
QImage compressedImage(prefix + compressedFile);
QCOMPARE(uncompressedImage, compressedImage);
}
void tst_qtiff::tiffEndianness()
{
QImage littleEndian(prefix + "rgba_nocompression_littleendian.tif");
QImage bigEndian(prefix + "rgba_nocompression_bigendian.tif");
QCOMPARE(littleEndian, bigEndian);
}
void tst_qtiff::tiffOrientation_data()
{
QTest::addColumn<QString>("expected");
QTest::addColumn<QString>("oriented");
QTest::newRow("Indexed TIFF, orientation1") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_1.tiff";
QTest::newRow("Indexed TIFF, orientation2") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_2.tiff";
QTest::newRow("Indexed TIFF, orientation3") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_3.tiff";
QTest::newRow("Indexed TIFF, orientation4") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_4.tiff";
QTest::newRow("Indexed TIFF, orientation5") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_5.tiff";
QTest::newRow("Indexed TIFF, orientation6") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_6.tiff";
QTest::newRow("Indexed TIFF, orientation7") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_7.tiff";
QTest::newRow("Indexed TIFF, orientation8") << "tiff_oriented/original_indexed.tiff" << "tiff_oriented/indexed_orientation_8.tiff";
QTest::newRow("Mono TIFF, orientation1") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_1.tiff";
QTest::newRow("Mono TIFF, orientation2") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_2.tiff";
QTest::newRow("Mono TIFF, orientation3") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_3.tiff";
QTest::newRow("Mono TIFF, orientation4") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_4.tiff";
QTest::newRow("Mono TIFF, orientation5") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_5.tiff";
QTest::newRow("Mono TIFF, orientation6") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_6.tiff";
QTest::newRow("Mono TIFF, orientation7") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_7.tiff";
QTest::newRow("Mono TIFF, orientation8") << "tiff_oriented/original_mono.tiff" << "tiff_oriented/mono_orientation_8.tiff";
QTest::newRow("RGB TIFF, orientation1") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_1.tiff";
QTest::newRow("RGB TIFF, orientation2") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_2.tiff";
QTest::newRow("RGB TIFF, orientation3") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_3.tiff";
QTest::newRow("RGB TIFF, orientation4") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_4.tiff";
QTest::newRow("RGB TIFF, orientation5") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_5.tiff";
QTest::newRow("RGB TIFF, orientation6") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_6.tiff";
QTest::newRow("RGB TIFF, orientation7") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_7.tiff";
QTest::newRow("RGB TIFF, orientation8") << "tiff_oriented/original_rgb.tiff" << "tiff_oriented/rgb_orientation_8.tiff";
}
void tst_qtiff::tiffOrientation()
{
QFETCH(QString, expected);
QFETCH(QString, oriented);
QImage expectedImage(prefix + expected);
QImage orientedImage(prefix + oriented);
QCOMPARE(expectedImage, orientedImage);
}
void tst_qtiff::tiffGrayscale()
{
QImage actualImage(prefix + "grayscale.tif");
QImage expectedImage(prefix + "grayscale-ref.tif");
QCOMPARE(expectedImage, actualImage.convertToFormat(expectedImage.format()));
}
void tst_qtiff::dotsPerMeter_data()
{
QTest::addColumn<QString>("fileName");
QTest::addColumn<int>("expectedDotsPerMeterX");
QTest::addColumn<int>("expectedDotsPerMeterY");
QTest::newRow("TIFF: 72 dpi") << ("rgba_nocompression_littleendian.tif") << qRound(72 * (100 / 2.54)) << qRound(72 * (100 / 2.54));
QTest::newRow("TIFF: 100 dpi") << ("image_100dpi.tif") << qRound(100 * (100 / 2.54)) << qRound(100 * (100 / 2.54));
}
void tst_qtiff::dotsPerMeter()
{
QFETCH(QString, fileName);
QFETCH(int, expectedDotsPerMeterX);
QFETCH(int, expectedDotsPerMeterY);
QImage image(prefix + fileName);
QCOMPARE(image.dotsPerMeterX(), expectedDotsPerMeterX);
QCOMPARE(image.dotsPerMeterY(), expectedDotsPerMeterY);
}
void tst_qtiff::physicalDpi_data()
{
QTest::addColumn<QString>("fileName");
QTest::addColumn<int>("expectedPhysicalDpiX");
QTest::addColumn<int>("expectedPhysicalDpiY");
QTest::newRow("TIFF: 72 dpi") << "rgba_nocompression_littleendian.tif" << 72 << 72;
QTest::newRow("TIFF: 100 dpi") << "image_100dpi.tif" << 100 << 100;
}
void tst_qtiff::physicalDpi()
{
QFETCH(QString, fileName);
QFETCH(int, expectedPhysicalDpiX);
QFETCH(int, expectedPhysicalDpiY);
QImage image(prefix + fileName);
QCOMPARE(image.physicalDpiX(), expectedPhysicalDpiX);
QCOMPARE(image.physicalDpiY(), expectedPhysicalDpiY);
}
void tst_qtiff::writeImage_data()
{
QTest::addColumn<QString>("fileName");
QTest::addColumn<bool>("lossy");
QTest::addColumn<QByteArray>("format");
QTest::newRow("TIFF: teapot") << QString("teapot.tiff") << false << QByteArray("tiff");
}
void tst_qtiff::writeImage()
{
QFETCH(QString, fileName);
QFETCH(bool, lossy);
QFETCH(QByteArray, format);
QImage image;
{
QImageReader reader(prefix + fileName);
image = reader.read();
QVERIFY2(!image.isNull(), qPrintable(reader.errorString()));
}
QByteArray output;
{
QBuffer buf(&output);
QVERIFY(buf.open(QIODevice::WriteOnly));
QImageWriter writer(&buf, format);
QVERIFY(writer.write(image));
}
QImage image2;
{
QBuffer buf(&output);
QVERIFY(buf.open(QIODevice::ReadOnly));
QImageReader reader(&buf);
image2 = reader.read();
QVERIFY2(!image.isNull(), qPrintable(reader.errorString()));
}
if (!lossy) {
QCOMPARE(image, image2);
} else {
QCOMPARE(image.format(), image2.format());
QCOMPARE(image.depth(), image2.depth());
}
}
void tst_qtiff::readWriteNonDestructive_data()
{
QTest::addColumn<QImage::Format>("format");
QTest::addColumn<QImage::Format>("expectedFormat");
QTest::addColumn<bool>("grayscale");
QTest::newRow("tiff mono") << QImage::Format_Mono << QImage::Format_Mono << false;
QTest::newRow("tiff indexed") << QImage::Format_Indexed8 << QImage::Format_Indexed8 << false;
QTest::newRow("tiff rgb32") << QImage::Format_ARGB32 << QImage::Format_ARGB32 << false;
QTest::newRow("tiff grayscale") << QImage::Format_Indexed8 << QImage::Format_Indexed8 << true;
}
void tst_qtiff::readWriteNonDestructive()
{
QFETCH(QImage::Format, format);
QFETCH(QImage::Format, expectedFormat);
QFETCH(bool, grayscale);
QImage image = QImage(prefix + "colorful.bmp").convertToFormat(format);
QVERIFY(!image.isNull());
if (grayscale) {
QVector<QRgb> colors;
for (int i = 0; i < 256; ++i)
colors << qRgb(i, i, i);
image.setColorTable(colors);
}
QByteArray output;
QBuffer buf(&output);
QVERIFY(buf.open(QIODevice::WriteOnly));
QVERIFY(image.save(&buf, "tiff"));
buf.close();
QVERIFY(buf.open(QIODevice::ReadOnly));
QImageReader reader(&buf);
QImage image2 = reader.read();
QVERIFY2(!image.isNull(), qPrintable(reader.errorString()));
QImage::Format readFormat = image2.format();
QCOMPARE(readFormat, expectedFormat);
QCOMPARE(image, image2);
}
void tst_qtiff::largeTiff()
{
#if defined(Q_OS_WINCE)
QSKIP("not tested on WinCE");
#endif
QImage img(4096, 2048, QImage::Format_ARGB32);
QPainter p(&img);
img.fill(0x0);
p.fillRect(0, 0, 4096, 2048, QBrush(Qt::CrossPattern));
p.end();
QByteArray array;
QBuffer writeBuffer(&array);
writeBuffer.open(QIODevice::WriteOnly);
QImageWriter writer(&writeBuffer, "tiff");
QVERIFY(writer.write(img));
writeBuffer.close();
QBuffer readBuffer(&array);
readBuffer.open(QIODevice::ReadOnly);
QImageReader reader(&readBuffer, "tiff");
QImage img2 = reader.read();
QVERIFY(!img2.isNull());
QCOMPARE(img, img2);
}
void tst_qtiff::supportsOption_data()
{
QTest::addColumn<QIntList>("options");
QTest::newRow("tiff") << (QIntList() << QImageIOHandler::Size
<< QImageIOHandler::CompressionRatio);
}
void tst_qtiff::supportsOption()
{
QFETCH(QIntList, options);
QSet<QImageIOHandler::ImageOption> allOptions;
allOptions << QImageIOHandler::Size
<< QImageIOHandler::ClipRect
<< QImageIOHandler::Description
<< QImageIOHandler::ScaledClipRect
<< QImageIOHandler::ScaledSize
<< QImageIOHandler::CompressionRatio
<< QImageIOHandler::Gamma
<< QImageIOHandler::Quality
<< QImageIOHandler::Name
<< QImageIOHandler::SubType
<< QImageIOHandler::IncrementalReading
<< QImageIOHandler::Endianness
<< QImageIOHandler::Animation
<< QImageIOHandler::BackgroundColor;
QImageWriter writer;
writer.setFormat("tiff");
for (int i = 0; i < options.size(); ++i) {
QVERIFY(writer.supportsOption(QImageIOHandler::ImageOption(options.at(i))));
allOptions.remove(QImageIOHandler::ImageOption(options.at(i)));
}
foreach (QImageIOHandler::ImageOption option, allOptions)
QVERIFY(!writer.supportsOption(option));
}
void tst_qtiff::resolution_data()
{
QTest::addColumn<QString>("filename");
QTest::addColumn<int>("expectedDotsPerMeterX");
QTest::addColumn<int>("expectedDotsPerMeterY");
QTest::newRow("TIFF: 100 dpi") << ("image_100dpi.tif") << qRound(100 * (100 / 2.54)) << qRound(100 * (100 / 2.54));
QTest::newRow("TIFF: 50 dpi") << ("image_50dpi.tif") << qRound(50 * (100 / 2.54)) << qRound(50 * (100 / 2.54));
QTest::newRow("TIFF: 300 dot per meter") << ("image_300dpm.tif") << 300 << 300;
}
void tst_qtiff::resolution()
{
QFETCH(QString, filename);
QFETCH(int, expectedDotsPerMeterX);
QFETCH(int, expectedDotsPerMeterY);
QImage image(prefix + QLatin1String("colorful.bmp"));
image.setDotsPerMeterX(expectedDotsPerMeterX);
image.setDotsPerMeterY(expectedDotsPerMeterY);
QByteArray output;
{
QBuffer buf(&output);
QVERIFY(buf.open(QIODevice::WriteOnly));
QImageWriter writer(&buf, "tiff");
QVERIFY(writer.write(image));
}
QBuffer buf(&output);
QVERIFY(buf.open(QIODevice::ReadOnly));
QImageReader reader(&buf);
const QImage generatedImage = reader.read();
QVERIFY(!generatedImage.isNull());
QCOMPARE(expectedDotsPerMeterX, generatedImage.dotsPerMeterX());
QCOMPARE(expectedDotsPerMeterY, generatedImage.dotsPerMeterY());
}
QTEST_MAIN(tst_qtiff)
#include "tst_qtiff.moc"