8#include "microexif_p.h"
12#include <QCoreApplication>
18#define TIFF_IMAGEWIDTH 0x100
19#define TIFF_IMAGEHEIGHT 0x101
20#define TIFF_BITSPERSAMPLE 0x102
21#define TIFF_IMAGEDESCRIPTION 0x10E
22#define TIFF_MAKE 0x10F
23#define TIFF_MODEL 0x110
24#define TIFF_ORIENT 0x0112
25#define TIFF_XRES 0x011A
26#define TIFF_YRES 0x011B
27#define TIFF_URES 0x0128
28#define TIFF_SOFTWARE 0x0131
29#define TIFF_ARTIST 0x013B
30#define TIFF_DATETIME 0x0132
31#define TIFF_COPYRIGHT 0x8298
33#define TIFF_VAL_URES_NOABSOLUTE 1
34#define TIFF_VAL_URES_INCH 2
35#define TIFF_VAL_URES_CENTIMETER 3
38#define EXIF_EXIFIFD 0x8769
39#define EXIF_GPSIFD 0x8825
40#define EXIF_EXIFVERSION 0x9000
41#define EXIF_DATETIMEORIGINAL 0x9003
42#define EXIF_DATETIMEDIGITIZED 0x9004
43#define EXIF_OFFSETTIME 0x9010
44#define EXIF_OFFSETTIMEORIGINAL 0x9011
45#define EXIF_OFFSETTIMEDIGITIZED 0x9012
46#define EXIF_COLORSPACE 0xA001
47#define EXIF_PIXELXDIM 0xA002
48#define EXIF_PIXELYDIM 0xA003
49#define EXIF_IMAGEUNIQUEID 0xA420
50#define EXIF_BODYSERIALNUMBER 0xA431
51#define EXIF_LENSMAKE 0xA433
52#define EXIF_LENSMODEL 0xA434
53#define EXIF_LENSSERIALNUMBER 0xA435
54#define EXIF_IMAGETITLE 0xA436
56#define EXIF_VAL_COLORSPACE_SRGB 1
57#define EXIF_VAL_COLORSPACE_UNCAL 0xFFFF
59#define GPS_GPSVERSION 0
60#define GPS_LATITUDEREF 1
62#define GPS_LONGITUDEREF 3
63#define GPS_LONGITUDE 4
64#define GPS_ALTITUDEREF 5
66#define GPS_IMGDIRECTIONREF 16
67#define GPS_IMGDIRECTION 17
68#define EXIF_TAG_VALUE(n, byteSize) (((n) << 6) | ((byteSize) & 0x3F))
69#define EXIF_TAG_SIZEOF(dataType) (quint16(dataType) & 0x3F)
70#define EXIF_TAG_DATATYPE(dataType) (quint16(dataType) >> 6)
72enum class ExifTagType : quint16 {
74 Byte = EXIF_TAG_VALUE(1, 1),
75 Ascii = EXIF_TAG_VALUE(2, 1),
76 Short = EXIF_TAG_VALUE(3, 2),
77 Long = EXIF_TAG_VALUE(4, 4),
78 Rational = EXIF_TAG_VALUE(5, 8),
81 SByte = EXIF_TAG_VALUE(6, 1),
82 Undefined = EXIF_TAG_VALUE(7, 1),
83 SShort = EXIF_TAG_VALUE(8, 2),
84 SLong = EXIF_TAG_VALUE(9, 4),
85 SRational = EXIF_TAG_VALUE(10, 8),
87 Float = EXIF_TAG_VALUE(11, 4),
88 Double = EXIF_TAG_VALUE(12, 8),
89 Ifd = EXIF_TAG_VALUE(13, 4),
92 Long8 = EXIF_TAG_VALUE(16, 8),
93 SLong8 = EXIF_TAG_VALUE(17, 8),
94 Ifd8 = EXIF_TAG_VALUE(18, 8),
97 Utf8 = EXIF_TAG_VALUE(129, 1)
102using TagInfo = std::pair<quint16, ExifTagType>;
110static const KnownTags staticTagTypes = {
111 TagInfo(TIFF_IMAGEWIDTH, ExifTagType::Long),
112 TagInfo(TIFF_IMAGEHEIGHT, ExifTagType::Long),
113 TagInfo(TIFF_BITSPERSAMPLE, ExifTagType::Short),
114 TagInfo(TIFF_IMAGEDESCRIPTION, ExifTagType::Ascii),
115 TagInfo(TIFF_MAKE, ExifTagType::Ascii),
116 TagInfo(TIFF_MODEL, ExifTagType::Ascii),
117 TagInfo(TIFF_ORIENT, ExifTagType::Short),
118 TagInfo(TIFF_XRES, ExifTagType::Rational),
119 TagInfo(TIFF_YRES, ExifTagType::Rational),
120 TagInfo(TIFF_URES, ExifTagType::Short),
121 TagInfo(TIFF_SOFTWARE, ExifTagType::Ascii),
122 TagInfo(TIFF_ARTIST, ExifTagType::Ascii),
123 TagInfo(TIFF_DATETIME, ExifTagType::Ascii),
124 TagInfo(TIFF_COPYRIGHT, ExifTagType::Ascii),
125 TagInfo(EXIF_EXIFIFD, ExifTagType::Long),
126 TagInfo(EXIF_GPSIFD, ExifTagType::Long),
127 TagInfo(EXIF_DATETIMEORIGINAL, ExifTagType::Ascii),
128 TagInfo(EXIF_OFFSETTIMEDIGITIZED, ExifTagType::Ascii),
129 TagInfo(EXIF_OFFSETTIME, ExifTagType::Ascii),
130 TagInfo(EXIF_OFFSETTIMEORIGINAL, ExifTagType::Ascii),
131 TagInfo(EXIF_OFFSETTIMEDIGITIZED, ExifTagType::Ascii),
132 TagInfo(EXIF_COLORSPACE, ExifTagType::Short),
133 TagInfo(EXIF_PIXELXDIM, ExifTagType::Long),
134 TagInfo(EXIF_PIXELYDIM, ExifTagType::Long),
135 TagInfo(EXIF_IMAGEUNIQUEID, ExifTagType::Ascii),
136 TagInfo(EXIF_BODYSERIALNUMBER, ExifTagType::Ascii),
137 TagInfo(EXIF_LENSMAKE, ExifTagType::Ascii),
138 TagInfo(EXIF_LENSMODEL, ExifTagType::Ascii),
139 TagInfo(EXIF_LENSSERIALNUMBER, ExifTagType::Ascii),
140 TagInfo(EXIF_IMAGETITLE, ExifTagType::Ascii),
141 TagInfo(EXIF_EXIFVERSION, ExifTagType::Undefined)
149static const KnownTags staticGpsTagTypes = {
150 TagInfo(GPS_GPSVERSION, ExifTagType::Byte),
151 TagInfo(GPS_LATITUDEREF, ExifTagType::Ascii),
152 TagInfo(GPS_LATITUDE, ExifTagType::Rational),
153 TagInfo(GPS_LONGITUDEREF, ExifTagType::Ascii),
154 TagInfo(GPS_LONGITUDE, ExifTagType::Rational),
155 TagInfo(GPS_ALTITUDEREF, ExifTagType::Byte),
156 TagInfo(GPS_ALTITUDE, ExifTagType::Rational),
157 TagInfo(GPS_IMGDIRECTIONREF, ExifTagType::Ascii),
158 TagInfo(GPS_IMGDIRECTION, ExifTagType::Rational)
168 std::pair<quint16, QString>(TIFF_IMAGEDESCRIPTION, QStringLiteral(META_KEY_DESCRIPTION)),
169 std::pair<quint16, QString>(TIFF_ARTIST, QStringLiteral(META_KEY_AUTHOR)),
170 std::pair<quint16, QString>(TIFF_SOFTWARE, QStringLiteral(META_KEY_SOFTWARE)),
171 std::pair<quint16, QString>(TIFF_COPYRIGHT, QStringLiteral(META_KEY_COPYRIGHT)),
172 std::pair<quint16, QString>(TIFF_MAKE, QStringLiteral(META_KEY_MANUFACTURER)),
173 std::pair<quint16, QString>(TIFF_MODEL, QStringLiteral(META_KEY_MODEL))
183 std::pair<quint16, QString>(EXIF_BODYSERIALNUMBER, QStringLiteral(META_KEY_SERIALNUMBER)),
184 std::pair<quint16, QString>(EXIF_LENSMAKE, QStringLiteral(META_KEY_LENS_MANUFACTURER)),
185 std::pair<quint16, QString>(EXIF_LENSMODEL, QStringLiteral(META_KEY_LENS_MODEL)),
186 std::pair<quint16, QString>(EXIF_LENSSERIALNUMBER, QStringLiteral(META_KEY_LENS_SERIALNUMBER)),
187 std::pair<quint16, QString>(EXIF_IMAGETITLE, QStringLiteral(META_KEY_TITLE)),
196static qint16 timeOffset(
const QString& offset)
198 if (offset.
size() != 6 || offset.
at(3) != u
':')
204 auto mm = offset.
mid(4, 2).
toInt(&ok) * (hh < 0 ? -1 : 1);
207 return qint16(hh * 60 + mm);
215static QString timeOffset(qint16 offset)
217 auto absOff = quint16(std::abs(offset));
218 return QStringLiteral(
"%1%2:%3")
219 .arg(offset < 0 ? QStringLiteral(
"-") : QStringLiteral(
"+"))
220 .arg(absOff / 60, 2, 10,
QChar(u
'0'))
221 .arg(absOff % 60, 2, 10,
QChar(u
'0'));
234 if (order == 0x4949) {
236 }
else if (order == 0x4d4d) {
244 if (version != 0x002A && version != 0x01BC)
275static qint32 countBytes(
const ExifTagType &dataType,
const QVariant &value)
278 if (dataType == ExifTagType::Ascii) {
280 }
else if (dataType == ExifTagType::Utf8) {
282 }
else if (dataType == ExifTagType::Undefined) {
284 }
else if (dataType == ExifTagType::Byte) {
286 }
else if (dataType == ExifTagType::Short) {
288 }
else if (dataType == ExifTagType::Long || dataType == ExifTagType::Ifd) {
290 }
else if (dataType == ExifTagType::SByte) {
292 }
else if (dataType == ExifTagType::SShort) {
294 }
else if (dataType == ExifTagType::SLong) {
296 }
else if (dataType == ExifTagType::Rational || dataType == ExifTagType::SRational || dataType == ExifTagType::Double) {
298 }
else if (dataType == ExifTagType::Float) {
301 return std::max(1, count);
310 for (;l.size() < qsizetype(4 /
sizeof(T));)
316inline qint32 rationalPrecision(
double v)
319 return 8 - qBound(0, v < 1 ? 8 :
int(std::log10(v)), 8);
329 auto den = std::pow(10, rationalPrecision(v));
330 ds << T(qRound(v * den));
339 for (
auto n = ba.size(); n < 4; ++n)
345 if (dataType == ExifTagType::Ascii) {
347 }
else if (dataType == ExifTagType::Utf8) {
349 }
else if (dataType == ExifTagType::Undefined) {
351 }
else if (dataType == ExifTagType::Byte) {
352 writeList<quint8>(ds, value);
353 }
else if (dataType == ExifTagType::SByte) {
354 writeList<qint8>(ds, value);
355 }
else if (dataType == ExifTagType::Short) {
356 writeList<quint16>(ds, value);
357 }
else if (dataType == ExifTagType::SShort) {
358 writeList<qint16>(ds, value);
359 }
else if (dataType == ExifTagType::Long || dataType == ExifTagType::Ifd) {
360 writeList<quint32>(ds, value);
361 }
else if (dataType == ExifTagType::SLong) {
362 writeList<qint32>(ds, value);
363 }
else if (dataType == ExifTagType::Rational) {
364 writeRationalList<quint32>(ds, value);
365 }
else if (dataType == ExifTagType::SRational) {
366 writeRationalList<qint32>(ds, value);
378static bool writeIfd(
QDataStream &ds,
const MicroExif::Tags &tags, TagPos &positions, quint32 pos = 0,
const KnownTags &knownTags = staticTagTypes)
382 if (!updatePos(ds, pos))
385 auto keys = tags.keys();
386 auto entries = quint16(keys.size());
388 for (
auto &&key : keys) {
389 if (!knownTags.contains(key)) {
392 auto value = tags.
value(key);
393 auto dataType = knownTags.value(key);
394 auto count = countBytes(dataType, value);
397 ds << quint16(EXIF_TAG_DATATYPE(dataType));
398 ds << quint32(count);
400 auto valueSize = count * EXIF_TAG_SIZEOF(dataType);
404 writeData(ds, value, dataType);
411 for (
auto &&key : keys) {
412 if (!knownTags.contains(key)) {
416 auto value = tags.
value(key);
417 auto dataType = knownTags.value(key);
418 auto count = countBytes(dataType, value);
419 auto valueSize = count * EXIF_TAG_SIZEOF(dataType);
422 if (!updatePos(ds, positions.
value(key)))
424 writeData(ds, value, dataType);
435 for (quint32 i = 0; i < count; ++i) {
439 for (
auto n = count; n < quint32(4 /
sizeof(T)); ++n) {
449 for (quint32 i = 0; i < count; ++i) {
454 l.
append(den == 0 ? 0 :
double(num) /
double(den));
466 for (quint32 i = 0; i < count; ++i) {
470 if (asciiz && l.
at(l.
size() - 1) == 0) {
473 for (
auto n = count; n < 4; ++n) {
488static bool readIfd(
QDataStream &ds, MicroExif::Tags &tags, quint32 pos = 0,
const KnownTags &knownTags = staticTagTypes, quint32 *nextIfd =
nullptr)
490 auto localNextIfd = quint32();
491 if (nextIfd ==
nullptr)
492 nextIfd = &localNextIfd;
495 auto device = ds.
device();
496 if (pos && !device->seek(pos))
504 for (quint16 i = 0; i < entries; ++i) {
515 if (!knownTags.contains(tagId)) {
522 auto toRead = qint64(count) * EXIF_TAG_SIZEOF(knownTags.value(tagId));
523 if (toRead > qint64(device->size()))
526 auto curPos = qint64();
530 curPos = device->pos();
531 if (!device->seek(value))
535 if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Ascii) || dataType == EXIF_TAG_DATATYPE(ExifTagType::Utf8)) {
536 auto l = readBytes(ds, count,
true);
539 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Undefined)) {
540 auto l = readBytes(ds, count,
false);
542 tags.insert(tagId, l);
543 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Byte)) {
544 auto l = readList<quint8>(ds, count);
546 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::SByte)) {
547 auto l = readList<qint8>(ds, count);
549 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Short)) {
550 auto l = readList<quint16>(ds, count);
552 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::SShort)) {
553 auto l = readList<qint16>(ds, count);
555 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Long) || dataType == EXIF_TAG_DATATYPE(ExifTagType::Ifd)) {
556 auto l = readList<quint32>(ds, count);
558 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::SLong)) {
559 auto l = readList<qint32>(ds, count);
561 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::Rational)) {
562 auto l = readRationalList<quint32>(ds, count);
564 }
else if (dataType == EXIF_TAG_DATATYPE(ExifTagType::SRational)) {
565 auto l = readRationalList<qint32>(ds, count);
569 if (curPos > 0 && !device->seek(curPos))
577MicroExif::MicroExif()
582void MicroExif::clear()
589bool MicroExif::isEmpty()
const
591 return m_tiffTags.isEmpty() && m_exifTags.isEmpty() && m_gpsTags.isEmpty();
594double MicroExif::horizontalResolution()
const
596 auto u = m_tiffTags.value(TIFF_URES).toUInt();
597 auto v = m_tiffTags.value(TIFF_XRES).toDouble();
598 if (u == TIFF_VAL_URES_CENTIMETER)
603void MicroExif::setHorizontalResolution(
double hres)
605 auto u = m_tiffTags.value(TIFF_URES).toUInt();
606 if (u == TIFF_VAL_URES_CENTIMETER) {
608 }
else if (u < TIFF_VAL_URES_INCH) {
609 m_tiffTags.insert(TIFF_URES, TIFF_VAL_URES_INCH);
611 m_tiffTags.insert(TIFF_XRES, hres);
614double MicroExif::verticalResolution()
const
616 auto u = m_tiffTags.value(TIFF_URES).toUInt();
617 auto v = m_tiffTags.value(TIFF_YRES).toDouble();
618 if (u == TIFF_VAL_URES_CENTIMETER)
623void MicroExif::setVerticalResolution(
double vres)
625 auto u = m_tiffTags.value(TIFF_URES).toUInt();
626 if (u == TIFF_VAL_URES_CENTIMETER) {
628 }
else if (u < TIFF_VAL_URES_INCH) {
629 m_tiffTags.insert(TIFF_URES, TIFF_VAL_URES_INCH);
631 m_tiffTags.insert(TIFF_YRES, vres);
636 if (m_exifTags.value(EXIF_COLORSPACE).toUInt() == EXIF_VAL_COLORSPACE_SRGB)
638 return QColorSpace();
641void MicroExif::setColorSpace(
const QColorSpace &cs)
643 auto srgb = cs.
transferFunction() == QColorSpace::TransferFunction::SRgb && cs.
primaries() == QColorSpace::Primaries::SRgb;
644 m_exifTags.insert(EXIF_COLORSPACE, srgb ? EXIF_VAL_COLORSPACE_SRGB : EXIF_VAL_COLORSPACE_UNCAL);
650 m_exifTags.insert(EXIF_COLORSPACE, srgb ? EXIF_VAL_COLORSPACE_SRGB : EXIF_VAL_COLORSPACE_UNCAL);
653qint32 MicroExif::width()
const
655 return m_tiffTags.value(TIFF_IMAGEWIDTH).toUInt();
658void MicroExif::setWidth(qint32 w)
660 m_tiffTags.insert(TIFF_IMAGEWIDTH, w);
661 m_exifTags.insert(EXIF_PIXELXDIM, w);
664qint32 MicroExif::height()
const
666 return m_tiffTags.value(TIFF_IMAGEHEIGHT).toUInt();
669void MicroExif::setHeight(qint32 h)
671 m_tiffTags.insert(TIFF_IMAGEHEIGHT, h);
672 m_exifTags.insert(EXIF_PIXELYDIM, h);
675quint16 MicroExif::orientation()
const
677 return m_tiffTags.value(TIFF_ORIENT).toUInt();
680void MicroExif::setOrientation(quint16 orient)
682 if (orient < 1 || orient > 8)
683 m_tiffTags.remove(TIFF_ORIENT);
685 m_tiffTags.insert(TIFF_ORIENT, orient);
690 switch (orientation()) {
746QString MicroExif::software()
const
748 return tiffString(TIFF_SOFTWARE);
751void MicroExif::setSoftware(
const QString &s)
753 setTiffString(TIFF_SOFTWARE, s);
756QString MicroExif::description()
const
758 return tiffString(TIFF_IMAGEDESCRIPTION);
761void MicroExif::setDescription(
const QString &s)
763 setTiffString(TIFF_IMAGEDESCRIPTION, s);
766QString MicroExif::artist()
const
768 return tiffString(TIFF_ARTIST);
771void MicroExif::setArtist(
const QString &s)
773 setTiffString(TIFF_ARTIST, s);
776QString MicroExif::copyright()
const
778 return tiffString(TIFF_COPYRIGHT);
781void MicroExif::setCopyright(
const QString &s)
783 setTiffString(TIFF_COPYRIGHT, s);
788 return tiffString(TIFF_MAKE);
791void MicroExif::setMake(
const QString &s)
793 setTiffString(TIFF_MAKE, s);
796QString MicroExif::model()
const
798 return tiffString(TIFF_MODEL);
801void MicroExif::setModel(
const QString &s)
803 setTiffString(TIFF_MODEL, s);
806QString MicroExif::serialNumber()
const
808 return tiffString(EXIF_BODYSERIALNUMBER);
811void MicroExif::setSerialNumber(
const QString &s)
813 setTiffString(EXIF_BODYSERIALNUMBER, s);
816QString MicroExif::lensMake()
const
818 return tiffString(EXIF_LENSMAKE);
821void MicroExif::setLensMake(
const QString &s)
823 setTiffString(EXIF_LENSMAKE, s);
826QString MicroExif::lensModel()
const
828 return tiffString(EXIF_LENSMODEL);
831void MicroExif::setLensModel(
const QString &s)
833 setTiffString(EXIF_LENSMODEL, s);
836QString MicroExif::lensSerialNumber()
const
838 return tiffString(EXIF_LENSSERIALNUMBER);
841void MicroExif::setLensSerialNumber(
const QString &s)
843 setTiffString(EXIF_LENSSERIALNUMBER, s);
849 auto ofTag = exifString(EXIF_OFFSETTIME);
850 if (dt.isValid() && !ofTag.isEmpty())
855void MicroExif::setDateTime(
const QDateTime &dt)
858 m_tiffTags.remove(TIFF_DATETIME);
859 m_exifTags.remove(EXIF_OFFSETTIME);
862 setTiffString(TIFF_DATETIME, dt.
toString(QStringLiteral(
"yyyy:MM:dd HH:mm:ss")));
863 setExifString(EXIF_OFFSETTIME, timeOffset(dt.
offsetFromUtc() / 60));
866QDateTime MicroExif::dateTimeOriginal()
const
868 auto dt =
QDateTime::fromString(exifString(EXIF_DATETIMEORIGINAL), QStringLiteral(
"yyyy:MM:dd HH:mm:ss"));
869 auto ofTag = exifString(EXIF_OFFSETTIMEORIGINAL);
870 if (dt.
isValid() && !ofTag.isEmpty())
875void MicroExif::setDateTimeOriginal(
const QDateTime &dt)
878 m_exifTags.remove(EXIF_DATETIMEORIGINAL);
879 m_exifTags.remove(EXIF_OFFSETTIMEORIGINAL);
882 setExifString(EXIF_DATETIMEORIGINAL, dt.
toString(QStringLiteral(
"yyyy:MM:dd HH:mm:ss")));
883 setExifString(EXIF_OFFSETTIMEORIGINAL, timeOffset(dt.
offsetFromUtc() / 60));
886QDateTime MicroExif::dateTimeDigitized()
const
888 auto dt =
QDateTime::fromString(exifString(EXIF_DATETIMEDIGITIZED), QStringLiteral(
"yyyy:MM:dd HH:mm:ss"));
889 auto ofTag = exifString(EXIF_OFFSETTIMEDIGITIZED);
890 if (dt.
isValid() && !ofTag.isEmpty())
895void MicroExif::setDateTimeDigitized(
const QDateTime &dt)
898 m_exifTags.remove(EXIF_DATETIMEDIGITIZED);
899 m_exifTags.remove(EXIF_OFFSETTIMEDIGITIZED);
902 setExifString(EXIF_DATETIMEDIGITIZED, dt.
toString(QStringLiteral(
"yyyy:MM:dd HH:mm:ss")));
903 setExifString(EXIF_OFFSETTIMEDIGITIZED, timeOffset(dt.
offsetFromUtc() / 60));
906QString MicroExif::title()
const
908 return exifString(EXIF_IMAGETITLE);
911void MicroExif::setImageTitle(
const QString &s)
913 setExifString(EXIF_IMAGETITLE, s);
916QUuid MicroExif::uniqueId()
const
918 auto s = exifString(EXIF_IMAGEUNIQUEID);
919 if (s.length() == 32) {
920 auto tmp = QStringLiteral(
"%1-%2-%3-%4-%5").arg(s.left(8), s.mid(8, 4), s.mid(12, 4), s.mid(16, 4), s.mid(20));
926void MicroExif::setUniqueId(
const QUuid &uuid)
929 setExifString(EXIF_IMAGEUNIQUEID, QString());
934double MicroExif::latitude()
const
936 auto ref = gpsString(GPS_LATITUDEREF).toUpper();
937 if (ref != QStringLiteral(
"N") && ref != QStringLiteral(
"S"))
939 auto lat = m_gpsTags.value(GPS_LATITUDE).value<QList<double>>();
942 auto degree = lat.at(0) + lat.at(1) / 60 + lat.at(2) / 3600;
943 if (degree < -90.0 || degree > 90.0)
945 return ref == QStringLiteral(
"N") ? degree : -degree;
948void MicroExif::setLatitude(
double degree)
950 if (qIsNaN(degree)) {
951 m_gpsTags.remove(GPS_LATITUDEREF);
952 m_gpsTags.remove(GPS_LATITUDE);
954 if (degree < -90.0 || degree > 90.0)
956 auto adeg = qAbs(degree);
957 auto min = (adeg - int(adeg)) * 60;
958 auto sec = (min - int(min)) * 60;
959 m_gpsTags.insert(GPS_LATITUDEREF, degree < 0 ? QStringLiteral(
"S") : QStringLiteral(
"N"));
960 m_gpsTags.insert(GPS_LATITUDE,
QVariant::fromValue(QList<double>() <<
int(adeg) <<
int(min) << sec));
963double MicroExif::longitude()
const
965 auto ref = gpsString(GPS_LONGITUDEREF).toUpper();
966 if (ref != QStringLiteral(
"E") && ref != QStringLiteral(
"W"))
968 auto lon = m_gpsTags.value(GPS_LONGITUDE).value<QList<double>>();
971 auto degree = lon.at(0) + lon.at(1) / 60 + lon.at(2) / 3600;
972 if (degree < -180.0 || degree > 180.0)
974 return ref == QStringLiteral(
"E") ? degree : -degree;
977void MicroExif::setLongitude(
double degree)
979 if (qIsNaN(degree)) {
980 m_gpsTags.remove(GPS_LONGITUDEREF);
981 m_gpsTags.remove(GPS_LONGITUDE);
983 if (degree < -180.0 || degree > 180.0)
985 auto adeg = qAbs(degree);
986 auto min = (adeg - int(adeg)) * 60;
987 auto sec = (min - int(min)) * 60;
988 m_gpsTags.insert(GPS_LONGITUDEREF, degree < 0 ? QStringLiteral(
"W") : QStringLiteral(
"E"));
989 m_gpsTags.insert(GPS_LONGITUDE,
QVariant::fromValue(QList<double>() <<
int(adeg) <<
int(min) << sec));
992double MicroExif::altitude()
const
994 auto ref = m_gpsTags.value(GPS_ALTITUDEREF);
997 if (!m_gpsTags.contains(GPS_ALTITUDE))
999 auto alt = m_gpsTags.value(GPS_ALTITUDE).toDouble();
1000 return (ref.toInt() == 0 || ref.toInt() == 2) ? alt : -alt;
1003void MicroExif::setAltitude(
double meters)
1005 if (qIsNaN(meters)) {
1006 m_gpsTags.remove(GPS_ALTITUDEREF);
1007 m_gpsTags.remove(GPS_ALTITUDE);
1009 m_gpsTags.insert(GPS_ALTITUDEREF, quint8(meters < 0 ? 1 : 0));
1010 m_gpsTags.insert(GPS_ALTITUDE, meters);
1013double MicroExif::imageDirection(
bool *isMagnetic)
const
1016 if (isMagnetic ==
nullptr)
1018 if (!m_gpsTags.contains(GPS_IMGDIRECTION))
1020 auto ref = gpsString(GPS_IMGDIRECTIONREF).toUpper();
1021 *isMagnetic = (ref == QStringLiteral(
"M"));
1022 return m_gpsTags.value(GPS_IMGDIRECTION).toDouble();
1025void MicroExif::setImageDirection(
double degree,
bool isMagnetic)
1027 if (qIsNaN(degree)) {
1028 m_gpsTags.remove(GPS_IMGDIRECTIONREF);
1029 m_gpsTags.remove(GPS_IMGDIRECTION);
1031 m_gpsTags.insert(GPS_IMGDIRECTIONREF, isMagnetic ? QStringLiteral(
"M") : QStringLiteral(
"T"));
1032 m_gpsTags.insert(GPS_IMGDIRECTION, degree);
1040 if (!write(&buf, byteOrder))
1052 auto exifTags = m_exifTags;
1053 exifTags.insert(EXIF_EXIFVERSION, QByteArray(
"0300"));
1055 if (!writeIfd(ds, exifTags, positions))
1065 return readIfd(ds, m_exifTags, 0, staticTagTypes);
1074 auto gpsTags = m_gpsTags;
1075 gpsTags.insert(GPS_GPSVERSION, QByteArray(
"2400"));
1077 if (!writeIfd(ds, gpsTags, positions, 0, staticGpsTagTypes))
1087 return readIfd(ds, m_gpsTags, 0, staticGpsTagTypes);
1092 if (device ==
nullptr || device->
isSequential() || isEmpty())
1095 QDataStream ds(device);
1097 if (!writeHeader(ds))
1106void MicroExif::updateImageMetadata(
QImage &targetImage,
bool replaceExisting)
const
1109 for (
auto &&p : tiffStrMap) {
1110 if (!replaceExisting && !targetImage.
text(p.second).
isEmpty())
1112 auto s = tiffString(p.first);
1114 targetImage.
setText(p.second, s);
1118 for (
auto &&p : exifStrMap) {
1119 if (!replaceExisting && !targetImage.
text(p.second).
isEmpty())
1121 auto s = exifString(p.first);
1123 targetImage.
setText(p.second, s);
1127 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_MODIFICATIONDATE)).isEmpty()) {
1128 auto dt = dateTime();
1132 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_CREATIONDATE)).isEmpty()) {
1133 auto dt = dateTimeOriginal();
1139 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_ALTITUDE)).isEmpty()) {
1140 auto v = altitude();
1142 targetImage.
setText(QStringLiteral(META_KEY_ALTITUDE), QStringLiteral(
"%1").arg(v, 0,
'g', 9));
1144 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_LATITUDE)).isEmpty()) {
1145 auto v = latitude();
1147 targetImage.
setText(QStringLiteral(META_KEY_LATITUDE), QStringLiteral(
"%1").arg(v, 0,
'g', 9));
1149 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_LONGITUDE)).isEmpty()) {
1150 auto v = longitude();
1152 targetImage.
setText(QStringLiteral(META_KEY_LONGITUDE), QStringLiteral(
"%1").arg(v, 0,
'g', 9));
1154 if (replaceExisting || targetImage.
text(QStringLiteral(META_KEY_DIRECTION)).isEmpty()) {
1155 auto v = imageDirection();
1157 targetImage.
setText(QStringLiteral(META_KEY_DIRECTION), QStringLiteral(
"%1").arg(v, 0,
'g', 9));
1161void MicroExif::updateImageResolution(
QImage &targetImage)
1163 if (horizontalResolution() > 0)
1164 targetImage.
setDotsPerMeterX(qRound(horizontalResolution() / 25.4 * 1000));
1165 if (verticalResolution() > 0)
1169MicroExif MicroExif::fromByteArray(
const QByteArray &ba,
bool searchHeader)
1173 auto idxLE = ba0.indexOf(QByteArray(
"II"));
1174 auto idxBE = ba0.indexOf(QByteArray(
"MM"));
1176 if (idxLE > -1 && idxBE > -1)
1177 idx = std::min(idxLE, idxBE);
1179 idx = idxLE > -1 ? idxLE : idxBE;
1185 return fromDevice(&buf);
1188MicroExif MicroExif::fromRawData(
const char *data,
size_t size,
bool searchHeader)
1190 if (data ==
nullptr || size == 0)
1195MicroExif MicroExif::fromDevice(
QIODevice *device)
1202 QDataStream ds(device);
1203 if (!checkHeader(ds))
1209 if (!readIfd(ds, exif.m_tiffTags))
1213 if (
auto pos = exif.m_tiffTags.value(EXIF_EXIFIFD).toUInt()) {
1214 if (!readIfd(ds, exif.m_exifTags, pos))
1219 if (
auto pos = exif.m_tiffTags.value(EXIF_GPSIFD).toUInt()) {
1220 if (!readIfd(ds, exif.m_gpsTags, pos, staticGpsTagTypes))
1227MicroExif MicroExif::fromImage(
const QImage &image)
1234 exif.setWidth(image.
width());
1235 exif.setHeight(image.
height());
1236 exif.setHorizontalResolution(image.
dotsPerMeterX() * 25.4 / 1000);
1237 exif.setVerticalResolution(image.
dotsPerMeterY() * 25.4 / 1000);
1241 for (
auto &&p : tiffStrMap) {
1242 exif.setTiffString(p.first, image.
text(p.second));
1246 for (
auto &&p : exifStrMap) {
1247 exif.setExifString(p.first, image.
text(p.second));
1251 if (exif.software().isEmpty()) {
1254 if (!sw.isEmpty() && !ver.isEmpty())
1255 sw.append(QStringLiteral(
" %1").arg(ver));
1256 exif.setSoftware(sw.trimmed());
1263 exif.setDateTime(dt);
1269 exif.setDateTimeOriginal(dt);
1273 auto alt = image.
text(QStringLiteral(META_KEY_ALTITUDE)).
toDouble(&ok);
1275 exif.setAltitude(alt);
1276 auto lat = image.
text(QStringLiteral(META_KEY_LATITUDE)).
toDouble(&ok);
1278 exif.setLatitude(lat);
1279 auto lon = image.
text(QStringLiteral(META_KEY_LONGITUDE)).
toDouble(&ok);
1281 exif.setLongitude(lon);
1282 auto dir = image.
text(QStringLiteral(META_KEY_DIRECTION)).
toDouble(&ok);
1284 exif.setImageDirection(dir);
1289void MicroExif::setTiffString(quint16 tagId,
const QString &s)
1291 MicroExif::setString(m_tiffTags, tagId, s);
1294QString MicroExif::tiffString(quint16 tagId)
const
1296 return MicroExif::string(m_tiffTags, tagId);
1299void MicroExif::setExifString(quint16 tagId,
const QString &s)
1301 MicroExif::setString(m_exifTags, tagId, s);
1304QString MicroExif::exifString(quint16 tagId)
const
1306 return MicroExif::string(m_exifTags, tagId);
1309void MicroExif::setGpsString(quint16 tagId,
const QString &s)
1311 MicroExif::setString(m_gpsTags, tagId, s);
1314QString MicroExif::gpsString(quint16 tagId)
const
1316 return MicroExif::string(m_gpsTags, tagId);
1322 ds << quint16(0x4949);
1324 ds << quint16(0x4d4d);
1325 ds << quint16(0x002a);
1332 auto tiffTags = m_tiffTags;
1333 auto exifTags = m_exifTags;
1334 auto gpsTags = m_gpsTags;
1335 updateTags(tiffTags, exifTags, gpsTags);
1338 if (!writeIfd(ds, tiffTags, positions))
1340 if (!writeIfd(ds, exifTags, positions, positions.
value(EXIF_EXIFIFD)))
1342 if (!writeIfd(ds, gpsTags, positions, positions.
value(EXIF_GPSIFD), staticGpsTagTypes))
1347void MicroExif::updateTags(Tags &tiffTags, Tags &exifTags, Tags &gpsTags)
const
1349 if (exifTags.isEmpty()) {
1350 tiffTags.remove(EXIF_EXIFIFD);
1352 tiffTags.insert(EXIF_EXIFIFD, quint32());
1353 exifTags.insert(EXIF_EXIFVERSION, QByteArray(
"0300"));
1355 if (gpsTags.isEmpty()) {
1356 tiffTags.remove(EXIF_GPSIFD);
1358 tiffTags.insert(EXIF_GPSIFD, quint32());
1359 gpsTags.insert(GPS_GPSVERSION, QByteArray(
"2400"));
1363void MicroExif::setString(Tags &tags, quint16 tagId,
const QString &s)
1368 tags.insert(tagId, s);
1371QString MicroExif::string(
const Tags &tags, quint16 tagId)
1373 return tags.value(tagId).toString();
KDB_EXPORT KDbVersionInfo version()
KIOCORE_EXPORT QString dir(const QString &fileClass)
void setData(const QByteArray &data)
QByteArray & append(QByteArrayView data)
char at(qsizetype i) const const
QByteArray first(qsizetype n) const const
QByteArray fromRawData(const char *data, qsizetype size)
bool isEmpty() const const
QByteArray & removeLast()
qsizetype size() const const
Primaries primaries() const const
TransferFunction transferFunction() const const
ByteOrder byteOrder() const const
QIODevice * device() const const
void setByteOrder(ByteOrder bo)
Status status() const const
QDateTime currentDateTime()
QDateTime fromString(QStringView string, QStringView format, QCalendar cal)
bool isValid() const const
int offsetFromUtc() const const
void setTimeZone(const QTimeZone &toZone)
QString toString(QStringView format, QCalendar cal) const const
iterator insert(const Key &key, const T &value)
bool remove(const Key &key)
T value(const Key &key) const const
QColorSpace colorSpace() const const
int dotsPerMeterX() const const
int dotsPerMeterY() const const
bool isNull() const const
void setDotsPerMeterX(int x)
void setDotsPerMeterY(int y)
void setText(const QString &key, const QString &text)
QString text(const QString &key) const const
virtual bool isSequential() const const
virtual bool open(QIODeviceBase::OpenMode mode)
virtual qint64 pos() const const
void append(QList< T > &&value)
const QChar at(qsizetype position) const const
QString fromLatin1(QByteArrayView str)
QString fromUtf8(QByteArrayView str)
bool isEmpty() const const
QString left(qsizetype n) const const
QString mid(qsizetype position, qsizetype n) const const
QString & replace(QChar before, QChar after, Qt::CaseSensitivity cs)
qsizetype size() const const
double toDouble(bool *ok) const const
int toInt(bool *ok, int base) const const
QByteArray toLatin1() const const
QByteArray toUtf8() const const
QTimeZone fromSecondsAheadOfUtc(int offset)
bool isNull() const const
QString toString(StringFormat mode) const const
QVariant fromValue(T &&value)
QByteArray toByteArray() const const
double toDouble(bool *ok) const const
int toInt(bool *ok) const const
QString toString() const const