24 QString QgsImportPhotosAlgorithm::name()
const 26 return QStringLiteral(
"importphotos" );
29 QString QgsImportPhotosAlgorithm::displayName()
const 31 return QObject::tr(
"Import geotagged photos" );
34 QStringList QgsImportPhotosAlgorithm::tags()
const 36 return QObject::tr(
"exif,metadata,gps,jpeg,jpg" ).split(
',' );
39 QString QgsImportPhotosAlgorithm::group()
const 41 return QObject::tr(
"Vector creation" );
44 QString QgsImportPhotosAlgorithm::groupId()
const 46 return QStringLiteral(
"vectorcreation" );
49 void QgsImportPhotosAlgorithm::initAlgorithm(
const QVariantMap & )
54 std::unique_ptr< QgsProcessingParameterFeatureSink > output = qgis::make_unique< QgsProcessingParameterFeatureSink >( QStringLiteral(
"OUTPUT" ), QObject::tr(
"Photos" ),
QgsProcessing::TypeVectorPoint, QVariant(), true );
55 output->setCreateByDefault(
true );
56 addParameter( output.release() );
58 std::unique_ptr< QgsProcessingParameterFeatureSink > invalid = qgis::make_unique< QgsProcessingParameterFeatureSink >( QStringLiteral(
"INVALID" ), QObject::tr(
"Invalid photos table" ),
QgsProcessing::TypeVector, QVariant(), true );
59 invalid->setCreateByDefault(
false );
60 addParameter( invalid.release() );
63 QString QgsImportPhotosAlgorithm::shortHelpString()
const 65 return QObject::tr(
"Creates a point layer corresponding to the geotagged locations from JPEG images from a source folder. Optionally the folder can be recursively scanned.\n\n" 66 "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 " 67 "to set the point's Z value.\n\n" 68 "Optionally, a table of unreadable or non-geotagged photos can also be created." );
71 QgsImportPhotosAlgorithm *QgsImportPhotosAlgorithm::createInstance()
const 73 return new QgsImportPhotosAlgorithm();
76 QVariant QgsImportPhotosAlgorithm::parseMetadataValue(
const QString &value )
78 QRegularExpression numRx( QStringLiteral(
"^\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*$" ) );
79 QRegularExpressionMatch numMatch = numRx.match( value );
80 if ( numMatch.hasMatch() )
82 return numMatch.captured( 1 ).toDouble();
87 bool QgsImportPhotosAlgorithm::extractGeoTagFromMetadata(
const QVariantMap &metadata,
QgsPointXY &tag )
90 if ( metadata.contains( QStringLiteral(
"EXIF_GPSLongitude" ) ) )
93 x = metadata.value( QStringLiteral(
"EXIF_GPSLongitude" ) ).toDouble( &ok );
97 if ( metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toString().rightRef( 1 ).compare( QLatin1String(
"W" ), Qt::CaseInsensitive ) == 0
98 || metadata.value( QStringLiteral(
"EXIF_GPSLongitudeRef" ) ).toDouble() < 0 )
107 if ( metadata.contains( QStringLiteral(
"EXIF_GPSLatitude" ) ) )
110 y = metadata.value( QStringLiteral(
"EXIF_GPSLatitude" ) ).toDouble( &ok );
114 if ( metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toString().rightRef( 1 ).compare( QLatin1String(
"S" ), Qt::CaseInsensitive ) == 0
115 || metadata.value( QStringLiteral(
"EXIF_GPSLatitudeRef" ) ).toDouble() < 0 )
127 QVariant QgsImportPhotosAlgorithm::extractAltitudeFromMetadata(
const QVariantMap &metadata )
130 if ( metadata.contains( QStringLiteral(
"EXIF_GPSAltitude" ) ) )
132 double alt = metadata.value( QStringLiteral(
"EXIF_GPSAltitude" ) ).toDouble();
133 if ( metadata.contains( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ) &&
134 ( ( metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).type() == QVariant::String && metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).toString().right( 1 ) == QLatin1String(
"1" ) )
135 || metadata.value( QStringLiteral(
"EXIF_GPSAltitudeRef" ) ).toDouble() < 0 ) )
142 QVariant QgsImportPhotosAlgorithm::extractDirectionFromMetadata(
const QVariantMap &metadata )
145 if ( metadata.contains( QStringLiteral(
"EXIF_GPSImgDirection" ) ) )
147 direction = metadata.value( QStringLiteral(
"EXIF_GPSImgDirection" ) ).toDouble();
152 QVariant QgsImportPhotosAlgorithm::extractTimestampFromMetadata(
const QVariantMap &metadata )
155 if ( metadata.contains( QStringLiteral(
"EXIF_DateTimeOriginal" ) ) )
157 ts = metadata.value( QStringLiteral(
"EXIF_DateTimeOriginal" ) );
159 else if ( metadata.contains( QStringLiteral(
"EXIF_DateTimeDigitized" ) ) )
161 ts = metadata.value( QStringLiteral(
"EXIF_DateTimeDigitized" ) );
163 else if ( metadata.contains( QStringLiteral(
"EXIF_DateTime" ) ) )
165 ts = metadata.value( QStringLiteral(
"EXIF_DateTime" ) );
171 QRegularExpression dsRegEx( QStringLiteral(
"(\\d+):(\\d+):(\\d+)\\s+(\\d+):(\\d+):(\\d+)" ) );
172 QRegularExpressionMatch dsMatch = dsRegEx.match( ts.toString() );
173 if ( dsMatch.hasMatch() )
175 int year = dsMatch.captured( 1 ).toInt();
176 int month = dsMatch.captured( 2 ).toInt();
177 int day = dsMatch.captured( 3 ).toInt();
178 int hour = dsMatch.captured( 4 ).toInt();
179 int min = dsMatch.captured( 5 ).toInt();
180 int sec = dsMatch.captured( 6 ).toInt();
181 return QDateTime( QDate( year, month, day ), QTime( hour, min, sec ) );
189 QVariant QgsImportPhotosAlgorithm::parseCoord(
const QString &
string )
191 QRegularExpression coordRx( QStringLiteral(
"^\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*\\(\\s*([-\\.\\d]+)\\s*\\)\\s*$" ) );
192 QRegularExpressionMatch coordMatch = coordRx.match(
string );
193 if ( coordMatch.hasMatch() )
195 double hours = coordMatch.captured( 1 ).toDouble();
196 double minutes = coordMatch.captured( 2 ).toDouble();
197 double seconds = coordMatch.captured( 3 ).toDouble();
198 return hours + minutes / 60.0 + seconds / 3600.0;
206 QVariantMap QgsImportPhotosAlgorithm::parseMetadataList(
const QStringList &input )
209 QRegularExpression splitRx( QStringLiteral(
"(.*?)=(.*)" ) );
210 for (
const QString &item : input )
212 QRegularExpressionMatch match = splitRx.match( item );
213 if ( !match.hasMatch() )
216 QString tag = match.captured( 1 );
217 QVariant value = parseMetadataValue( match.captured( 2 ) );
219 if ( tag == QLatin1String(
"EXIF_GPSLatitude" ) || tag == QLatin1String(
"EXIF_GPSLongitude" ) )
220 value = parseCoord( value.toString() );
221 results.insert( tag, value );
233 if (
QgsVectorLayer *vl = qobject_cast< QgsVectorLayer * >( layer ) )
237 config.insert( QStringLiteral(
"DocumentViewer" ), 1 );
238 config.insert( QStringLiteral(
"FileWidget" ),
true );
239 vl->setEditorWidgetSetup( 0,
QgsEditorWidgetSetup( QStringLiteral(
"ExternalResource" ), config ) );
243 config.insert( QStringLiteral(
"FileWidgetButton" ),
true );
244 config.insert( QStringLiteral(
"StorageMode" ), 1 );
245 vl->setEditorWidgetSetup( 2,
QgsEditorWidgetSetup( QStringLiteral(
"ExternalResource" ), config ) );
252 QString folder = parameterAsFile( parameters, QStringLiteral(
"FOLDER" ), context );
254 QDir importDir( folder );
255 if ( !importDir.exists() )
260 bool recurse = parameterAsBool( parameters, QStringLiteral(
"RECURSIVE" ), context );
263 outFields.
append(
QgsField( QStringLiteral(
"photo" ), QVariant::String ) );
264 outFields.
append(
QgsField( QStringLiteral(
"filename" ), QVariant::String ) );
265 outFields.
append(
QgsField( QStringLiteral(
"directory" ), QVariant::String ) );
266 outFields.
append(
QgsField( QStringLiteral(
"altitude" ), QVariant::Double ) );
267 outFields.
append(
QgsField( QStringLiteral(
"direction" ), QVariant::Double ) );
268 outFields.
append(
QgsField( QStringLiteral(
"longitude" ), QVariant::String ) );
269 outFields.
append(
QgsField( QStringLiteral(
"latitude" ), QVariant::String ) );
270 outFields.
append(
QgsField( QStringLiteral(
"timestamp" ), QVariant::DateTime ) );
272 std::unique_ptr< QgsFeatureSink > outputSink( parameterAsSink( parameters, QStringLiteral(
"OUTPUT" ), context, outputDest, outFields,
276 invalidFields.
append(
QgsField( QStringLiteral(
"photo" ), QVariant::String ) );
277 invalidFields.
append(
QgsField( QStringLiteral(
"filename" ), QVariant::String ) );
278 invalidFields.
append(
QgsField( QStringLiteral(
"directory" ), QVariant::String ) );
279 invalidFields.
append(
QgsField( QStringLiteral(
"readable" ), QVariant::Bool ) );
281 std::unique_ptr< QgsFeatureSink > invalidSink( parameterAsSink( parameters, QStringLiteral(
"INVALID" ), context, invalidDest, invalidFields ) );
283 QStringList nameFilters {
"*.jpeg",
"*.jpg" };
288 QFileInfoList fileInfoList = importDir.entryInfoList( nameFilters, QDir::NoDotAndDotDot | QDir::Files );
289 for (
auto infoIt = fileInfoList.constBegin(); infoIt != fileInfoList.constEnd(); ++infoIt )
291 files.append( infoIt->absoluteFilePath() );
296 QDirIterator it( folder, nameFilters, QDir::NoDotAndDotDot | QDir::Files, QDirIterator::Subdirectories );
297 while ( it.hasNext() )
300 files.append( it.filePath() );
304 auto saveInvalidFile = [&invalidSink](
QgsAttributes & attributes,
bool readable )
309 attributes.append( readable );
315 double step = files.count() > 0 ? 100.0 / files.count() : 1;
317 for (
const QString &file : files )
327 QFileInfo fi( file );
329 attributes << QDir::toNativeSeparators( file )
330 << fi.completeBaseName()
331 << QDir::toNativeSeparators( fi.absolutePath() );
336 feedback->
reportError( QObject::tr(
"Could not open %1" ).arg( QDir::toNativeSeparators( file ) ) );
337 saveInvalidFile( attributes,
false );
341 if (
char **GDALmetadata = GDALGetMetadata( hDS.get(), nullptr ) )
350 if ( !extractGeoTagFromMetadata( metadata, tag ) )
353 feedback->
reportError( QObject::tr(
"Could not retrieve geotag for %1" ).arg( QDir::toNativeSeparators( file ) ) );
354 saveInvalidFile( attributes,
true );
358 QVariant altitude = extractAltitudeFromMetadata( metadata );
364 << extractDirectionFromMetadata( metadata )
367 << extractTimestampFromMetadata( metadata );
373 feedback->
reportError( QObject::tr(
"No metadata found in %1" ).arg( QDir::toNativeSeparators( file ) ) );
374 saveInvalidFile( attributes,
true );
381 outputs.insert( QStringLiteral(
"OUTPUT" ), outputDest );
390 outputs.insert( QStringLiteral(
"INVALID" ), invalidDest );
A boolean parameter for processing algorithms.
bool isCanceled() const
Tells whether the operation has been canceled already.
An input file or folder parameter for processing algorithms.
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
Base class for all map layer types.
Base class for providing feedback from a processing algorithm.
An interface for layer post-processing handlers for execution following a processing algorithm operat...
A class to represent a 2D point.
void setProgress(double progress)
Sets the current progress for the feedback object.
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...
void setPostProcessor(QgsProcessingLayerPostProcessorInterface *processor)
Sets the layer post-processor.
Container of fields for a vector layer.
A geometry is the spatial representation of a feature.
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
The feature class encapsulates a single feature including its id, geometry and a list of field/values...
Custom exception class for processing related exceptions.
bool append(const QgsField &field, FieldOrigin origin=OriginProvider, int originIndex=-1)
Appends a field. The field must have unique name, otherwise it is rejected (returns false) ...
Encapsulate a field in an attribute table or data source.
Point geometry type, with support for z-dimension and m-values.
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...
This class represents a coordinate reference system (CRS).
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
virtual void postProcessLayer(QgsMapLayer *layer, QgsProcessingContext &context, QgsProcessingFeedback *feedback)=0
Post-processes the specified layer, following successful execution of a processing algorithm...
std::unique_ptr< std::remove_pointer< GDALDatasetH >::type, GDALDatasetCloser > dataset_unique_ptr
Scoped GDAL dataset.
static QStringList cStringListToQStringList(char **stringList)
Converts a c string list to a QStringList.
Represents a vector layer which manages a vector based data sets.
Contains information about the context in which a processing algorithm is executed.
virtual void reportError(const QString &error, bool fatalError=false)
Reports that the algorithm encountered an error while executing.