21#include <QDirIterator>
23#include <QRegularExpression>
27QString QgsImportPhotosAlgorithm::name()
const
29 return QStringLiteral(
"importphotos" );
32QString QgsImportPhotosAlgorithm::displayName()
const
34 return QObject::tr(
"Import geotagged photos" );
37QStringList QgsImportPhotosAlgorithm::tags()
const
39 return QObject::tr(
"exif,metadata,gps,jpeg,jpg" ).split(
',' );
42QString QgsImportPhotosAlgorithm::group()
const
44 return QObject::tr(
"Vector creation" );
47QString QgsImportPhotosAlgorithm::groupId()
const
49 return QStringLiteral(
"vectorcreation" );
52void QgsImportPhotosAlgorithm::initAlgorithm(
const QVariantMap & )
57 std::unique_ptr<QgsProcessingParameterFeatureSink> output = std::make_unique<QgsProcessingParameterFeatureSink>( QStringLiteral(
"OUTPUT" ), QObject::tr(
"Photos" ),
Qgis::ProcessingSourceType::VectorPoint, QVariant(),
true );
58 output->setCreateByDefault(
true );
59 addParameter( output.release() );
61 std::unique_ptr<QgsProcessingParameterFeatureSink> invalid = std::make_unique<QgsProcessingParameterFeatureSink>( QStringLiteral(
"INVALID" ), QObject::tr(
"Invalid photos table" ),
Qgis::ProcessingSourceType::Vector, QVariant(),
true );
62 invalid->setCreateByDefault(
false );
63 addParameter( invalid.release() );
66QString QgsImportPhotosAlgorithm::shortHelpString()
const
68 return QObject::tr(
"Creates a point layer corresponding to the geotagged locations from JPEG or HEIF/HEIC images from a source folder. Optionally the folder can be recursively scanned.\n\n"
69 "The point layer will contain a single PointZ feature per input file from which the geotags could be read. Any altitude information from the geotags will be used "
70 "to set the point's Z value.\n\n"
71 "Optionally, a table of unreadable or non-geotagged photos can also be created." );
74QgsImportPhotosAlgorithm *QgsImportPhotosAlgorithm::createInstance()
const
76 return new QgsImportPhotosAlgorithm();
79QVariant QgsImportPhotosAlgorithm::parseMetadataValue(
const QString &value )
81 const thread_local QRegularExpression numRx( QStringLiteral(
"^\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*$" ) );
82 const QRegularExpressionMatch numMatch = numRx.match( value );
83 if ( numMatch.hasMatch() )
85 return numMatch.captured( 1 ).toDouble();
90bool QgsImportPhotosAlgorithm::extractGeoTagFromMetadata(
const QVariantMap &metadata,
QgsPointXY &tag )
93 if ( metadata.contains( QStringLiteral(
"EXIF_GPSLongitude" ) ) )
96 x = metadata.value( QStringLiteral(
"EXIF_GPSLongitude" ) ).toDouble( &ok );
100#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
101 if ( metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toString().rightRef( 1 ).compare( QLatin1String(
"W" ), Qt::CaseInsensitive ) == 0
102 || metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toDouble() < 0 )
104 if ( QStringView { metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toString() }.right( 1 ).compare( QLatin1String(
"W" ), Qt::CaseInsensitive ) == 0
105 || metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toDouble() < 0 )
117 if ( metadata.contains( QStringLiteral(
"EXIF_GPSLatitude" ) ) )
120 y = metadata.value( QStringLiteral(
"EXIF_GPSLatitude" ) ).toDouble( &ok );
124#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
125 if ( metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toString().rightRef( 1 ).compare( QLatin1String(
"S" ), Qt::CaseInsensitive ) == 0
126 || metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toDouble() < 0 )
128 if ( QStringView { metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toString() }.right( 1 ).compare( QLatin1String(
"S" ), Qt::CaseInsensitive ) == 0
129 || metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toDouble() < 0 )
144QVariant QgsImportPhotosAlgorithm::extractAltitudeFromMetadata(
const QVariantMap &metadata )
147 if ( metadata.contains( QStringLiteral(
"EXIF_GPSAltitude" ) ) )
149 double alt = metadata.value( QStringLiteral(
"EXIF_GPSAltitude" ) ).toDouble();
150 if ( metadata.contains( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ) && ( ( metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).userType() == QMetaType::Type::QString && metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).toString().right( 1 ) == QLatin1String(
"1" ) ) || metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).toDouble() < 0 ) )
157QVariant QgsImportPhotosAlgorithm::extractDirectionFromMetadata(
const QVariantMap &metadata )
160 if ( metadata.contains( QStringLiteral(
"EXIF_GPSImgDirection" ) ) )
162 direction = metadata.value( QStringLiteral(
"EXIF_GPSImgDirection" ) ).toDouble();
167QVariant QgsImportPhotosAlgorithm::extractOrientationFromMetadata(
const QVariantMap &metadata )
169 QVariant orientation;
170 if ( metadata.contains( QStringLiteral(
"EXIF_Orientation" ) ) )
172 switch ( metadata.value( QStringLiteral(
"EXIF_Orientation" ) ).toInt() )
203QVariant QgsImportPhotosAlgorithm::extractTimestampFromMetadata(
const QVariantMap &metadata )
206 if ( metadata.contains( QStringLiteral(
"EXIF_DateTimeOriginal" ) ) )
208 ts = metadata.value( QStringLiteral(
"EXIF_DateTimeOriginal" ) );
210 else if ( metadata.contains( QStringLiteral(
"EXIF_DateTimeDigitized" ) ) )
212 ts = metadata.value( QStringLiteral(
"EXIF_DateTimeDigitized" ) );
214 else if ( metadata.contains( QStringLiteral(
"EXIF_DateTime" ) ) )
216 ts = metadata.value( QStringLiteral(
"EXIF_DateTime" ) );
222 const thread_local QRegularExpression dsRegEx( QStringLiteral(
"(\\d+):(\\d+):(\\d+)\\s+(\\d+):(\\d+):(\\d+)" ) );
223 const QRegularExpressionMatch dsMatch = dsRegEx.match( ts.toString() );
224 if ( dsMatch.hasMatch() )
226 const int year = dsMatch.captured( 1 ).toInt();
227 const int month = dsMatch.captured( 2 ).toInt();
228 const int day = dsMatch.captured( 3 ).toInt();
229 const int hour = dsMatch.captured( 4 ).toInt();
230 const int min = dsMatch.captured( 5 ).toInt();
231 const int sec = dsMatch.captured( 6 ).toInt();
232 return QDateTime( QDate( year, month, day ), QTime( hour, min, sec ) );
240QVariant QgsImportPhotosAlgorithm::parseCoord(
const QString &
string )
242 const thread_local QRegularExpression coordRx( QStringLiteral(
"^\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*$" ) );
243 const QRegularExpressionMatch coordMatch = coordRx.match(
string );
244 if ( coordMatch.hasMatch() )
246 const double hours = coordMatch.captured( 1 ).toDouble();
247 const double minutes = coordMatch.captured( 2 ).toDouble();
248 const double seconds = coordMatch.captured( 3 ).toDouble();
249 return hours + minutes / 60.0 + seconds / 3600.0;
257QVariantMap QgsImportPhotosAlgorithm::parseMetadataList(
const QStringList &input )
260 const thread_local QRegularExpression splitRx( QStringLiteral(
"(.*?)=(.*)" ) );
261 for (
const QString &item : input )
263 const QRegularExpressionMatch match = splitRx.match( item );
264 if ( !match.hasMatch() )
267 const QString tag = match.captured( 1 );
268 QVariant value = parseMetadataValue( match.captured( 2 ) );
270 if ( tag == QLatin1String(
"EXIF_GPSLatitude" ) || tag == QLatin1String(
"EXIF_GPSLongitude" ) )
271 value = parseCoord( value.toString() );
272 results.insert( tag, value );
283 if (
QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( layer ) )
287 config.insert( QStringLiteral(
"DocumentViewer" ), 1 );
288 config.insert( QStringLiteral(
"FileWidget" ),
true );
289 config.insert( QStringLiteral(
"UseLink" ),
true );
290 config.insert( QStringLiteral(
"FullUrl" ),
true );
291 vl->setEditorWidgetSetup( vl->fields().lookupField( QStringLiteral(
"photo" ) ),
QgsEditorWidgetSetup( QStringLiteral(
"ExternalResource" ), config ) );
295 config.insert( QStringLiteral(
"FileWidgetButton" ),
true );
296 config.insert( QStringLiteral(
"StorageMode" ), 1 );
297 config.insert( QStringLiteral(
"UseLink" ),
true );
298 config.insert( QStringLiteral(
"FullUrl" ),
true );
299 vl->setEditorWidgetSetup( vl->fields().lookupField( QStringLiteral(
"directory" ) ),
QgsEditorWidgetSetup( QStringLiteral(
"ExternalResource" ), config ) );
306 const QString folder = parameterAsFile( parameters, QStringLiteral(
"FOLDER" ), context );
308 const QDir importDir( folder );
309 if ( !importDir.exists() )
314 const bool recurse = parameterAsBoolean( parameters, QStringLiteral(
"RECURSIVE" ), context );
317 outFields.
append(
QgsField( QStringLiteral(
"photo" ), QMetaType::Type::QString ) );
318 outFields.
append(
QgsField( QStringLiteral(
"filename" ), QMetaType::Type::QString ) );
319 outFields.
append(
QgsField( QStringLiteral(
"directory" ), QMetaType::Type::QString ) );
320 outFields.
append(
QgsField( QStringLiteral(
"altitude" ), QMetaType::Type::Double ) );
321 outFields.
append(
QgsField( QStringLiteral(
"direction" ), QMetaType::Type::Double ) );
322 outFields.
append(
QgsField( QStringLiteral(
"rotation" ), QMetaType::Type::Int ) );
323 outFields.
append(
QgsField( QStringLiteral(
"longitude" ), QMetaType::Type::QString ) );
324 outFields.
append(
QgsField( QStringLiteral(
"latitude" ), QMetaType::Type::QString ) );
325 outFields.
append(
QgsField( QStringLiteral(
"timestamp" ), QMetaType::Type::QDateTime ) );
330 invalidFields.
append(
QgsField( QStringLiteral(
"photo" ), QMetaType::Type::QString ) );
331 invalidFields.
append(
QgsField( QStringLiteral(
"filename" ), QMetaType::Type::QString ) );
332 invalidFields.
append(
QgsField( QStringLiteral(
"directory" ), QMetaType::Type::QString ) );
333 invalidFields.
append(
QgsField( QStringLiteral(
"readable" ), QMetaType::Type::Bool ) );
335 std::unique_ptr<QgsFeatureSink> invalidSink( parameterAsSink( parameters, QStringLiteral(
"INVALID" ), context, invalidDest, invalidFields ) );
337 const QStringList nameFilters {
"*.jpeg",
"*.jpg",
"*.heic" };
342 const QFileInfoList fileInfoList = importDir.entryInfoList( nameFilters, QDir::NoDotAndDotDot | QDir::Files );
343 for (
auto infoIt = fileInfoList.constBegin(); infoIt != fileInfoList.constEnd(); ++infoIt )
345 files.append( infoIt->absoluteFilePath() );
350 QDirIterator it( folder, nameFilters, QDir::NoDotAndDotDot | QDir::Files, QDirIterator::Subdirectories );
351 while ( it.hasNext() )
354 files.append( it.filePath() );
358 auto saveInvalidFile = [&invalidSink, ¶meters](
QgsAttributes &attributes,
bool readable ) {
362 attributes.append( readable );
365 throw QgsProcessingException( writeFeatureError( invalidSink.get(), parameters, QStringLiteral(
"INVALID" ) ) );
369 const double step =
files.count() > 0 ? 100.0 /
files.count() : 1;
371 for (
const QString &file :
files )
381 const QFileInfo fi( file );
383 attributes << QDir::toNativeSeparators( file )
384 << fi.completeBaseName()
385 << QDir::toNativeSeparators( fi.absolutePath() );
390 feedback->
reportError( QObject::tr(
"Could not open %1" ).arg( QDir::toNativeSeparators( file ) ) );
391 saveInvalidFile( attributes,
false );
395 char **GDALmetadata = GDALGetMetadata( hDS.get(),
nullptr );
398 GDALmetadata = GDALGetMetadata( hDS.get(),
"EXIF" );
402 feedback->
reportError( QObject::tr(
"No metadata found in %1" ).arg( QDir::toNativeSeparators( file ) ) );
403 saveInvalidFile( attributes,
true );
414 if ( !extractGeoTagFromMetadata( metadata, tag ) )
417 feedback->
reportError( QObject::tr(
"Could not retrieve geotag for %1" ).arg( QDir::toNativeSeparators( file ) ) );
418 saveInvalidFile( attributes,
true );
422 const QVariant altitude = extractAltitudeFromMetadata( metadata );
428 << extractDirectionFromMetadata( metadata )
429 << extractOrientationFromMetadata( metadata )
432 << extractTimestampFromMetadata( metadata );
435 throw QgsProcessingException( writeFeatureError( outputSink.get(), parameters, QStringLiteral(
"OUTPUT" ) ) );
442 outputSink->finalize();
443 outputs.insert( QStringLiteral(
"OUTPUT" ), outputDest );
453 invalidSink->finalize();
454 outputs.insert( QStringLiteral(
"INVALID" ), invalidDest );
@ Vector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
@ VectorPoint
Vector point layers.
@ Folder
Parameter is a folder.
This class represents a coordinate reference system (CRS).
@ FastInsert
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
bool isCanceled() const
Tells whether the operation has been canceled already.
void setProgress(double progress)
Sets the current progress for the feedback object.
Encapsulate a field in an attribute table or data source.
Container of fields for a vector layer.
bool append(const QgsField &field, Qgis::FieldOrigin origin=Qgis::FieldOrigin::Provider, int originIndex=-1)
Appends a field.
A geometry is the spatial representation of a feature.
Base class for all map layer types.
static QStringList cStringListToQStringList(char **stringList)
Converts a c string list to a QStringList.
A class to represent a 2D point.
Point geometry type, with support for z-dimension and m-values.
void setPostProcessor(QgsProcessingLayerPostProcessorInterface *processor)
Sets the layer post-processor.
Contains information about the context in which a processing algorithm is executed.
QgsProcessingContext::LayerDetails & layerToLoadOnCompletionDetails(const QString &layer)
Returns a reference to the details for a given layer which is loaded on completion of the algorithm o...
bool willLoadLayerOnCompletion(const QString &layer) const
Returns true if the given layer (by ID or datasource) will be loaded into the current project upon co...
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
virtual void reportError(const QString &error, bool fatalError=false)
Reports that the algorithm encountered an error while executing.
An interface for layer post-processing handlers for execution following a processing algorithm operat...
virtual void postProcessLayer(QgsMapLayer *layer, QgsProcessingContext &context, QgsProcessingFeedback *feedback)=0
Post-processes the specified layer, following successful execution of a processing algorithm.
A boolean parameter for processing algorithms.
An input file or folder parameter for processing algorithms.
Represents a vector layer which manages a vector based data sets.
CORE_EXPORT const QStringList files(const QString &zip)
Returns the list of files within a zip file.
std::unique_ptr< std::remove_pointer< GDALDatasetH >::type, GDALDatasetCloser > dataset_unique_ptr
Scoped GDAL dataset.