mirror of https://github.com/qTox/qTox.git
Browse Source
Previous exif transformations were not valid. The exif spec defines the orientations as where the 0th row and the 0th column should end. The previous mappings used in qTox did not respect these mappings and needed to be updated.pull/6143/head
7 changed files with 320 additions and 78 deletions
@ -0,0 +1,103 @@
@@ -0,0 +1,103 @@
|
||||
/*
|
||||
Copyright © 2020 by The qTox Project Contributors |
||||
|
||||
This file is part of qTox, a Qt-based graphical interface for Tox. |
||||
|
||||
qTox is libre software: you can redistribute it and/or modify |
||||
it under the terms of the GNU General Public License as published by |
||||
the Free Software Foundation, either version 3 of the License, or |
||||
(at your option) any later version. |
||||
|
||||
qTox is distributed in the hope that it will be useful, |
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
GNU General Public License for more details. |
||||
|
||||
You should have received a copy of the GNU General Public License |
||||
along with qTox. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/ |
||||
|
||||
#include "exiftransform.h" |
||||
|
||||
#include <libexif/exif-loader.h> |
||||
|
||||
#include <QDebug> |
||||
|
||||
namespace ExifTransform |
||||
{ |
||||
Orientation getOrientation(QByteArray imageData) |
||||
{ |
||||
auto data = imageData.constData(); |
||||
auto size = imageData.size(); |
||||
|
||||
ExifData* exifData = exif_data_new_from_data(reinterpret_cast<const unsigned char*>(data), size); |
||||
|
||||
if (!exifData) { |
||||
return Orientation::TopLeft; |
||||
} |
||||
|
||||
int orientation = 0; |
||||
const ExifByteOrder byteOrder = exif_data_get_byte_order(exifData); |
||||
const ExifEntry* const exifEntry = exif_data_get_entry(exifData, EXIF_TAG_ORIENTATION); |
||||
if (exifEntry) { |
||||
orientation = exif_get_short(exifEntry->data, byteOrder); |
||||
} |
||||
exif_data_free(exifData); |
||||
|
||||
switch (orientation){ |
||||
case 1: |
||||
return Orientation::TopLeft; |
||||
case 2: |
||||
return Orientation::TopRight; |
||||
case 3: |
||||
return Orientation::BottomRight; |
||||
case 4: |
||||
return Orientation::BottomLeft; |
||||
case 5: |
||||
return Orientation::LeftTop; |
||||
case 6: |
||||
return Orientation::RightTop; |
||||
case 7: |
||||
return Orientation::RightBottom; |
||||
case 8: |
||||
return Orientation::LeftBottom; |
||||
default: |
||||
qWarning() << "Invalid exif orientation"; |
||||
return Orientation::TopLeft; |
||||
} |
||||
} |
||||
|
||||
QImage applyTransformation(QImage image, Orientation orientation) |
||||
{ |
||||
QTransform exifTransform; |
||||
switch (orientation) { |
||||
case Orientation::TopLeft: |
||||
break; |
||||
case Orientation::TopRight: |
||||
image = image.mirrored(1, 0); |
||||
break; |
||||
case Orientation::BottomRight: |
||||
exifTransform.rotate(180); |
||||
break; |
||||
case Orientation::BottomLeft: |
||||
image = image.mirrored(0, 1); |
||||
break; |
||||
case Orientation::LeftTop: |
||||
exifTransform.rotate(-90); |
||||
image = image.mirrored(1, 0); |
||||
break; |
||||
case Orientation::RightTop: |
||||
exifTransform.rotate(90); |
||||
break; |
||||
case Orientation::RightBottom: |
||||
exifTransform.rotate(90); |
||||
image = image.mirrored(1, 0); |
||||
break; |
||||
case Orientation::LeftBottom: |
||||
exifTransform.rotate(-90); |
||||
break; |
||||
} |
||||
image = image.transformed(exifTransform); |
||||
return image; |
||||
} |
||||
}; |
@ -0,0 +1,45 @@
@@ -0,0 +1,45 @@
|
||||
/*
|
||||
Copyright © 2020 by The qTox Project Contributors |
||||
|
||||
This file is part of qTox, a Qt-based graphical interface for Tox. |
||||
|
||||
qTox is libre software: you can redistribute it and/or modify |
||||
it under the terms of the GNU General Public License as published by |
||||
the Free Software Foundation, either version 3 of the License, or |
||||
(at your option) any later version. |
||||
|
||||
qTox is distributed in the hope that it will be useful, |
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
GNU General Public License for more details. |
||||
|
||||
You should have received a copy of the GNU General Public License |
||||
along with qTox. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/ |
||||
|
||||
#pragma once |
||||
|
||||
#include <QPixmap> |
||||
|
||||
namespace ExifTransform |
||||
{ |
||||
enum class Orientation |
||||
{ |
||||
/* name corresponds to where the 0 row and 0 column is in form row-column
|
||||
* i.e. entry 5 here means that the 0'th row corresponds to the left side of the scene and |
||||
* the 0'th column corresponds to the top of the captured scene. This means that the image |
||||
* needs to be mirrored and rotated to be displayed. |
||||
*/ |
||||
TopLeft, |
||||
TopRight, |
||||
BottomRight, |
||||
BottomLeft, |
||||
LeftTop, |
||||
RightTop, |
||||
RightBottom, |
||||
LeftBottom |
||||
}; |
||||
|
||||
Orientation getOrientation(QByteArray imageData); |
||||
QImage applyTransformation(QImage image, Orientation orientation); |
||||
}; |
@ -0,0 +1,165 @@
@@ -0,0 +1,165 @@
|
||||
/*
|
||||
Copyright © 2020 by The qTox Project Contributors |
||||
|
||||
This file is part of qTox, a Qt-based graphical interface for Tox. |
||||
|
||||
qTox is libre software: you can redistribute it and/or modify |
||||
it under the terms of the GNU General Public License as published by |
||||
the Free Software Foundation, either version 3 of the License, or |
||||
(at your option) any later version. |
||||
|
||||
qTox is distributed in the hope that it will be useful, |
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of |
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
||||
GNU General Public License for more details. |
||||
|
||||
You should have received a copy of the GNU General Public License |
||||
along with qTox. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/ |
||||
|
||||
#include "src/model/exiftransform.h" |
||||
|
||||
#include <QPainter> |
||||
#include <QTest> |
||||
|
||||
static const auto rowColor = QColor(Qt::green).rgb(); |
||||
static const auto colColor = QColor(Qt::blue).rgb(); |
||||
|
||||
enum class Side |
||||
{ |
||||
top, |
||||
bottom, |
||||
left, |
||||
right |
||||
}; |
||||
|
||||
static QPoint getPosition(Side side) |
||||
{ |
||||
int x, y; |
||||
switch (side) |
||||
{ |
||||
case Side::top: |
||||
{ |
||||
x = 1; |
||||
y = 0; |
||||
break; |
||||
} |
||||
case Side::bottom: |
||||
{ |
||||
x = 1; |
||||
y = 2; |
||||
break; |
||||
} |
||||
case Side::left: |
||||
{ |
||||
x = 0; |
||||
y = 1; |
||||
break; |
||||
} |
||||
case Side::right: |
||||
{ |
||||
x = 2; |
||||
y = 1; |
||||
break; |
||||
} |
||||
} |
||||
|
||||
return {x, y}; |
||||
} |
||||
|
||||
static QRgb getColor(const QImage& image, Side side) |
||||
{ |
||||
return image.pixel(getPosition(side)); |
||||
} |
||||
|
||||
|
||||
class TestExifTransform : public QObject |
||||
{ |
||||
Q_OBJECT |
||||
|
||||
private slots: |
||||
void init(); |
||||
void testTopLeft(); |
||||
void testTopRight(); |
||||
void testBottomRight(); |
||||
void testBottomLeft(); |
||||
void testLeftTop(); |
||||
void testRightTop(); |
||||
void testRightBottom(); |
||||
void testLeftBottom(); |
||||
|
||||
private: |
||||
QImage inputImage; |
||||
}; |
||||
|
||||
void TestExifTransform::init() |
||||
{ |
||||
inputImage = QImage(QSize(3, 3), QImage::Format_RGB32); |
||||
QPainter painter(&inputImage); |
||||
painter.fillRect(QRect(0, 0, 3, 3), Qt::white); |
||||
// First row has a green dot in the middle
|
||||
painter.setPen(rowColor); |
||||
painter.drawPoint(getPosition(Side::top)); |
||||
// First column has a blue dot in the middle
|
||||
painter.setPen(colColor); |
||||
painter.drawPoint(getPosition(Side::left)); |
||||
} |
||||
|
||||
void TestExifTransform::testTopLeft() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::TopLeft); |
||||
QVERIFY(getColor(image, Side::top) == rowColor); |
||||
QVERIFY(getColor(image, Side::left) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testTopRight() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::TopRight); |
||||
QVERIFY(getColor(image, Side::top) == rowColor); |
||||
QVERIFY(getColor(image, Side::right) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testBottomRight() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::BottomRight); |
||||
QVERIFY(getColor(image, Side::bottom) == rowColor); |
||||
QVERIFY(getColor(image, Side::right) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testBottomLeft() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::BottomLeft); |
||||
QVERIFY(getColor(image, Side::bottom) == rowColor); |
||||
QVERIFY(getColor(image, Side::left) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testLeftTop() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::LeftTop); |
||||
QVERIFY(getColor(image, Side::left) == rowColor); |
||||
QVERIFY(getColor(image, Side::top) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testRightTop() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::RightTop); |
||||
QVERIFY(getColor(image, Side::right) == rowColor); |
||||
QVERIFY(getColor(image, Side::top) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testRightBottom() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::RightBottom); |
||||
QVERIFY(getColor(image, Side::right) == rowColor); |
||||
QVERIFY(getColor(image, Side::bottom) == colColor); |
||||
} |
||||
|
||||
void TestExifTransform::testLeftBottom() |
||||
{ |
||||
auto image = ExifTransform::applyTransformation(inputImage, ExifTransform::Orientation::LeftBottom); |
||||
QVERIFY(getColor(image, Side::left) == rowColor); |
||||
QVERIFY(getColor(image, Side::bottom) == colColor); |
||||
} |
||||
|
||||
QTEST_APPLESS_MAIN(TestExifTransform) |
||||
#include "exiftransform_test.moc" |
Loading…
Reference in new issue