You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1628 lines
69 KiB
C++

// ***************************************************************** -*- C++ -*-
/*
* Copyright (C) 2004-2021 Exiv2 authors
* This program is part of the Exiv2 distribution.
*
* This program is free 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 2
* of the License, or (at your option) any later version.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, 5th Floor, Boston, MA 02110-1301 USA.
*/
/*
File: convert.cpp
Author(s): Andreas Huggel (ahu) <ahuggel@gmx.net>
Vladimir Nadvornik (vn) <nadvornik@suse.cz>
History: 17-Mar-08, ahu: created basic converter framework
20-May-08, vn: added actual conversion logic
*/
// *****************************************************************************
// included header files
#include "config.h"
#include "types.hpp"
#include "error.hpp"
#include "exif.hpp"
#include "iptc.hpp"
#include "xmp_exiv2.hpp"
#include "futils.hpp"
#include "convert.hpp"
#include "unused.h"
// + standard includes
#include <stdio.h> // for snprintf (C99)
#include <algorithm>
#include <cstring>
#include <iomanip>
#include <ios>
#include <iostream>
#include <sstream>
#include <utility>
#if defined WIN32 && !defined __CYGWIN__
# include <windows.h>
#endif
#ifdef EXV_HAVE_ICONV
# include <iconv.h>
# include <errno.h>
#endif
// Adobe XMP Toolkit
#ifdef EXV_HAVE_XMP_TOOLKIT
# define TXMP_STRING_TYPE std::string
# ifdef EXV_ADOBE_XMPSDK
# include <XMP.hpp>
# else
# include <XMPSDK.hpp>
# endif
# include <MD5.h>
#endif // EXV_HAVE_XMP_TOOLKIT
// *****************************************************************************
// local declarations
namespace {
#if defined WIN32 && !defined __CYGWIN__
// Convert string charset with Windows functions.
bool convertStringCharsetWindows(std::string& str, const char* from, const char* to);
#endif
#if defined EXV_HAVE_ICONV
// Convert string charset with iconv.
bool convertStringCharsetIconv(std::string& str, const char* from, const char* to);
#endif
/*!
@brief Get the text value of an XmpDatum \em pos.
If \em pos refers to a LangAltValue, \em value is set to the default language
entry without the x-default qualifier. If there is no default but
exactly one entry, \em value is set to this entry, without the qualifier.
The return code indicates if the operation was successful.
*/
bool getTextValue(std::string& value, const Exiv2::XmpData::iterator& pos);
} // namespace
// *****************************************************************************
// class member definitions
namespace Exiv2 {
//! Metadata conversions.
class Converter {
public:
/*!
@brief Type for metadata converter functions, taking two key strings,
\em from and \em to.
These functions have access to both the source and destination metadata
containers and store the result directly in the destination container.
*/
using ConvertFct = void (Converter::*)(const char*, const char*);
//! Structure to define conversions between two keys.
struct Conversion {
MetadataId metadataId_; //!< Type of metadata for the first key.
const char* key1_; //!< First metadata key.
const char* key2_; //!< Second metadata key (always an XMP key for now).
ConvertFct key1ToKey2_; //!< Conversion from first to second key.
ConvertFct key2ToKey1_; //!< Conversion from second to first key.
};
//! @name Creators
//@{
//! Constructor for Exif tags and XMP properties.
Converter(ExifData& exifData, XmpData& xmpData);
//! Constructor for Iptc tags and XMP properties.
Converter(IptcData& iptcData, XmpData& xmpData, const char* iptcCharset = nullptr);
//@}
//! @name Manipulators
//@{
//! Convert Exif tags or IPTC datasets to XMP properties according to the conversion table.
void cnvToXmp();
//! Convert XMP properties to Exif tags or IPTC datasets according to the conversion table.
void cnvFromXmp();
/*!
@brief Set the erase flag.
This flag indicates whether successfully converted source records are erased.
*/
void setErase(bool onoff =true) { erase_ = onoff; }
/*!
@brief Set the overwrite flag.
This flag indicates whether existing target records are overwritten.
*/
void setOverwrite(bool onoff =true) { overwrite_ = onoff; }
//@}
//! @name Conversion functions (manipulators)
//@{
/*!
@brief Do nothing conversion function.
Use when, for example, a one-way conversion is needed.
*/
void cnvNone(const char*, const char*);
/*!
@brief Simple Exif to XMP conversion function.
Sets the XMP property to an XmpText value containing the Exif value string.
*/
void cnvExifValue(const char* from, const char* to);
/*!
@brief Convert the tag Exif.Photo.UserComment to XMP.
Todo: Convert the Exif comment to UTF-8 if necessary.
*/
void cnvExifComment(const char* from, const char* to);
/*!
@brief Converts Exif tag with multiple components to XMP array.
Converts Exif tag with multiple components to XMP array. This function is
used for ComponentsConfiguration tag.
*/
void cnvExifArray(const char* from, const char* to);
/*!
@brief Exif date to XMP conversion function.
Sets the XMP property to an XmpText value containing date and time. This function
combines values from multiple Exif tags as described in XMP specification. It
is used for DateTime, DateTimeOriginal, DateTimeDigitized and GPSTimeStamp.
*/
void cnvExifDate(const char* from, const char* to);
/*!
@brief Exif version to XMP conversion function.
Converts ExifVersion tag to XmpText value.
*/
void cnvExifVersion(const char* from, const char* to);
/*!
@brief Exif GPS version to XMP conversion function.
Converts GPSVersionID tag to XmpText value.
*/
void cnvExifGPSVersion(const char* from, const char* to);
/*!
@brief Exif Flash to XMP conversion function.
Converts Flash tag to XMP structure.
*/
void cnvExifFlash(const char* from, const char* to);
/*!
@brief Exif GPS coordinate to XMP conversion function.
Converts GPS coordinates tag to XmpText value. It combines multiple Exif tags
as described in XMP specification.
*/
void cnvExifGPSCoord(const char* from, const char* to);
/*!
@brief Simple XMP to Exif conversion function.
Sets the Exif tag according to the XMP property.
For LangAlt values, only the x-default entry is used.
Todo: Escape non-ASCII characters in XMP text values
*/
void cnvXmpValue(const char* from, const char* to);
/*!
@brief Convert the tag Xmp.exif.UserComment to Exif.
*/
void cnvXmpComment(const char* from, const char* to);
/*!
@brief Converts XMP array to Exif tag with multiple components.
Converts XMP array to Exif tag with multiple components. This function is
used for ComponentsConfiguration tag.
*/
void cnvXmpArray(const char* from, const char* to);
/*!
@brief XMP to Exif date conversion function.
Converts the XmpText value to Exif date and time. This function
sets multiple Exif tags as described in XMP specification. It
is used for DateTime, DateTimeOriginal, DateTimeDigitized and GPSTimeStamp.
*/
void cnvXmpDate(const char* from, const char* to);
/*!
@brief XMP to Exif version conversion function.
Converts XmpText value to ExifVersion tag.
*/
void cnvXmpVersion(const char* from, const char* to);
/*!
@brief XMP to Exif GPS version conversion function.
Converts XmpText value to GPSVersionID tag.
*/
void cnvXmpGPSVersion(const char* from, const char* to);
/*!
@brief XMP to Exif Flash conversion function.
Converts XMP structure to Flash tag.
*/
void cnvXmpFlash(const char* from, const char* to);
/*!
@brief XMP to Exif GPS coordinate conversion function.
Converts XmpText value to GPS coordinates tags. It sets multiple Exif tags
as described in XMP specification.
*/
void cnvXmpGPSCoord(const char* from, const char* to);
/*!
@brief IPTC dataset to XMP conversion function.
Multiple IPTC datasets with the same key are converted to an XMP array.
*/
void cnvIptcValue(const char* from, const char* to);
/*!
@brief XMP to IPTC dataset conversion function.
Each array element of an XMP array value is added as one IPTC dataset.
*/
void cnvXmpValueToIptc(const char* from, const char* to);
/*!
@brief Write exif:NativeDigest and tiff:NativeDigest properties to XMP.
Compute digests from Exif values and write them to exif:NativeDigest
and tiff:NativeDigest properties. This should be compatible with XMP SDK.
*/
void writeExifDigest();
/*!
@brief Copies metadata in appropriate direction.
From values of exif:NativeDigest and tiff:NativeDigest detects which of
XMP and Exif was updated more recently and copies metadata in appropriate direction.
*/
void syncExifWithXmp();
//@}
//! @name Accessors
//@{
//! Get the value of the erase flag, see also setErase(bool on).
bool erase() const { return erase_; }
//@}
private:
bool prepareExifTarget(const char* to, bool force =false);
bool prepareIptcTarget(const char* to, bool force =false);
bool prepareXmpTarget(const char* to, bool force =false);
std::string computeExifDigest(bool tiff);
// DATA
static const Conversion conversion_[]; //<! Conversion rules
bool erase_;
bool overwrite_;
ExifData *exifData_;
IptcData *iptcData_;
XmpData *xmpData_;
const char *iptcCharset_;
}; // class Converter
// Order is important for computing digests
const Converter::Conversion Converter::conversion_[] = {
{ mdExif, "Exif.Image.ImageWidth", "Xmp.tiff.ImageWidth", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.ImageLength", "Xmp.tiff.ImageLength", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.BitsPerSample", "Xmp.tiff.BitsPerSample", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Compression", "Xmp.tiff.Compression", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.PhotometricInterpretation", "Xmp.tiff.PhotometricInterpretation", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Orientation", "Xmp.tiff.Orientation", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.SamplesPerPixel", "Xmp.tiff.SamplesPerPixel", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.PlanarConfiguration", "Xmp.tiff.PlanarConfiguration", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.YCbCrSubSampling", "Xmp.tiff.YCbCrSubSampling", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.YCbCrPositioning", "Xmp.tiff.YCbCrPositioning", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.XResolution", "Xmp.tiff.XResolution", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.YResolution", "Xmp.tiff.YResolution", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.ResolutionUnit", "Xmp.tiff.ResolutionUnit", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.TransferFunction", "Xmp.tiff.TransferFunction", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.WhitePoint", "Xmp.tiff.WhitePoint", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.PrimaryChromaticities", "Xmp.tiff.PrimaryChromaticities", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.YCbCrCoefficients", "Xmp.tiff.YCbCrCoefficients", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.ReferenceBlackWhite", "Xmp.tiff.ReferenceBlackWhite", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.DateTime", "Xmp.xmp.ModifyDate", &Converter::cnvExifDate , &Converter::cnvXmpDate }, // MWG Guidelines
{ mdExif, "Exif.Image.ImageDescription", "Xmp.dc.description", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Make", "Xmp.tiff.Make", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Model", "Xmp.tiff.Model", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Software", "Xmp.tiff.Software", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Artist", "Xmp.dc.creator", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Rating", "Xmp.xmp.Rating", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Image.Copyright", "Xmp.dc.rights", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ExifVersion", "Xmp.exif.ExifVersion", &Converter::cnvExifVersion, &Converter::cnvXmpVersion },
{ mdExif, "Exif.Photo.FlashpixVersion", "Xmp.exif.FlashpixVersion", &Converter::cnvExifVersion, &Converter::cnvXmpVersion },
{ mdExif, "Exif.Photo.ColorSpace", "Xmp.exif.ColorSpace", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ComponentsConfiguration", "Xmp.exif.ComponentsConfiguration", &Converter::cnvExifArray, &Converter::cnvXmpArray },
{ mdExif, "Exif.Photo.CompressedBitsPerPixel", "Xmp.exif.CompressedBitsPerPixel", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.PixelXDimension", "Xmp.exif.PixelXDimension", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.PixelYDimension", "Xmp.exif.PixelYDimension", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.UserComment", "Xmp.exif.UserComment", &Converter::cnvExifComment, &Converter::cnvXmpComment },
{ mdExif, "Exif.Photo.RelatedSoundFile", "Xmp.exif.RelatedSoundFile", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.DateTimeOriginal", "Xmp.photoshop.DateCreated", &Converter::cnvExifDate, &Converter::cnvXmpDate }, // MWG Guidelines
{ mdExif, "Exif.Photo.DateTimeDigitized", "Xmp.xmp.CreateDate", &Converter::cnvExifDate, &Converter::cnvXmpDate }, // MWG Guidelines
{ mdExif, "Exif.Photo.ExposureTime", "Xmp.exif.ExposureTime", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FNumber", "Xmp.exif.FNumber", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ExposureProgram", "Xmp.exif.ExposureProgram", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SpectralSensitivity", "Xmp.exif.SpectralSensitivity", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ISOSpeedRatings", "Xmp.exif.ISOSpeedRatings", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.OECF", "Xmp.exif.OECF", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.ShutterSpeedValue", "Xmp.exif.ShutterSpeedValue", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ApertureValue", "Xmp.exif.ApertureValue", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.BrightnessValue", "Xmp.exif.BrightnessValue", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ExposureBiasValue", "Xmp.exif.ExposureBiasValue", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.MaxApertureValue", "Xmp.exif.MaxApertureValue", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SubjectDistance", "Xmp.exif.SubjectDistance", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.MeteringMode", "Xmp.exif.MeteringMode", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.LightSource", "Xmp.exif.LightSource", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.Flash", "Xmp.exif.Flash", &Converter::cnvExifFlash, &Converter::cnvXmpFlash },
{ mdExif, "Exif.Photo.FocalLength", "Xmp.exif.FocalLength", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SubjectArea", "Xmp.exif.SubjectArea", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FlashEnergy", "Xmp.exif.FlashEnergy", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SpatialFrequencyResponse", "Xmp.exif.SpatialFrequencyResponse", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.FocalPlaneXResolution", "Xmp.exif.FocalPlaneXResolution", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FocalPlaneYResolution", "Xmp.exif.FocalPlaneYResolution", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FocalPlaneResolutionUnit", "Xmp.exif.FocalPlaneResolutionUnit", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SubjectLocation", "Xmp.exif.SubjectLocation", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ExposureIndex", "Xmp.exif.ExposureIndex", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SensingMethod", "Xmp.exif.SensingMethod", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FileSource", "Xmp.exif.FileSource", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.SceneType", "Xmp.exif.SceneType", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.CFAPattern", "Xmp.exif.CFAPattern", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.CustomRendered", "Xmp.exif.CustomRendered", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ExposureMode", "Xmp.exif.ExposureMode", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.WhiteBalance", "Xmp.exif.WhiteBalance", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.DigitalZoomRatio", "Xmp.exif.DigitalZoomRatio", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.FocalLengthIn35mmFilm", "Xmp.exif.FocalLengthIn35mmFilm", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.SceneCaptureType", "Xmp.exif.SceneCaptureType", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.GainControl", "Xmp.exif.GainControl", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.Contrast", "Xmp.exif.Contrast", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.Saturation", "Xmp.exif.Saturation", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.Sharpness", "Xmp.exif.Sharpness", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.DeviceSettingDescription", "Xmp.exif.DeviceSettingDescription", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.Photo.SubjectDistanceRange", "Xmp.exif.SubjectDistanceRange", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.Photo.ImageUniqueID", "Xmp.exif.ImageUniqueID", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSVersionID", "Xmp.exif.GPSVersionID", &Converter::cnvExifGPSVersion, &Converter::cnvXmpGPSVersion },
{ mdExif, "Exif.GPSInfo.GPSLatitude", "Xmp.exif.GPSLatitude", &Converter::cnvExifGPSCoord, &Converter::cnvXmpGPSCoord },
{ mdExif, "Exif.GPSInfo.GPSLongitude", "Xmp.exif.GPSLongitude", &Converter::cnvExifGPSCoord, &Converter::cnvXmpGPSCoord },
{ mdExif, "Exif.GPSInfo.GPSAltitudeRef", "Xmp.exif.GPSAltitudeRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSAltitude", "Xmp.exif.GPSAltitude", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSTimeStamp", "Xmp.exif.GPSTimeStamp", &Converter::cnvExifDate, &Converter::cnvXmpDate }, // FIXME ?
{ mdExif, "Exif.GPSInfo.GPSSatellites", "Xmp.exif.GPSSatellites", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSStatus", "Xmp.exif.GPSStatus", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSMeasureMode", "Xmp.exif.GPSMeasureMode", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSDOP", "Xmp.exif.GPSDOP", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSSpeedRef", "Xmp.exif.GPSSpeedRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSSpeed", "Xmp.exif.GPSSpeed", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSTrackRef", "Xmp.exif.GPSTrackRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSTrack", "Xmp.exif.GPSTrack", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSImgDirectionRef", "Xmp.exif.GPSImgDirectionRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSImgDirection", "Xmp.exif.GPSImgDirection", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSMapDatum", "Xmp.exif.GPSMapDatum", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSDestLatitude", "Xmp.exif.GPSDestLatitude", &Converter::cnvExifGPSCoord, &Converter::cnvXmpGPSCoord },
{ mdExif, "Exif.GPSInfo.GPSDestLongitude", "Xmp.exif.GPSDestLongitude", &Converter::cnvExifGPSCoord, &Converter::cnvXmpGPSCoord },
{ mdExif, "Exif.GPSInfo.GPSDestBearingRef", "Xmp.exif.GPSDestBearingRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSDestBearing", "Xmp.exif.GPSDestBearing", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSDestDistanceRef", "Xmp.exif.GPSDestDistanceRef", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSDestDistance", "Xmp.exif.GPSDestDistance", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdExif, "Exif.GPSInfo.GPSProcessingMethod", "Xmp.exif.GPSProcessingMethod", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.GPSInfo.GPSAreaInformation", "Xmp.exif.GPSAreaInformation", &Converter::cnvExifValue, &Converter::cnvXmpValue }, // FIXME ?
{ mdExif, "Exif.GPSInfo.GPSDifferential", "Xmp.exif.GPSDifferential", &Converter::cnvExifValue, &Converter::cnvXmpValue },
{ mdIptc, "Iptc.Application2.ObjectName", "Xmp.dc.title", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Urgency", "Xmp.photoshop.Urgency", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Category", "Xmp.photoshop.Category", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.SuppCategory", "Xmp.photoshop.SupplementalCategories", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Keywords", "Xmp.dc.subject", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.SubLocation", "Xmp.iptc.Location", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.SpecialInstructions","Xmp.photoshop.Instructions", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.DateCreated", "Xmp.photoshop.DateCreated", &Converter::cnvNone, &Converter::cnvXmpValueToIptc }, // FIXME to IPTC Date and IPTC Time
{ mdIptc, "Iptc.Application2.DigitizationDate", "Xmp.xmp.CreateDate", &Converter::cnvNone, &Converter::cnvXmpValueToIptc }, // FIXME to IPTC Date and IPTC Time
{ mdIptc, "Iptc.Application2.Byline", "Xmp.dc.creator", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.BylineTitle", "Xmp.photoshop.AuthorsPosition", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.City", "Xmp.photoshop.City", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.ProvinceState", "Xmp.photoshop.State", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.CountryCode", "Xmp.iptc.CountryCode", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.CountryName", "Xmp.photoshop.Country", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.TransmissionReference", "Xmp.photoshop.TransmissionReference", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Headline", "Xmp.photoshop.Headline", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Credit", "Xmp.photoshop.Credit", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Source", "Xmp.photoshop.Source", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Copyright", "Xmp.dc.rights", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Caption", "Xmp.dc.description", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc },
{ mdIptc, "Iptc.Application2.Writer", "Xmp.photoshop.CaptionWriter", &Converter::cnvIptcValue, &Converter::cnvXmpValueToIptc }
};
Converter::Converter(ExifData& exifData, XmpData& xmpData)
: erase_(false),
overwrite_(true),
exifData_(&exifData),
iptcData_(nullptr),
xmpData_(&xmpData),
iptcCharset_(nullptr)
{
}
Converter::Converter(IptcData& iptcData, XmpData& xmpData, const char* iptcCharset)
: erase_(false),
overwrite_(true),
exifData_(nullptr),
iptcData_(&iptcData),
xmpData_(&xmpData),
iptcCharset_(iptcCharset)
{
}
void Converter::cnvToXmp()
{
for (auto&& c : conversion_) {
if ( (c.metadataId_ == mdExif && exifData_)
|| (c.metadataId_ == mdIptc && iptcData_)) {
EXV_CALL_MEMBER_FN(*this, c.key1ToKey2_)(c.key1_, c.key2_);
}
}
}
void Converter::cnvFromXmp()
{
for (auto&& c : conversion_) {
if ( (c.metadataId_ == mdExif && exifData_)
|| (c.metadataId_ == mdIptc && iptcData_)) {
EXV_CALL_MEMBER_FN(*this, c.key2ToKey1_)(c.key2_, c.key1_);
}
}
}
void Converter::cnvNone(const char*, const char*)
{
}
bool Converter::prepareExifTarget(const char* to, bool force)
{
auto pos = exifData_->findKey(ExifKey(to));
if (pos == exifData_->end())
return true;
if (!overwrite_ && !force)
return false;
exifData_->erase(pos);
return true;
}
bool Converter::prepareIptcTarget(const char* to, bool force)
{
auto pos = iptcData_->findKey(IptcKey(to));
if (pos == iptcData_->end())
return true;
if (!overwrite_ && !force)
return false;
while ((pos = iptcData_->findKey(IptcKey(to))) != iptcData_->end()) {
iptcData_->erase(pos);
}
return true;
}
bool Converter::prepareXmpTarget(const char* to, bool force)
{
auto pos = xmpData_->findKey(XmpKey(to));
if (pos == xmpData_->end())
return true;
if (!overwrite_ && !force)
return false;
xmpData_->erase(pos);
return true;
}
void Converter::cnvExifValue(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
if (!prepareXmpTarget(to))
return;
(*xmpData_)[to] = value;
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifComment(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
const auto cv = dynamic_cast<const CommentValue*>(&pos->value());
if (cv == nullptr) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
// Todo: Convert to UTF-8 if necessary
(*xmpData_)[to] = cv->comment();
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifArray(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
for (size_t i = 0; i < pos->count(); ++i) {
std::string value = pos->toString(static_cast<long>(i));
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
(*xmpData_)[to] = value;
}
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifDate(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
int year=0, month=0, day=0, hour=0, min=0, sec=0;
std::string subsec;
char buf[30];
if (std::string(from) != "Exif.GPSInfo.GPSTimeStamp") {
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
if (sscanf(value.c_str(), "%d:%d:%d %d:%d:%d", &year, &month, &day, &hour, &min, &sec) != 6) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to
<< ", unable to parse '" << value << "'\n";
#endif
return;
}
}
else { // "Exif.GPSInfo.GPSTimeStamp"
bool ok = true;
if (pos->count() != 3) ok = false;
if (ok) {
for (int i = 0; i < 3; ++i) {
if (pos->toRational(i).second == 0) {
ok = false;
break;
}
}
}
if (!ok) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
double dhour = pos->toFloat(0);
double dmin = pos->toFloat(1);
// Hack: Need Value::toDouble
Rational r = pos->toRational(2);
double dsec = static_cast<double>(r.first)/r.second;
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
dsec = dhour * 3600.0 + dmin * 60.0 + dsec;
hour = static_cast<int>(dsec / 3600.0);
dsec -= hour * 3600;
min = static_cast<int>(dsec / 60.0);
dsec -= min * 60;
sec = static_cast<int>(dsec);
dsec -= sec;
snprintf(buf, sizeof(buf), "%.9f", dsec);
buf[sizeof(buf) - 1] = 0;
buf[1] = '.'; // some locales use ','
subsec = buf + 1;
auto datePos = exifData_->findKey(ExifKey("Exif.GPSInfo.GPSDateStamp"));
if (datePos == exifData_->end()) {
datePos = exifData_->findKey(ExifKey("Exif.Photo.DateTimeOriginal"));
}
if (datePos == exifData_->end()) {
datePos = exifData_->findKey(ExifKey("Exif.Photo.DateTimeDigitized"));
}
if (datePos == exifData_->end()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
std::string value = datePos->toString();
if (sscanf(value.c_str(), "%d:%d:%d", &year, &month, &day) != 3) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to
<< ", unable to parse '" << value << "'\n";
#endif
return;
}
}
const char* subsecTag = nullptr;
if (std::string(from) == "Exif.Image.DateTime") {
subsecTag = "Exif.Photo.SubSecTime";
}
else if (std::string(from) == "Exif.Photo.DateTimeOriginal") {
subsecTag = "Exif.Photo.SubSecTimeOriginal";
}
else if (std::string(from) == "Exif.Photo.DateTimeDigitized") {
subsecTag = "Exif.Photo.SubSecTimeDigitized";
}
if (subsecTag) {
auto subsec_pos = exifData_->findKey(ExifKey(subsecTag));
if (subsec_pos != exifData_->end()) {
if (subsec_pos->typeId() == asciiString) {
std::string ss = subsec_pos->toString();
if (!ss.empty()) {
bool ok = false;
stringTo<long>(ss, ok);
if (ok) subsec = std::string(".") + ss;
}
}
if (erase_) exifData_->erase(subsec_pos);
}
}
if (subsec.size() > 10) subsec = subsec.substr(0, 10);
snprintf(buf, sizeof(buf), "%4d-%02d-%02dT%02d:%02d:%02d%s",
year, month, day, hour, min, sec, subsec.c_str());
buf[sizeof(buf) - 1] = 0;
(*xmpData_)[to] = buf;
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifVersion(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
std::ostringstream value;
for (size_t i = 0; i < pos->count(); ++i) {
value << static_cast<char>(pos->toInt64(i));
}
(*xmpData_)[to] = value.str();
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifGPSVersion(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
std::ostringstream value;
for (size_t i = 0; i < pos->count(); ++i) {
if (i > 0) value << '.';
value << pos->toInt64(static_cast<long>(i));
}
(*xmpData_)[to] = value.str();
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifFlash(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end() || pos->count() == 0)
return;
if (!prepareXmpTarget(to))
return;
auto value = pos->toInt64();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
(*xmpData_)["Xmp.exif.Flash/exif:Fired"] = static_cast<bool>(value & 1);
(*xmpData_)["Xmp.exif.Flash/exif:Return"] = (value >> 1) & 3;
(*xmpData_)["Xmp.exif.Flash/exif:Mode"] = (value >> 3) & 3;
(*xmpData_)["Xmp.exif.Flash/exif:Function"] = static_cast<bool>((value >> 5) & 1);
(*xmpData_)["Xmp.exif.Flash/exif:RedEyeMode"] = static_cast<bool>((value >> 6) & 1);
if (erase_) exifData_->erase(pos);
}
void Converter::cnvExifGPSCoord(const char* from, const char* to)
{
auto pos = exifData_->findKey(ExifKey(from));
if (pos == exifData_->end())
return;
if (!prepareXmpTarget(to))
return;
if (pos->count() != 3) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
auto refPos = exifData_->findKey(ExifKey(std::string(from) + "Ref"));
if (refPos == exifData_->end()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
double deg[3];
for (int i = 0; i < 3; ++i) {
const int32_t z = pos->toRational(i).first;
const int32_t d = pos->toRational(i).second;
if (d == 0) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
// Hack: Need Value::toDouble
deg[i] = static_cast<double>(z)/d;
}
double min = deg[0] * 60.0 + deg[1] + deg[2] / 60.0;
int ideg = static_cast<int>(min / 60.0);
min -= ideg * 60;
std::ostringstream oss;
oss << ideg << ","
<< std::fixed << std::setprecision(7) << min
<< refPos->toString().c_str()[0];
(*xmpData_)[to] = oss.str();
if (erase_) exifData_->erase(pos);
if (erase_) exifData_->erase(refPos);
}
void Converter::cnvXmpValue(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
std::string value;
if (!getTextValue(value, pos)) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
// Todo: Escape non-ASCII characters in XMP text values
ExifKey key(to);
Exifdatum ed(key);
if (0 == ed.setValue(value)) {
exifData_->add(ed);
}
if (erase_) xmpData_->erase(pos);
}
void Converter::cnvXmpComment(const char* from, const char* to)
{
if (!prepareExifTarget(to))
return;
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
std::string value;
if (!getTextValue(value, pos)) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
// Assumes the XMP value is encoded in UTF-8, as it should be
(*exifData_)[to] = "charset=Unicode " + value;
if (erase_) xmpData_->erase(pos);
}
void Converter::cnvXmpArray(const char* from, const char* to)
{
if (!prepareExifTarget(to))
return;
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
std::ostringstream array;
for (size_t i = 0; i < pos->count(); ++i) {
std::string value = pos->toString(static_cast<long>(i));
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
array << value;
if (i != pos->count() - 1)
array << " ";
}
(*exifData_)[to] = array.str();
if (erase_)
xmpData_->erase(pos);
}
void Converter::cnvXmpDate(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
#ifdef EXV_HAVE_XMP_TOOLKIT
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
XMP_DateTime datetime;
try {
SXMPUtils::ConvertToDate(value, &datetime);
char buf[30];
if (std::string(to) != "Exif.GPSInfo.GPSTimeStamp") {
SXMPUtils::ConvertToLocalTime(&datetime);
snprintf(buf, sizeof(buf), "%4d:%02d:%02d %02d:%02d:%02d",
static_cast<int>(datetime.year),
static_cast<int>(datetime.month),
static_cast<int>(datetime.day),
static_cast<int>(datetime.hour),
static_cast<int>(datetime.minute),
static_cast<int>(datetime.second));
buf[sizeof(buf) - 1] = 0;
(*exifData_)[to] = buf;
if (datetime.nanoSecond) {
const char* subsecTag = nullptr;
if (std::string(to) == "Exif.Image.DateTime") {
subsecTag = "Exif.Photo.SubSecTime";
}
else if (std::string(to) == "Exif.Photo.DateTimeOriginal") {
subsecTag = "Exif.Photo.SubSecTimeOriginal";
}
else if (std::string(to) == "Exif.Photo.DateTimeDigitized") {
subsecTag = "Exif.Photo.SubSecTimeDigitized";
}
if (subsecTag) {
prepareExifTarget(subsecTag, true);
(*exifData_)[subsecTag] = toString(datetime.nanoSecond);
}
}
}
else { // "Exif.GPSInfo.GPSTimeStamp"
// Ignore the time zone, assuming the time is in UTC as it should be
URational rhour(datetime.hour, 1);
URational rmin(datetime.minute, 1);
URational rsec(datetime.second, 1);
if (datetime.nanoSecond != 0) {
if (datetime.second != 0) {
// Add the seconds to rmin so that the ns fit into rsec
rmin.second = 60;
rmin.first *= 60;
rmin.first += datetime.second;
}
rsec.second = 1000000000;
rsec.first = datetime.nanoSecond;
}
std::ostringstream array;
array << rhour << " " << rmin << " " << rsec;
(*exifData_)[to] = array.str();
prepareExifTarget("Exif.GPSInfo.GPSDateStamp", true);
snprintf(buf, sizeof(buf), "%4d:%02d:%02d",
static_cast<int>(datetime.year),
static_cast<int>(datetime.month),
static_cast<int>(datetime.day));
buf[sizeof(buf) - 1] = 0;
(*exifData_)["Exif.GPSInfo.GPSDateStamp"] = buf;
}
}
#ifndef SUPPRESS_WARNINGS
catch (const XMP_Error& e) {
EXV_WARNING << "Failed to convert " << from << " to " << to << " (" << e.GetErrMsg() << ")\n";
return;
}
#else
catch (const XMP_Error&) {
return;
}
#endif // SUPPRESS_WARNINGS
if (erase_) xmpData_->erase(pos);
#else
# ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
# endif
#endif // !EXV_HAVE_XMP_TOOLKIT
}
void Converter::cnvXmpVersion(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
std::string value = pos->toString();
if (!pos->value().ok() || value.length() < 4) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
std::ostringstream array;
array << static_cast<int>(value[0]) << " "
<< static_cast<int>(value[1]) << " "
<< static_cast<int>(value[2]) << " "
<< static_cast<int>(value[3]);
(*exifData_)[to] = array.str();
if (erase_) xmpData_->erase(pos);
}
void Converter::cnvXmpGPSVersion(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
std::replace(value.begin(), value.end(), '.', ' ');
(*exifData_)[to] = value;
if (erase_) xmpData_->erase(pos);
}
void Converter::cnvXmpFlash(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(std::string(from) + "/exif:Fired"));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
unsigned short value = 0;
if (pos != xmpData_->end() && pos->count() > 0) {
auto fired = pos->toInt64();
if (pos->value().ok())
value |= fired & 1;
#ifndef SUPPRESS_WARNINGS
else
EXV_WARNING << "Failed to convert " << std::string(from) + "/exif:Fired" << " to " << to << "\n";
#endif
}
pos = xmpData_->findKey(XmpKey(std::string(from) + "/exif:Return"));
if (pos != xmpData_->end() && pos->count() > 0) {
auto ret = pos->toInt64();
if (pos->value().ok())
value |= (ret & 3) << 1;
#ifndef SUPPRESS_WARNINGS
else
EXV_WARNING << "Failed to convert " << std::string(from) + "/exif:Return" << " to " << to << "\n";
#endif
}
pos = xmpData_->findKey(XmpKey(std::string(from) + "/exif:Mode"));
if (pos != xmpData_->end() && pos->count() > 0) {
auto mode = pos->toInt64();
if (pos->value().ok())
value |= (mode & 3) << 3;
#ifndef SUPPRESS_WARNINGS
else
EXV_WARNING << "Failed to convert " << std::string(from) + "/exif:Mode" << " to " << to << "\n";
#endif
}
pos = xmpData_->findKey(XmpKey(std::string(from) + "/exif:Function"));
if (pos != xmpData_->end() && pos->count() > 0) {
auto function = pos->toInt64();
if (pos->value().ok())
value |= (function & 1) << 5;
#ifndef SUPPRESS_WARNINGS
else
EXV_WARNING << "Failed to convert " << std::string(from) + "/exif:Function" << " to " << to << "\n";
#endif
}
pos = xmpData_->findKey(XmpKey(std::string(from) + "/exif:RedEyeMode"));
if (pos != xmpData_->end()) {
if (pos->count() > 0) {
auto red = pos->toInt64();
if (pos->value().ok())
value |= (red & 1) << 6;
#ifndef SUPPRESS_WARNINGS
else
EXV_WARNING << "Failed to convert " << std::string(from) + "/exif:RedEyeMode" << " to " << to << "\n";
#endif
}
if (erase_) xmpData_->erase(pos);
}
(*exifData_)[to] = value;
}
void Converter::cnvXmpGPSCoord(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareExifTarget(to))
return;
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
if (value.empty()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << from << " is empty\n";
#endif
return;
}
double deg = 0.0;
double min = 0.0;
double sec = 0.0;
char ref = value[value.length() - 1];
char sep1 = '\0';
char sep2 = '\0';
value.erase(value.length() - 1);
std::istringstream in(value);
in >> deg >> sep1 >> min >> sep2;
if (sep2 == ',') {
in >> sec;
}
else {
sec = (min - static_cast<int>(min)) * 60.0;
min = static_cast<double>(static_cast<int>(min));
}
if ( in.bad() || !(ref == 'N' || ref == 'S' || ref == 'E' || ref == 'W')
Fix issues detected with PVS-Studio + other little improvements (#1689) * avoid re-declaration of constant variables * Replace pthreads critical section with std::mutex * ci - better naming * cmake - increase minimum version to 3.11. Use project DESCRIPTION * fix - do not treat string::find() return type as bool * remove conditions that were always true * remove condition that were always false * Remove EXV_HAVE_GMTIME_R which is not used anymore * pixelWidth_ was inherited from Exiv2::Image The width & height variables in the TiffImage class need to be mutable to be able to change their values on the getters pixelHeight() and pixelWidth() ... Do not ask me why ... * Remove superfluous if * pvs:V766 item with identical key added already * pvs:V730 not all members were initialized (time) * pvs:V730 not all members are initialized * pvs:v668 no point in testing pointer against null after new * pvs:V1048 variable assigned the same value * replace c-style dynamic vector with std one * pvs:547 fakeData is always true * Remove useless constructor in derived class * pvs:V690 modern way to disable copy-ctor * Replace malloc/free with new/delete. No need to check for null * pvs:V1028 cast operands and not result * Remove custom MIN/MAX functions * pvs:V595 pointer used before verified against null * pvs: index used before being checked * pvs:V1028 possible overflow. Cast operands * pvs:v575 potential null pointer passed to other functions * pvs:V547 deal with always true/false expressions * pvs:V560 part of conditional expressions always false or true * pvs:V701 possible break in realloc -> move to std::vector * Make some classes 'final' * Replace sprintf with std::to_string() * fix compilation on windows
4 years ago
|| sep1 != ',' || !in.eof()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
Rational rdeg = floatToRationalCast(static_cast<float>(deg));
Rational rmin = floatToRationalCast(static_cast<float>(min));
Rational rsec = floatToRationalCast(static_cast<float>(sec));
std::ostringstream array;
array << rdeg << " " << rmin << " " << rsec;
(*exifData_)[to] = array.str();
prepareExifTarget((std::string(to) + "Ref").c_str(), true);
char ref_str[2] = {ref, 0};
(*exifData_)[std::string(to) + "Ref"] = ref_str;
if (erase_) xmpData_->erase(pos);
}
void Converter::cnvIptcValue(const char* from, const char* to)
{
auto pos = iptcData_->findKey(IptcKey(from));
if (pos == iptcData_->end())
return;
if (!prepareXmpTarget(to))
return;
while (pos != iptcData_->end()) {
if (pos->key() == from) {
std::string value = pos->toString();
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
++pos;
continue;
}
if (iptcCharset_) convertStringCharset(value, iptcCharset_, "UTF-8");
(*xmpData_)[to] = value;
if (erase_) {
pos = iptcData_->erase(pos);
continue;
}
}
++pos;
}
}
void Converter::cnvXmpValueToIptc(const char* from, const char* to)
{
auto pos = xmpData_->findKey(XmpKey(from));
if (pos == xmpData_->end())
return;
if (!prepareIptcTarget(to))
return;
if (pos->typeId() == langAlt || pos->typeId() == xmpText) {
std::string value;
if (!getTextValue(value, pos)) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
return;
}
(*iptcData_)[to] = value;
(*iptcData_)["Iptc.Envelope.CharacterSet"] = "\033%G"; // indicate UTF-8 encoding
if (erase_) xmpData_->erase(pos);
return;
}
size_t count = pos->count();
bool added = false;
for (size_t i = 0; i < count; ++i) {
std::string value = pos->toString(static_cast<long>(i));
if (!pos->value().ok()) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Failed to convert " << from << " to " << to << "\n";
#endif
continue;
}
IptcKey key(to);
Iptcdatum id(key);
id.setValue(value);
iptcData_->add(id);
added = true;
}
if (added) (*iptcData_)["Iptc.Envelope.CharacterSet"] = "\033%G"; // indicate UTF-8 encoding
if (erase_) xmpData_->erase(pos);
}
#ifdef EXV_HAVE_XMP_TOOLKIT
std::string Converter::computeExifDigest(bool tiff)
{
std::ostringstream res;
MD5_CTX context;
unsigned char digest[16];
MD5Init ( &context );
for (auto&& c : conversion_) {
if (c.metadataId_ == mdExif) {
Exiv2::ExifKey key(c.key1_);
if (tiff && key.groupName() != "Image") continue;
if (!tiff && key.groupName() == "Image") continue;
if (!res.str().empty()) res << ',';
res << key.tag();
auto pos = exifData_->findKey(key);
if (pos == exifData_->end()) continue;
DataBuf data(pos->size());
pos->copy(data.data(), littleEndian /* FIXME ? */);
MD5Update ( &context, data.c_data(), static_cast<uint32_t>(data.size()));
}
}
MD5Final(digest, &context);
res << ';';
res << std::setw(2) << std::setfill('0') << std::hex << std::uppercase;
for (auto&& i : digest) {
res << static_cast<int>(i);
}
return res.str();
}
#else
std::string Converter::computeExifDigest(bool)
{
return std::string("");
}
#endif
void Converter::writeExifDigest()
{
#ifdef EXV_HAVE_XMP_TOOLKIT
(*xmpData_)["Xmp.tiff.NativeDigest"] = computeExifDigest(true);
(*xmpData_)["Xmp.exif.NativeDigest"] = computeExifDigest(false);
#endif
}
void Converter::syncExifWithXmp()
{
auto td = xmpData_->findKey(XmpKey("Xmp.tiff.NativeDigest"));
auto ed = xmpData_->findKey(XmpKey("Xmp.exif.NativeDigest"));
if (td != xmpData_->end() && ed != xmpData_->end()) {
if (td->value().toString() == computeExifDigest(true) &&
ed->value().toString() == computeExifDigest(false)) {
// We have both digests and the values match
// XMP is up-to-date, we should update Exif
setOverwrite(true);
setErase(false);
cnvFromXmp();
writeExifDigest();
return;
}
// We have both digests and the values do not match
// Exif was modified after XMP, we should update XMP
setOverwrite(true);
setErase(false);
cnvToXmp();
writeExifDigest();
return;
}
// We don't have both digests, it is probably the first conversion to XMP
setOverwrite(false); // to be safe
setErase(false);
cnvToXmp();
writeExifDigest();
}
// *************************************************************************
// free functions
void copyExifToXmp(const ExifData& exifData, XmpData& xmpData)
{
/// \todo the const_cast is "lying". We are modifying the input data. Check if this might have any bad side effect
Converter converter(const_cast<ExifData&>(exifData), xmpData);
converter.cnvToXmp();
}
3 years ago
/// \todo not used internally. We should at least have unit tests for this.
void moveExifToXmp(ExifData& exifData, XmpData& xmpData)
{
Converter converter(exifData, xmpData);
converter.setErase();
converter.cnvToXmp();
}
void copyXmpToExif(const XmpData& xmpData, ExifData& exifData)
{
Converter converter(exifData, const_cast<XmpData&>(xmpData));
converter.cnvFromXmp();
}
3 years ago
/// \todo not used internally. We should at least have unit tests for this.
void moveXmpToExif(XmpData& xmpData, ExifData& exifData)
{
Converter converter(exifData, xmpData);
converter.setErase();
converter.cnvFromXmp();
}
void syncExifWithXmp(ExifData& exifData, XmpData& xmpData)
{
Converter converter(exifData, xmpData);
converter.syncExifWithXmp();
}
void copyIptcToXmp(const IptcData& iptcData, XmpData& xmpData, const char *iptcCharset)
{
if (!iptcCharset) iptcCharset = iptcData.detectCharset();
if (!iptcCharset) iptcCharset = "ISO-8859-1";
Converter converter(const_cast<IptcData&>(iptcData), xmpData, iptcCharset);
converter.cnvToXmp();
}
3 years ago
/// \todo not used internally. We should at least have unit tests for this.
void moveIptcToXmp(IptcData& iptcData, XmpData& xmpData, const char *iptcCharset)
{
if (!iptcCharset) iptcCharset = iptcData.detectCharset();
if (!iptcCharset) iptcCharset = "ISO-8859-1";
Converter converter(iptcData, xmpData, iptcCharset);
converter.setErase();
converter.cnvToXmp();
}
void copyXmpToIptc(const XmpData& xmpData, IptcData& iptcData)
{
Converter converter(iptcData, const_cast<XmpData&>(xmpData));
converter.cnvFromXmp();
}
3 years ago
/// \todo not used internally. We should at least have unit tests for this.
void moveXmpToIptc(XmpData& xmpData, IptcData& iptcData)
{
Converter converter(iptcData, xmpData);
converter.setErase();
converter.cnvFromXmp();
}
bool convertStringCharset(std::string &str, const char* from, const char* to)
{
if (0 == strcmp(from, to))
return true; // nothing to do
bool ret = false;
#if defined EXV_HAVE_ICONV
ret = convertStringCharsetIconv(str, from, to);
#elif defined WIN32 && !defined __CYGWIN__
ret = convertStringCharsetWindows(str, from, to);
#else
# ifndef SUPPRESS_WARNINGS
EXV_WARNING << "Charset conversion required but no character mapping functionality available.\n";
# endif
UNUSED(str);
#endif
return ret;
}
} // namespace Exiv2
// *****************************************************************************
// local definitions
namespace {
using namespace Exiv2;
#if defined WIN32 && !defined __CYGWIN__
bool swapBytes(std::string& str)
{
// Naive byte-swapping, I'm sure this can be done more efficiently
if (str.size() & 1) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "swapBytes: Size " << str.size() << " of input string is not even.\n";
#endif
return false;
}
for (unsigned int i = 0; i < str.size() / 2; ++i) {
char t = str[2 * i];
str[2 * i] = str[2 * i + 1];
str[2 * i + 1] = t;
}
return true;
}
bool mb2wc(UINT cp, std::string& str)
{
if (str.empty())
return true;
int len = MultiByteToWideChar(cp, 0, str.c_str(), (int)str.size(), 0, 0);
if (len == 0) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "mb2wc: Failed to determine required size of output buffer.\n";
#endif
return false;
}
std::vector<std::string::value_type> out;
out.resize(len * 2);
int ret = MultiByteToWideChar(cp, 0, str.c_str(), (int)str.size(), (LPWSTR)&out[0], len * 2);
if (ret == 0) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "mb2wc: Failed to convert the input string to a wide character string.\n";
#endif
return false;
}
str.assign(out.begin(), out.end());
return true;
}
bool wc2mb(UINT cp, std::string& str)
{
if (str.empty())
return true;
if (str.size() & 1) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "wc2mb: Size " << str.size() << " of input string is not even.\n";
#endif
return false;
}
int len = WideCharToMultiByte(cp, 0, (LPCWSTR)str.data(), (int)str.size() / 2, 0, 0, 0, 0);
if (len == 0) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "wc2mb: Failed to determine required size of output buffer.\n";
#endif
return false;
}
std::vector<std::string::value_type> out;
out.resize(len);
int ret = WideCharToMultiByte(cp, 0, (LPCWSTR)str.data(), (int)str.size() / 2, (LPSTR)&out[0], len, 0, 0);
if (ret == 0) {
#ifdef EXIV2_DEBUG_MESSAGES
EXV_DEBUG << "wc2mb: Failed to convert the input string to a multi byte string.\n";
#endif
return false;
}
str.assign(out.begin(), out.end());
return true;
}
bool utf8ToUcs2be(std::string& str)
{
bool ret = mb2wc(CP_UTF8, str);
if (ret) ret = swapBytes(str);
return ret;
}
bool utf8ToUcs2le(std::string& str)
{
return mb2wc(CP_UTF8, str);
}
bool ucs2beToUtf8(std::string& str)
{
bool ret = swapBytes(str);
if (ret) ret = wc2mb(CP_UTF8, str);
return ret;
}
bool ucs2beToUcs2le(std::string& str)
{
return swapBytes(str);
}
bool ucs2leToUtf8(std::string& str)
{
return wc2mb(CP_UTF8, str);
}
bool ucs2leToUcs2be(std::string& str)
{
return swapBytes(str);
}
bool iso88591ToUtf8(std::string& str)
{
bool ret = mb2wc(28591, str);
if (ret) ret = wc2mb(CP_UTF8, str);
return ret;
}
bool asciiToUtf8(std::string& /*str*/)
{
// nothing to do
return true;
}
using ConvFct = bool (*)(std::string& str);
struct ConvFctList {
bool operator==(const std::pair<const char*, const char*> &fromTo) const
{ return 0 == strcmp(from_, fromTo.first) && 0 == strcmp(to_, fromTo.second); }
const char* from_;
const char* to_;
ConvFct convFct_;
};
const ConvFctList convFctList[] = {
{ "UTF-8", "UCS-2BE", utf8ToUcs2be },
{ "UTF-8", "UCS-2LE", utf8ToUcs2le },
{ "UCS-2BE", "UTF-8", ucs2beToUtf8 },
{ "UCS-2BE", "UCS-2LE", ucs2beToUcs2le },
{ "UCS-2LE", "UTF-8", ucs2leToUtf8 },
{ "UCS-2LE", "UCS-2BE", ucs2leToUcs2be },
{ "ISO-8859-1", "UTF-8", iso88591ToUtf8 },
{ "ASCII", "UTF-8", asciiToUtf8 }
// Update the convertStringCharset() documentation if you add more here!
};
bool convertStringCharsetWindows(std::string& str, const char* from, const char* to)
{
bool ret = false;
const ConvFctList* p = find(convFctList, std::make_pair(from, to));
std::string tmpstr = str;
if (p) ret = p->convFct_(tmpstr);
#ifndef SUPPRESS_WARNINGS
else {
EXV_WARNING << "No Windows function to map character string from " << from << " to " << to << " available.\n";
}
#endif
if (ret) str = tmpstr;
return ret;
}
#endif // defined WIN32 && !defined __CYGWIN__
#if defined EXV_HAVE_ICONV
bool convertStringCharsetIconv(std::string& str, const char* from, const char* to)
{
if (0 == strcmp(from, to))
return true; // nothing to do
bool ret = true;
iconv_t cd;
cd = iconv_open(to, from);
if (cd == (iconv_t)(-1)) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "iconv_open: " << strError() << "\n";
#endif
return false;
}
std::string outstr;
auto inptr = const_cast<char*>(str.c_str());
size_t inbytesleft = str.length();
while (inbytesleft) {
char outbuf[256];
char* outptr = outbuf;
size_t outbytesleft = sizeof(outbuf);
size_t rc = iconv(cd,
&inptr,
&inbytesleft,
&outptr,
&outbytesleft);
const size_t outbytesProduced = sizeof(outbuf) - outbytesleft;
if (rc == size_t(-1) && errno != E2BIG) {
#ifndef SUPPRESS_WARNINGS
EXV_WARNING << "iconv: " << strError()
<< " inbytesleft = " << inbytesleft << "\n";
#endif
ret = false;
break;
}
outstr.append(std::string(outbuf, outbytesProduced));
}
if (cd != (iconv_t)(-1)) {
iconv_close(cd);
}
if (ret) str = outstr;
return ret;
}
#endif // EXV_HAVE_ICONV
bool getTextValue(std::string& value, const XmpData::iterator& pos)
{
if (pos->typeId() == langAlt) {
// get the default language entry without x-default qualifier
value = pos->toString(0);
if (!pos->value().ok() && pos->count() == 1) {
// If there is no default but exactly one entry, take that
// without the qualifier
value = pos->toString();
if ( pos->value().ok()
&& value.length() > 5 && value.substr(0, 5) == "lang=") {
const std::string::size_type first_space_pos = value.find_first_of(' ');
if (first_space_pos != std::string::npos) {
value = value.substr(first_space_pos + 1);
}
else {
value.clear();
}
}
}
}
else {
value = pos->toString();
}
return pos->value().ok();
}
} // namespace