QGIS API Documentation 3.43.0-Master (3ee7834ace6)
qgsrasterlayertemporalproperties.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsrasterlayertemporalproperties.cpp
3 ---------------
4 begin : February 2020
5 copyright : (C) 2020 by Samweli Mwakisambwe
6 email : samweli at kartoza dot com
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
19#include "moc_qgsrasterlayertemporalproperties.cpp"
21#include "qgsrasterlayer.h"
22
24 : QgsMapLayerTemporalProperties( parent, enabled )
25{
26 mTemporalRepresentationScale.setDays( 1.0 );
27}
28
30{
31 if ( !isActive() )
32 return true;
33
34 switch ( mMode )
35 {
38 return range.isInfinite() || mFixedRange.isInfinite() || mFixedRange.overlaps( range );
39
41 {
42 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
43 {
44 if ( it.value().overlaps( range ) )
45 return true;
46 }
47 return false;
48 }
49
53 return true;
54 }
55 return true;
56}
57
59{
60 QgsRasterLayer *rasterLayer = qobject_cast< QgsRasterLayer *>( layer );
61 if ( !rasterLayer )
62 return QgsDateTimeRange();
63
64 switch ( mMode )
65 {
68 return mFixedRange;
69
72
74 {
75 QDateTime begin;
76 QDateTime end;
77 bool includeBeginning = true;
78 bool includeEnd = true;
79 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
80 {
81 if ( it.value().begin() < begin || !begin.isValid() )
82 {
83 begin = it.value().begin();
84 includeBeginning = it.value().includeBeginning();
85 }
86 else if ( !includeBeginning && it.value().begin() == begin && it.value().includeBeginning() )
87 {
88 includeBeginning = true;
89 }
90 if ( it.value().end() > end || !end.isValid() )
91 {
92 end = it.value().end();
93 includeEnd = it.value().includeEnd();
94 }
95 else if ( !includeEnd && it.value().end() == end && it.value().includeEnd() )
96 {
97 includeEnd = true;
98 }
99 }
100 return QgsDateTimeRange( begin, end, includeBeginning, includeEnd );
101 }
102
105 break;
106 }
107
108 return QgsDateTimeRange();
109}
110
112{
113 QgsRasterLayer *rasterLayer = qobject_cast< QgsRasterLayer *>( layer );
114
115 switch ( mMode )
116 {
119 return { mFixedRange };
120
122 {
123 QList<QgsDateTimeRange> results;
124 results.reserve( mRangePerBand.size() );
125 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
126 {
127 results.append( it.value() );
128 }
129 return results;
130 }
131
133 {
134 if ( !rasterLayer || !rasterLayer->dataProvider() )
135 return {};
136
137 const QList< QgsDateTimeRange > ranges = rasterLayer->dataProvider()->temporalCapabilities()->allAvailableTemporalRanges();
138 return ranges.empty() ? QList< QgsDateTimeRange > { rasterLayer->dataProvider()->temporalCapabilities()->availableTemporalRange() } : ranges;
139 }
140
143 break;
144 }
145
146 return {};
147}
148
153
155{
156 if ( mMode == mode )
157 return;
158 mMode = mode;
159}
160
177
182
184{
185 if ( mIntervalHandlingMethod == method )
186 return;
187 mIntervalHandlingMethod = method;
188}
189
191{
192 mFixedRange = range;
193}
194
196{
197 return mFixedRange;
198}
199
201{
202 return mRangePerBand;
203}
204
205void QgsRasterLayerTemporalProperties::setFixedRangePerBand( const QMap<int, QgsDateTimeRange> &ranges )
206{
207 if ( mRangePerBand == ranges )
208 return;
209
210 mRangePerBand = ranges;
211 emit changed();
212}
213
215{
216 switch ( mMode )
217 {
222 return -1;
223
225 {
226 // find the latest-most band which matches the map range
227 int currentMatchingBand = -1;
228 QgsDateTimeRange currentMatchingRange;
229 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
230 {
231 if ( it.value().overlaps( range ) )
232 {
233 if ( currentMatchingRange.isInfinite()
234 || ( it.value().includeEnd() && it.value().end() >= currentMatchingRange.end() ) // cppcheck-suppress mismatchingContainerExpression
235 || ( !currentMatchingRange.includeEnd() && it.value().end() >= currentMatchingRange.end() ) ) // cppcheck-suppress mismatchingContainerExpression
236 {
237 currentMatchingBand = it.key();
238 currentMatchingRange = it.value();
239 }
240 }
241 }
242 return currentMatchingBand;
243 }
244
246 return mBandNumber;
247 }
249}
250
252{
253 switch ( mMode )
254 {
259 {
260 const int bandCount = layer->bandCount();
261 QList< int > res;
262 res.reserve( bandCount );
263 for ( int i = 1; i <= bandCount; ++i )
264 res.append( i );
265 return res;
266 }
267
269 {
270 QList<int> res;
271 res.reserve( mRangePerBand.size() );
272 // find the latest-most band which matches the map range
273 QgsDateTimeRange currentMatchingRange;
274 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
275 {
276 if ( it.value().overlaps( range ) )
277 {
278 res.append( it.key() );
279 }
280 }
281 return res;
282 }
283
285 return QList<int>() << mBandNumber;
286 }
288}
289
291{
292 return mBandNumber;
293}
294
296{
297 if ( mBandNumber == number )
298 return;
299
300 mBandNumber = number;
301}
302
304{
305 return mTemporalRepresentationOffset;
306}
307
309{
310 if ( mTemporalRepresentationOffset == offset )
311 return;
312
313 mTemporalRepresentationOffset = offset;
314}
315
317{
318 return mTemporalRepresentationScale;
319}
320
322{
323 if ( mTemporalRepresentationScale == scale )
324 return;
325
326 mTemporalRepresentationScale = scale;
327}
328
329bool QgsRasterLayerTemporalProperties::readXml( const QDomElement &element, const QgsReadWriteContext &context )
330{
331 Q_UNUSED( context )
332 // TODO add support for raster layers with multi-temporal properties.
333
334 const QDomElement temporalNode = element.firstChildElement( QStringLiteral( "temporal" ) );
335
336 setIsActive( temporalNode.attribute( QStringLiteral( "enabled" ), QStringLiteral( "0" ) ).toInt() );
337
338 mMode = static_cast< Qgis::RasterTemporalMode >( temporalNode.attribute( QStringLiteral( "mode" ), QStringLiteral( "0" ) ). toInt() );
339 mBandNumber = temporalNode.attribute( QStringLiteral( "bandNumber" ), QStringLiteral( "1" ) ).toInt();
340 mIntervalHandlingMethod = static_cast< Qgis::TemporalIntervalMatchMethod >( temporalNode.attribute( QStringLiteral( "fetchMode" ), QStringLiteral( "0" ) ). toInt() );
341
342 switch ( mMode )
343 {
345 {
346 const QDomNode instantElement = temporalNode.namedItem( QStringLiteral( "fixedInstant" ) );
347 const QDateTime date = QDateTime::fromString( instantElement.toElement().text(), Qt::ISODate );
348
349 const QgsDateTimeRange range = QgsDateTimeRange( date, date );
350 setFixedTemporalRange( range );
351 break;
352 }
353
355 {
356 const QDomNode rangeElement = temporalNode.namedItem( QStringLiteral( "fixedRange" ) );
357
358 const QDomNode begin = rangeElement.namedItem( QStringLiteral( "start" ) );
359 const QDomNode end = rangeElement.namedItem( QStringLiteral( "end" ) );
360
361 const QDateTime beginDate = QDateTime::fromString( begin.toElement().text(), Qt::ISODate );
362 const QDateTime endDate = QDateTime::fromString( end.toElement().text(), Qt::ISODate );
363
364 const QgsDateTimeRange range = QgsDateTimeRange( beginDate, endDate );
365 setFixedTemporalRange( range );
366 break;
367 }
368
370 {
371 mRangePerBand.clear();
372
373 const QDomNodeList ranges = temporalNode.firstChildElement( QStringLiteral( "ranges" ) ).childNodes();
374 for ( int i = 0; i < ranges.size(); ++i )
375 {
376 const QDomElement rangeElement = ranges.at( i ).toElement();
377 const int band = rangeElement.attribute( QStringLiteral( "band" ) ).toInt();
378 const QDateTime begin = QDateTime::fromString( rangeElement.attribute( QStringLiteral( "begin" ) ), Qt::ISODate );
379 const QDateTime end = QDateTime::fromString( rangeElement.attribute( QStringLiteral( "end" ) ), Qt::ISODate );
380 const bool includeBeginning = rangeElement.attribute( QStringLiteral( "includeBeginning" ) ).toInt();
381 const bool includeEnd = rangeElement.attribute( QStringLiteral( "includeEnd" ) ).toInt();
382 mRangePerBand.insert( band, QgsDateTimeRange( begin, end, includeBeginning, includeEnd ) );
383 }
384 break;
385 }
386
388 {
389 mTemporalRepresentationOffset = QDateTime::fromString( temporalNode.attribute( QStringLiteral( "temporalRepresentationOffset" ) ), Qt::ISODate );
390 mTemporalRepresentationScale = QgsInterval( temporalNode.attribute( QStringLiteral( "temporalRepresentationScale" ), QStringLiteral( "1" ) ).toDouble(),
391 static_cast< Qgis::TemporalUnit >( temporalNode.attribute( QStringLiteral( "temporalRepresentationScaleUnit" ), QStringLiteral( "4" ) ).toInt() ) );
392 break;
393 }
394
397 break;
398 }
399
400 return true;
401}
402
403QDomElement QgsRasterLayerTemporalProperties::writeXml( QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context )
404{
405 Q_UNUSED( context )
406 if ( element.isNull() )
407 return QDomElement();
408
409 QDomElement temporalElement = document.createElement( QStringLiteral( "temporal" ) );
410 temporalElement.setAttribute( QStringLiteral( "enabled" ), isActive() ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
411 temporalElement.setAttribute( QStringLiteral( "mode" ), QString::number( static_cast< int >( mMode ) ) );
412 temporalElement.setAttribute( QStringLiteral( "bandNumber" ), QString::number( mBandNumber ) );
413 temporalElement.setAttribute( QStringLiteral( "fetchMode" ), QString::number( static_cast< int >( mIntervalHandlingMethod ) ) );
414
415 switch ( mMode )
416 {
418 {
419 QDomElement instantElement = document.createElement( QStringLiteral( "fixedInstant" ) );
420 const QDomText instantText = document.createTextNode( mFixedRange.begin().toTimeSpec( Qt::OffsetFromUTC ).toString( Qt::ISODate ) );
421 instantElement.appendChild( instantText );
422
423 temporalElement.appendChild( instantElement );
424 break;
425 }
426
428 {
429
430 QDomElement rangeElement = document.createElement( QStringLiteral( "fixedRange" ) );
431
432 QDomElement startElement = document.createElement( QStringLiteral( "start" ) );
433 QDomElement endElement = document.createElement( QStringLiteral( "end" ) );
434
435 const QDomText startText = document.createTextNode( mFixedRange.begin().toTimeSpec( Qt::OffsetFromUTC ).toString( Qt::ISODate ) );
436 const QDomText endText = document.createTextNode( mFixedRange.end().toTimeSpec( Qt::OffsetFromUTC ).toString( Qt::ISODate ) );
437 startElement.appendChild( startText );
438 endElement.appendChild( endText );
439 rangeElement.appendChild( startElement );
440 rangeElement.appendChild( endElement );
441
442 temporalElement.appendChild( rangeElement );
443 break;
444 }
445
447 {
448 QDomElement ranges = document.createElement( QStringLiteral( "ranges" ) );
449 for ( auto it = mRangePerBand.constBegin(); it != mRangePerBand.constEnd(); ++it )
450 {
451 QDomElement range = document.createElement( QStringLiteral( "range" ) );
452 range.setAttribute( QStringLiteral( "band" ), it.key() );
453 range.setAttribute( QStringLiteral( "begin" ), it.value().begin().toString( Qt::ISODate ) );
454 range.setAttribute( QStringLiteral( "end" ), it.value().end().toString( Qt::ISODate ) );
455 range.setAttribute( QStringLiteral( "includeBeginning" ), it.value().includeBeginning() ? "1" : "0" );
456 range.setAttribute( QStringLiteral( "includeEnd" ), it.value().includeEnd() ? "1" : "0" );
457 ranges.appendChild( range );
458 }
459 temporalElement.appendChild( ranges );
460 break;
461 }
462
464 {
465 temporalElement.setAttribute( QStringLiteral( "temporalRepresentationOffset" ), mTemporalRepresentationOffset.toString( Qt::ISODate ) );
466 temporalElement.setAttribute( QStringLiteral( "temporalRepresentationScale" ), QString::number( mTemporalRepresentationScale.originalDuration() ) );
467 temporalElement.setAttribute( QStringLiteral( "temporalRepresentationScaleUnit" ), QString::number( static_cast< int >( mTemporalRepresentationScale.originalUnit() ) ) );
468 break;
469 }
470
473 break;
474 }
475
476 element.appendChild( temporalElement );
477
478 return element;
479}
480
482{
483 if ( const QgsRasterDataProviderTemporalCapabilities *rasterCaps = dynamic_cast< const QgsRasterDataProviderTemporalCapabilities *>( capabilities ) )
484 {
485 setIsActive( rasterCaps->hasTemporalCapabilities() );
486 setFixedTemporalRange( rasterCaps->availableTemporalRange() );
487
488 if ( rasterCaps->hasTemporalCapabilities() )
489 {
491 }
492
493 mIntervalHandlingMethod = rasterCaps->intervalHandlingMethod();
494 }
495}
TemporalIntervalMatchMethod
Method to use when resolving a temporal range to a data provider layer or band.
Definition qgis.h:2585
TemporalUnit
Temporal units.
Definition qgis.h:4984
RasterTemporalMode
Raster layer temporal modes.
Definition qgis.h:2569
@ RepresentsTemporalValues
Pixel values represent an datetime.
@ RedrawLayerOnly
Redraw the layer when temporal range changes, but don't apply any filtering. Useful when raster symbo...
@ FixedRangePerBand
Layer has a fixed temporal range per band.
@ TemporalRangeFromDataProvider
Mode when raster layer delegates temporal range handling to the dataprovider.
@ FixedTemporalRange
Mode when temporal properties have fixed start and end datetimes.
@ FixedDateTime
Layer has a fixed date time instant.
Base class for handling properties relating to a data provider's temporal capabilities.
A representation of the interval between two datetime values.
Definition qgsinterval.h:46
double originalDuration() const
Returns the original interval duration.
Qgis::TemporalUnit originalUnit() const
Returns the original interval temporal unit.
void setDays(double days)
Sets the interval duration in days.
Base class for storage of map layer temporal properties.
Base class for all map layer types.
Definition qgsmaplayer.h:76
Implementation of data provider temporal properties for QgsRasterDataProviders.
QList< QgsDateTimeRange > allAvailableTemporalRanges() const
Returns a list of all valid datetime ranges for which temporal data is available from the provider.
const QgsDateTimeRange & availableTemporalRange() const
Returns the overall datetime range extent from which temporal data is available from the provider.
QgsRasterDataProviderTemporalCapabilities * temporalCapabilities() override
Returns the provider's temporal capabilities.
QDomElement writeXml(QDomElement &element, QDomDocument &doc, const QgsReadWriteContext &context) override
Writes the properties to a DOM element, to be used later with readXml().
QDateTime temporalRepresentationOffset() const
Returns the temporal offset, which is a fixed datetime which should be added to individual pixel valu...
void setDefaultsFromDataProviderTemporalCapabilities(const QgsDataProviderTemporalCapabilities *capabilities) override
Sets the layers temporal settings to appropriate defaults based on a provider's temporal capabilities...
bool isVisibleInTemporalRange(const QgsDateTimeRange &range) const override
Returns true if the layer should be visible and rendered for the specified time range.
QgsTemporalProperty::Flags flags() const override
Returns flags associated to the temporal property.
QList< int > filteredBandsForTemporalRange(QgsRasterLayer *layer, const QgsDateTimeRange &range) const
Returns a filtered list of bands which match the specified range.
void setTemporalRepresentationOffset(const QDateTime &offset)
Sets the temporal offset, which is a fixed datetime which should be added to individual pixel values ...
Qgis::TemporalIntervalMatchMethod intervalHandlingMethod() const
Returns the desired method to use when resolving a temporal interval to matching layers or bands in t...
const QgsInterval & temporalRepresentationScale() const
Returns the scale, which is an interval factor which should be applied to individual pixel values fro...
void setIntervalHandlingMethod(Qgis::TemporalIntervalMatchMethod method)
Sets the desired method to use when resolving a temporal interval to matching layers or bands in the ...
Qgis::RasterTemporalMode mode() const
Returns the temporal properties mode.
void setTemporalRepresentationScale(const QgsInterval &scale)
Sets the scale, which is an interval factor which should be applied to individual pixel values from t...
int bandForTemporalRange(QgsRasterLayer *layer, const QgsDateTimeRange &range) const
Returns the band corresponding to the specified range.
QList< QgsDateTimeRange > allTemporalRanges(QgsMapLayer *layer) const override
Attempts to calculate the overall list of all temporal extents which are contained in the specified l...
void setMode(Qgis::RasterTemporalMode mode)
Sets the temporal properties mode.
QgsRasterLayerTemporalProperties(QObject *parent=nullptr, bool enabled=false)
Constructor for QgsRasterLayerTemporalProperties, with the specified parent object.
void setFixedTemporalRange(const QgsDateTimeRange &range)
Sets a temporal range to apply to the whole layer.
void setFixedRangePerBand(const QMap< int, QgsDateTimeRange > &ranges)
Sets the fixed temporal range for each band.
bool readXml(const QDomElement &element, const QgsReadWriteContext &context) override
Reads temporal properties from a DOM element previously written by writeXml().
QMap< int, QgsDateTimeRange > fixedRangePerBand() const
Returns the fixed temporal range for each band.
int bandNumber() const
Returns the band number from which temporal values should be taken.
const QgsDateTimeRange & fixedTemporalRange() const
Returns the fixed temporal range for the layer.
QgsDateTimeRange calculateTemporalExtent(QgsMapLayer *layer) const override
Attempts to calculate the overall temporal extent for the specified layer, using the settings defined...
void setBandNumber(int number)
Sets the band number from which temporal values should be taken.
Represents a raster layer.
int bandCount() const
Returns the number of bands in this layer.
QgsRasterDataProvider * dataProvider() override
Returns the source data provider.
The class is used as a container of context for various read/write operations on other objects.
void changed()
Emitted when the temporal properties have changed.
bool isActive() const
Returns true if the temporal property is active.
void setIsActive(bool active)
Sets whether the temporal property is active.
@ FlagDontInvalidateCachedRendersWhenRangeChanges
Any cached rendering will not be invalidated when temporal range context is modified.
T begin() const
Returns the beginning of the range.
Definition qgsrange.h:446
T end() const
Returns the upper bound of the range.
Definition qgsrange.h:453
bool overlaps(const QgsTemporalRange< T > &other) const
Returns true if this range overlaps another range.
Definition qgsrange.h:572
bool includeEnd() const
Returns true if the end is inclusive, or false if the end is exclusive.
Definition qgsrange.h:468
bool isInfinite() const
Returns true if the range consists of all possible values.
Definition qgsrange.h:482
#define BUILTIN_UNREACHABLE
Definition qgis.h:6873
QgsTemporalRange< QDateTime > QgsDateTimeRange
QgsRange which stores a range of date times.
Definition qgsrange.h:760