1
0
mirror of https://github.com/qTox/qTox.git synced 2024-03-22 14:00:36 +08:00

Merge pull request #6143

Mick Sayson (1):
      fix(preview): Fix exif orientations
This commit is contained in:
Anthony Bilinski 2020-05-16 17:46:28 -07:00
commit 14571abaa3
No known key found for this signature in database
GPG Key ID: 2AA8E0DA1B31FB3C
7 changed files with 320 additions and 78 deletions

View File

@ -339,6 +339,8 @@ set(${PROJECT_NAME}_SOURCES
src/model/contact.h
src/model/chatlogitem.cpp
src/model/chatlogitem.h
src/model/exiftransform.h
src/model/exiftransform.cpp
src/model/friend.cpp
src/model/friend.h
src/model/message.h

View File

@ -47,6 +47,7 @@ auto_test(model friendmessagedispatcher)
auto_test(model groupmessagedispatcher)
auto_test(model messageprocessor)
auto_test(model sessionchatlog)
auto_test(model exiftransform)
if (UNIX)
auto_test(platform posixsignalnotifier)

View File

@ -26,8 +26,7 @@
#include "src/widget/gui.h"
#include "src/widget/style.h"
#include "src/widget/widget.h"
#include <libexif/exif-loader.h>
#include "src/model/exiftransform.h"
#include <QBuffer>
#include <QDebug>
@ -523,13 +522,11 @@ void FileTransferWidget::showPreview(const QString& filename)
if (!imageFile.open(QIODevice::ReadOnly)) {
return;
}
const QByteArray imageFileData = imageFile.readAll();
QImage image = QImage::fromData(imageFileData);
const int exifOrientation =
getExifOrientation(imageFileData.constData(), imageFileData.size());
if (exifOrientation) {
applyTransformation(exifOrientation, image);
}
auto orientation = ExifTransform::getOrientation(imageFileData);
image = ExifTransform::applyTransformation(image, orientation);
const QPixmap iconPixmap = scaleCropIntoSquare(QPixmap::fromImage(image), size);
@ -599,59 +596,6 @@ QPixmap FileTransferWidget::scaleCropIntoSquare(const QPixmap& source, const int
return result;
}
int FileTransferWidget::getExifOrientation(const char* data, const int size)
{
ExifData* exifData = exif_data_new_from_data(reinterpret_cast<const unsigned char*>(data), size);
if (!exifData) {
return 0;
}
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);
return orientation;
}
void FileTransferWidget::applyTransformation(const int orientation, QImage& image)
{
QTransform exifTransform;
switch (static_cast<ExifOrientation>(orientation)) {
case ExifOrientation::TopLeft:
break;
case ExifOrientation::TopRight:
image = image.mirrored(1, 0);
break;
case ExifOrientation::BottomRight:
exifTransform.rotate(180);
break;
case ExifOrientation::BottomLeft:
image = image.mirrored(0, 1);
break;
case ExifOrientation::LeftTop:
exifTransform.rotate(90);
image = image.mirrored(0, 1);
break;
case ExifOrientation::RightTop:
exifTransform.rotate(-90);
break;
case ExifOrientation::RightBottom:
exifTransform.rotate(-90);
image = image.mirrored(0, 1);
break;
case ExifOrientation::LeftBottom:
exifTransform.rotate(90);
break;
default:
qWarning() << "Invalid exif orientation passed to applyTransformation!";
}
image = image.transformed(exifTransform);
}
void FileTransferWidget::updateWidget(ToxFile const& file)
{
assert(file == fileInfo);

View File

@ -89,22 +89,4 @@ private:
bool active;
ToxFile::FileStatus lastStatus = ToxFile::INITIALIZING;
enum class ExifOrientation
{
/* do not change values, this is exif spec
*
* 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 = 1,
TopRight = 2,
BottomRight = 3,
BottomLeft = 4,
LeftTop = 5,
RightTop = 6,
RightBottom = 7,
LeftBottom = 8
};
};

103
src/model/exiftransform.cpp Normal file
View File

@ -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;
}
};

45
src/model/exiftransform.h Normal file
View File

@ -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);
};

View File

@ -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"