mirror of
https://invent.kde.org/frameworks/kimageformats.git
synced 2025-06-03 17:08:08 -04:00
Add write tests for heif/avif/jxl
Unfortunately none of them pass since it seems they can't load a png, save it to their format with loseless quality and read it back and get exactly the same contents than the png
This commit is contained in:
parent
f04084e175
commit
4afafee6c1
@ -30,6 +30,12 @@ macro(kimageformats_read_tests)
|
|||||||
endmacro()
|
endmacro()
|
||||||
|
|
||||||
macro(kimageformats_write_tests)
|
macro(kimageformats_write_tests)
|
||||||
|
cmake_parse_arguments(KIF_RT "" "FUZZ" "" ${ARGN})
|
||||||
|
set(_fuzzarg)
|
||||||
|
if (KIF_RT_FUZZ)
|
||||||
|
set(_fuzzarg -f ${KIF_RT_FUZZ})
|
||||||
|
endif()
|
||||||
|
|
||||||
if (NOT TARGET writetest)
|
if (NOT TARGET writetest)
|
||||||
add_executable(writetest writetest.cpp)
|
add_executable(writetest writetest.cpp)
|
||||||
target_link_libraries(writetest Qt${QT_MAJOR_VERSION}::Gui)
|
target_link_libraries(writetest Qt${QT_MAJOR_VERSION}::Gui)
|
||||||
@ -37,16 +43,22 @@ macro(kimageformats_write_tests)
|
|||||||
PRIVATE IMAGEDIR="${CMAKE_CURRENT_SOURCE_DIR}/write")
|
PRIVATE IMAGEDIR="${CMAKE_CURRENT_SOURCE_DIR}/write")
|
||||||
ecm_mark_as_test(writetest)
|
ecm_mark_as_test(writetest)
|
||||||
endif()
|
endif()
|
||||||
foreach(_testname ${ARGN})
|
foreach(_testname ${KIF_RT_UNPARSED_ARGUMENTS})
|
||||||
string(REGEX MATCH "-lossless$" _is_lossless "${_testname}")
|
string(REGEX MATCH "-lossless$" _is_lossless "${_testname}")
|
||||||
|
string(REGEX MATCH "-nodatacheck" _is_no_data_check "${_testname}")
|
||||||
unset(lossless_arg)
|
unset(lossless_arg)
|
||||||
|
unset(no_data_check_arg)
|
||||||
if (_is_lossless)
|
if (_is_lossless)
|
||||||
set(lossless_arg "--lossless")
|
set(lossless_arg "--lossless")
|
||||||
string(REGEX REPLACE "-lossless$" "" _testname "${_testname}")
|
string(REGEX REPLACE "-lossless$" "" _testname "${_testname}")
|
||||||
endif()
|
endif()
|
||||||
|
if (_is_no_data_check)
|
||||||
|
set(no_data_check_arg "--no-data-check")
|
||||||
|
string(REGEX REPLACE "-nodatacheck$" "" _testname "${_testname}")
|
||||||
|
endif()
|
||||||
add_test(
|
add_test(
|
||||||
NAME kimageformats-write-${_testname}
|
NAME kimageformats-write-${_testname}
|
||||||
COMMAND writetest ${lossless_arg} ${_testname}
|
COMMAND writetest ${lossless_arg} ${no_data_check_arg} ${_fuzzarg} ${_testname}
|
||||||
)
|
)
|
||||||
endforeach(_testname)
|
endforeach(_testname)
|
||||||
endmacro()
|
endmacro()
|
||||||
@ -74,18 +86,29 @@ if (TARGET avif)
|
|||||||
kimageformats_read_tests(
|
kimageformats_read_tests(
|
||||||
avif
|
avif
|
||||||
)
|
)
|
||||||
|
# because the plug-ins use RGB->YUV conversion which sometimes results in 1 value difference.
|
||||||
|
kimageformats_write_tests(FUZZ 1
|
||||||
|
avif-nodatacheck-lossless
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (LibHeif_FOUND)
|
if (LibHeif_FOUND)
|
||||||
kimageformats_read_tests(
|
kimageformats_read_tests(
|
||||||
heif
|
heif
|
||||||
)
|
)
|
||||||
|
# because the plug-ins use RGB->YUV conversion which sometimes results in 1 value difference.
|
||||||
|
kimageformats_write_tests(FUZZ 1
|
||||||
|
heif-nodatacheck-lossless
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (LibJXL_FOUND AND LibJXLThreads_FOUND)
|
if (LibJXL_FOUND AND LibJXLThreads_FOUND)
|
||||||
kimageformats_read_tests(
|
kimageformats_read_tests(
|
||||||
jxl
|
jxl
|
||||||
)
|
)
|
||||||
|
kimageformats_write_tests(
|
||||||
|
jxl-nodatacheck-lossless
|
||||||
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Allow some fuzziness when reading this formats, to allow for
|
# Allow some fuzziness when reading this formats, to allow for
|
||||||
|
37
autotests/fuzzyeq.cpp
Normal file
37
autotests/fuzzyeq.cpp
Normal file
@ -0,0 +1,37 @@
|
|||||||
|
/*
|
||||||
|
SPDX-FileCopyrightText: 2014 Alex Merry <alex.merry@kdemail.net>
|
||||||
|
|
||||||
|
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
|
||||||
|
*/
|
||||||
|
|
||||||
|
template<class Trait>
|
||||||
|
static bool fuzzyeq(const QImage &im1, const QImage &im2, uchar fuzziness)
|
||||||
|
{
|
||||||
|
Q_ASSERT(im1.format() == im2.format());
|
||||||
|
Q_ASSERT(im1.depth() == 24 || im1.depth() == 32 || im1.depth() == 64);
|
||||||
|
|
||||||
|
const int height = im1.height();
|
||||||
|
const int width = im1.width();
|
||||||
|
for (int i = 0; i < height; ++i) {
|
||||||
|
const Trait *line1 = reinterpret_cast<const Trait *>(im1.scanLine(i));
|
||||||
|
const Trait *line2 = reinterpret_cast<const Trait *>(im2.scanLine(i));
|
||||||
|
for (int j = 0; j < width; ++j) {
|
||||||
|
if (line1[j] > line2[j]) {
|
||||||
|
if (line1[j] - line2[j] > fuzziness) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (line2[j] - line1[j] > fuzziness) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// allow each byte to be different by up to 1, to allow for rounding errors
|
||||||
|
static bool fuzzyeq(const QImage &im1, const QImage &im2, uchar fuzziness)
|
||||||
|
{
|
||||||
|
return (im1.depth() == 64) ? fuzzyeq<quint16>(im1, im2, fuzziness) : fuzzyeq<quint8>(im1, im2, fuzziness);
|
||||||
|
}
|
@ -16,6 +16,8 @@
|
|||||||
|
|
||||||
#include "../tests/format-enum.h"
|
#include "../tests/format-enum.h"
|
||||||
|
|
||||||
|
#include "fuzzyeq.cpp"
|
||||||
|
|
||||||
static void writeImageData(const char *name, const QString &filename, const QImage &image)
|
static void writeImageData(const char *name, const QString &filename, const QImage &image)
|
||||||
{
|
{
|
||||||
QFile file(filename);
|
QFile file(filename);
|
||||||
@ -31,38 +33,6 @@ static void writeImageData(const char *name, const QString &filename, const QIma
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
template<class Trait>
|
|
||||||
static bool fuzzyeq(const QImage &im1, const QImage &im2, uchar fuzziness)
|
|
||||||
{
|
|
||||||
Q_ASSERT(im1.format() == im2.format());
|
|
||||||
Q_ASSERT(im1.depth() == 24 || im1.depth() == 32 || im1.depth() == 64);
|
|
||||||
|
|
||||||
const int height = im1.height();
|
|
||||||
const int width = im1.width();
|
|
||||||
for (int i = 0; i < height; ++i) {
|
|
||||||
const Trait *line1 = reinterpret_cast<const Trait *>(im1.scanLine(i));
|
|
||||||
const Trait *line2 = reinterpret_cast<const Trait *>(im2.scanLine(i));
|
|
||||||
for (int j = 0; j < width; ++j) {
|
|
||||||
if (line1[j] > line2[j]) {
|
|
||||||
if (line1[j] - line2[j] > fuzziness) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (line2[j] - line1[j] > fuzziness) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// allow each byte to be different by up to 1, to allow for rounding errors
|
|
||||||
static bool fuzzyeq(const QImage &im1, const QImage &im2, uchar fuzziness)
|
|
||||||
{
|
|
||||||
return (im1.depth() == 64) ? fuzzyeq<quint16>(im1, im2, fuzziness) : fuzzyeq<quint8>(im1, im2, fuzziness);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Returns the original format if we support, or returns
|
// Returns the original format if we support, or returns
|
||||||
// format which we preferred to use for `fuzzyeq()`.
|
// format which we preferred to use for `fuzzyeq()`.
|
||||||
// We do only support formats with 8-bits/16-bits pre pixel.
|
// We do only support formats with 8-bits/16-bits pre pixel.
|
||||||
|
@ -16,6 +16,8 @@
|
|||||||
#include <QImageWriter>
|
#include <QImageWriter>
|
||||||
#include <QTextStream>
|
#include <QTextStream>
|
||||||
|
|
||||||
|
#include "fuzzyeq.cpp"
|
||||||
|
|
||||||
int main(int argc, char **argv)
|
int main(int argc, char **argv)
|
||||||
{
|
{
|
||||||
QCoreApplication app(argc, argv);
|
QCoreApplication app(argc, argv);
|
||||||
@ -31,7 +33,13 @@ int main(int argc, char **argv)
|
|||||||
parser.addPositionalArgument(QStringLiteral("format"), QStringLiteral("format to test."));
|
parser.addPositionalArgument(QStringLiteral("format"), QStringLiteral("format to test."));
|
||||||
QCommandLineOption lossless(QStringList() << QStringLiteral("l") << QStringLiteral("lossless"),
|
QCommandLineOption lossless(QStringList() << QStringLiteral("l") << QStringLiteral("lossless"),
|
||||||
QStringLiteral("Check that reading back the data gives the same image."));
|
QStringLiteral("Check that reading back the data gives the same image."));
|
||||||
|
QCommandLineOption ignoreDataCheck({QStringLiteral("no-data-check")}, QStringLiteral("Don't check that write data is exactly the same."));
|
||||||
|
QCommandLineOption fuzz(QStringList() << QStringLiteral("f") << QStringLiteral("fuzz"),
|
||||||
|
QStringLiteral("Allow for some deviation in ARGB data."),
|
||||||
|
QStringLiteral("max"));
|
||||||
parser.addOption(lossless);
|
parser.addOption(lossless);
|
||||||
|
parser.addOption(ignoreDataCheck);
|
||||||
|
parser.addOption(fuzz);
|
||||||
|
|
||||||
parser.process(app);
|
parser.process(app);
|
||||||
|
|
||||||
@ -44,11 +52,26 @@ int main(int argc, char **argv)
|
|||||||
parser.showHelp(1);
|
parser.showHelp(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uchar fuzziness = 0;
|
||||||
|
if (parser.isSet(fuzz)) {
|
||||||
|
bool ok;
|
||||||
|
uint fuzzarg = parser.value(fuzz).toUInt(&ok);
|
||||||
|
if (!ok || fuzzarg > 255) {
|
||||||
|
QTextStream(stderr) << "Error: max fuzz argument must be a number between 0 and 255\n";
|
||||||
|
parser.showHelp(1);
|
||||||
|
}
|
||||||
|
fuzziness = uchar(fuzzarg);
|
||||||
|
}
|
||||||
|
|
||||||
QString suffix = args.at(0);
|
QString suffix = args.at(0);
|
||||||
QByteArray format = suffix.toLatin1();
|
QByteArray format = suffix.toLatin1();
|
||||||
|
|
||||||
QDir imgdir(QStringLiteral(IMAGEDIR));
|
QDir imgdir(QStringLiteral(IMAGEDIR));
|
||||||
imgdir.setNameFilters(QStringList(QLatin1String("*.") + suffix));
|
if (parser.isSet(ignoreDataCheck)) {
|
||||||
|
imgdir.setNameFilters({QLatin1String("*.png")});
|
||||||
|
} else {
|
||||||
|
imgdir.setNameFilters(QStringList(QLatin1String("*.") + suffix));
|
||||||
|
}
|
||||||
imgdir.setFilter(QDir::Files);
|
imgdir.setFilter(QDir::Files);
|
||||||
|
|
||||||
int passed = 0;
|
int passed = 0;
|
||||||
@ -58,8 +81,13 @@ int main(int argc, char **argv)
|
|||||||
<< "Starting basic write tests for " << suffix << " images *********\n";
|
<< "Starting basic write tests for " << suffix << " images *********\n";
|
||||||
const QFileInfoList lstImgDir = imgdir.entryInfoList();
|
const QFileInfoList lstImgDir = imgdir.entryInfoList();
|
||||||
for (const QFileInfo &fi : lstImgDir) {
|
for (const QFileInfo &fi : lstImgDir) {
|
||||||
int suffixPos = fi.filePath().count() - suffix.count();
|
QString pngfile;
|
||||||
QString pngfile = fi.filePath().replace(suffixPos, suffix.count(), QStringLiteral("png"));
|
if (parser.isSet(ignoreDataCheck)) {
|
||||||
|
pngfile = fi.filePath();
|
||||||
|
} else {
|
||||||
|
int suffixPos = fi.filePath().count() - suffix.count();
|
||||||
|
pngfile = fi.filePath().replace(suffixPos, suffix.count(), QStringLiteral("png"));
|
||||||
|
}
|
||||||
QString pngfilename = QFileInfo(pngfile).fileName();
|
QString pngfilename = QFileInfo(pngfile).fileName();
|
||||||
|
|
||||||
QImageReader pngReader(pngfile, "png");
|
QImageReader pngReader(pngfile, "png");
|
||||||
@ -70,29 +98,13 @@ int main(int argc, char **argv)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
QFile expFile(fi.filePath());
|
|
||||||
if (!expFile.open(QIODevice::ReadOnly)) {
|
|
||||||
QTextStream(stdout) << "ERROR: " << fi.fileName() << ": could not open " << fi.fileName() << ": " << expFile.errorString() << "\n";
|
|
||||||
++failed;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
QByteArray expData = expFile.readAll();
|
|
||||||
if (expData.isEmpty()) {
|
|
||||||
// check if there was actually anything to read
|
|
||||||
expFile.reset();
|
|
||||||
char buf[1];
|
|
||||||
qint64 result = expFile.read(buf, 1);
|
|
||||||
if (result < 0) {
|
|
||||||
QTextStream(stdout) << "ERROR: " << fi.fileName() << ": could not load " << fi.fileName() << ": " << expFile.errorString() << "\n";
|
|
||||||
++failed;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
QByteArray writtenData;
|
QByteArray writtenData;
|
||||||
{
|
{
|
||||||
QBuffer buffer(&writtenData);
|
QBuffer buffer(&writtenData);
|
||||||
QImageWriter imgWriter(&buffer, format.constData());
|
QImageWriter imgWriter(&buffer, format.constData());
|
||||||
|
if (parser.isSet(lossless)) {
|
||||||
|
imgWriter.setQuality(100);
|
||||||
|
}
|
||||||
if (!imgWriter.write(pngImage)) {
|
if (!imgWriter.write(pngImage)) {
|
||||||
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": failed to write image data\n";
|
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": failed to write image data\n";
|
||||||
++failed;
|
++failed;
|
||||||
@ -100,10 +112,31 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (expData != writtenData) {
|
if (!parser.isSet(ignoreDataCheck)) {
|
||||||
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": written data differs from " << fi.fileName() << "\n";
|
QFile expFile(fi.filePath());
|
||||||
++failed;
|
if (!expFile.open(QIODevice::ReadOnly)) {
|
||||||
continue;
|
QTextStream(stdout) << "ERROR: " << fi.fileName() << ": could not open " << fi.fileName() << ": " << expFile.errorString() << "\n";
|
||||||
|
++failed;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
QByteArray expData = expFile.readAll();
|
||||||
|
if (expData.isEmpty()) {
|
||||||
|
// check if there was actually anything to read
|
||||||
|
expFile.reset();
|
||||||
|
char buf[1];
|
||||||
|
qint64 result = expFile.read(buf, 1);
|
||||||
|
if (result < 0) {
|
||||||
|
QTextStream(stdout) << "ERROR: " << fi.fileName() << ": could not load " << fi.fileName() << ": " << expFile.errorString() << "\n";
|
||||||
|
++failed;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (expData != writtenData) {
|
||||||
|
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": written data differs from " << fi.fileName() << "\n";
|
||||||
|
++failed;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
QImage reReadImage;
|
QImage reReadImage;
|
||||||
@ -119,8 +152,18 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (parser.isSet(lossless)) {
|
if (parser.isSet(lossless)) {
|
||||||
if (pngImage != reReadImage) {
|
if (!fuzzyeq(pngImage, reReadImage, fuzziness)) {
|
||||||
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": re-reading the data resulted in a different image\n";
|
QTextStream(stdout) << "FAIL : " << fi.fileName() << ": re-reading the data resulted in a different image\n";
|
||||||
|
if (pngImage.size() == reReadImage.size()) {
|
||||||
|
for (int i = 0; i < pngImage.width(); ++i) {
|
||||||
|
for (int j = 0; j < pngImage.height(); ++j) {
|
||||||
|
if (pngImage.pixel(i, j) != reReadImage.pixel(i, j)) {
|
||||||
|
QTextStream(stdout) << "Pixel is different " << i << ',' << j << ' ' << pngImage.pixel(i, j) << ' ' << reReadImage.pixel(i, j)
|
||||||
|
<< '\n';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
++failed;
|
++failed;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user