QGIS API Documentation 3.99.0-Master (357b655ed83)
Loading...
Searching...
No Matches
qgsalgorithmrasterzonalstats.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmrasterzonalstats.cpp
3 ---------------------
4 begin : December 2018
5 copyright : (C) 2018 by Nyall Dawson
6 email : nyall dot dawson at gmail 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
20#include "qgsrasterprojector.h"
22#include "qgsstringutils.h"
23#include "qgsunittypes.h"
24
25#include <QString>
26
27using namespace Qt::StringLiterals;
28
30
31QString QgsRasterLayerZonalStatsAlgorithm::name() const
32{
33 return u"rasterlayerzonalstats"_s;
34}
35
36QString QgsRasterLayerZonalStatsAlgorithm::displayName() const
37{
38 return QObject::tr( "Raster layer zonal statistics" );
39}
40
41QStringList QgsRasterLayerZonalStatsAlgorithm::tags() const
42{
43 return QObject::tr( "count,area,statistics,stats,zones,categories,minimum,maximum,mean,sum,total" ).split( ',' );
44}
45
46QString QgsRasterLayerZonalStatsAlgorithm::group() const
47{
48 return QObject::tr( "Raster analysis" );
49}
50
51QString QgsRasterLayerZonalStatsAlgorithm::groupId() const
52{
53 return u"rasteranalysis"_s;
54}
55
56void QgsRasterLayerZonalStatsAlgorithm::initAlgorithm( const QVariantMap & )
57{
58 addParameter( new QgsProcessingParameterRasterLayer( u"INPUT"_s, QObject::tr( "Input layer" ) ) );
59 addParameter( new QgsProcessingParameterBand( u"BAND"_s, QObject::tr( "Band number" ), 1, u"INPUT"_s ) );
60 addParameter( new QgsProcessingParameterRasterLayer( u"ZONES"_s, QObject::tr( "Zones layer" ) ) );
61 addParameter( new QgsProcessingParameterBand( u"ZONES_BAND"_s, QObject::tr( "Zones band number" ), 1, u"ZONES"_s ) );
62
63 auto refParam = std::make_unique<QgsProcessingParameterEnum>( u"REF_LAYER"_s, QObject::tr( "Reference layer" ), QStringList() << QObject::tr( "Input layer" ) << QObject::tr( "Zones layer" ), false, 0 );
64 refParam->setFlags( refParam->flags() | Qgis::ProcessingParameterFlag::Advanced );
65 addParameter( refParam.release() );
66
67 addParameter( new QgsProcessingParameterFeatureSink( u"OUTPUT_TABLE"_s, QObject::tr( "Statistics" ), Qgis::ProcessingSourceType::Vector ) );
68
69 addOutput( new QgsProcessingOutputString( u"EXTENT"_s, QObject::tr( "Extent" ) ) );
70 addOutput( new QgsProcessingOutputString( u"CRS_AUTHID"_s, QObject::tr( "CRS authority identifier" ) ) );
71 addOutput( new QgsProcessingOutputNumber( u"WIDTH_IN_PIXELS"_s, QObject::tr( "Width in pixels" ) ) );
72 addOutput( new QgsProcessingOutputNumber( u"HEIGHT_IN_PIXELS"_s, QObject::tr( "Height in pixels" ) ) );
73 addOutput( new QgsProcessingOutputNumber( u"TOTAL_PIXEL_COUNT"_s, QObject::tr( "Total pixel count" ) ) );
74 addOutput( new QgsProcessingOutputNumber( u"NODATA_PIXEL_COUNT"_s, QObject::tr( "NoData pixel count" ) ) );
75}
76
77QString QgsRasterLayerZonalStatsAlgorithm::shortDescription() const
78{
79 return QObject::tr( "Calculates statistics for a raster layer's values, categorized by zones defined in another raster layer." );
80}
81
82QString QgsRasterLayerZonalStatsAlgorithm::shortHelpString() const
83{
84 return QObject::tr( "This algorithm calculates statistics for a raster layer's values, categorized by zones defined in another raster layer.\n\n"
85 "If the reference layer parameter is set to \"Input layer\", then zones are determined by sampling the zone raster layer value at the centroid of each pixel from the source raster layer.\n\n"
86 "If the reference layer parameter is set to \"Zones layer\", then the input raster layer will be sampled at the centroid of each pixel from the zones raster layer.\n\n"
87 "If either the source raster layer or the zone raster layer value is NoData for a pixel, that pixel's value will be skipped and not included in the calculated statistics." );
88}
89
90QgsRasterLayerZonalStatsAlgorithm *QgsRasterLayerZonalStatsAlgorithm::createInstance() const
91{
92 return new QgsRasterLayerZonalStatsAlgorithm();
93}
94
95bool QgsRasterLayerZonalStatsAlgorithm::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
96{
97 mRefLayer = static_cast<RefLayer>( parameterAsEnum( parameters, u"REF_LAYER"_s, context ) );
98
99 QgsRasterLayer *layer = parameterAsRasterLayer( parameters, u"INPUT"_s, context );
100 const int band = parameterAsInt( parameters, u"BAND"_s, context );
101
102 if ( !layer )
103 throw QgsProcessingException( invalidRasterError( parameters, u"INPUT"_s ) );
104
105 mBand = parameterAsInt( parameters, u"BAND"_s, context );
106 if ( mBand < 1 || mBand > layer->bandCount() )
107 throw QgsProcessingException( QObject::tr( "Invalid band number for BAND (%1): Valid values for input raster are 1 to %2" ).arg( mBand ).arg( layer->bandCount() ) );
108
109 mHasNoDataValue = layer->dataProvider()->sourceHasNoDataValue( band );
110
111 QgsRasterLayer *zonesLayer = parameterAsRasterLayer( parameters, u"ZONES"_s, context );
112
113 if ( !zonesLayer )
114 throw QgsProcessingException( invalidRasterError( parameters, u"ZONES"_s ) );
115
116 mZonesBand = parameterAsInt( parameters, u"ZONES_BAND"_s, context );
117 if ( mZonesBand < 1 || mZonesBand > zonesLayer->bandCount() )
118 throw QgsProcessingException( QObject::tr( "Invalid band number for ZONES_BAND (%1): Valid values for input raster are 1 to %2" ).arg( mZonesBand ).arg( zonesLayer->bandCount() ) );
119 mZonesHasNoDataValue = zonesLayer->dataProvider()->sourceHasNoDataValue( band );
120
121 mSourceDataProvider.reset( layer->dataProvider()->clone() );
122 mSourceInterface = mSourceDataProvider.get();
123 mZonesDataProvider.reset( zonesLayer->dataProvider()->clone() );
124 mZonesInterface = mZonesDataProvider.get();
125
126 switch ( mRefLayer )
127 {
128 case Source:
129 mCrs = layer->crs();
130 mRasterUnitsPerPixelX = layer->rasterUnitsPerPixelX();
131 mRasterUnitsPerPixelY = layer->rasterUnitsPerPixelY();
132 mLayerWidth = layer->width();
133 mLayerHeight = layer->height();
134 mExtent = layer->extent();
135
136 // add projector if necessary
137 if ( layer->crs() != zonesLayer->crs() )
138 {
139 mProjector = std::make_unique<QgsRasterProjector>();
140 mProjector->setInput( mZonesDataProvider.get() );
141 mProjector->setCrs( zonesLayer->crs(), layer->crs(), context.transformContext() );
142 mZonesInterface = mProjector.get();
143 }
144 break;
145
146 case Zones:
147 mCrs = zonesLayer->crs();
148 mRasterUnitsPerPixelX = zonesLayer->rasterUnitsPerPixelX();
149 mRasterUnitsPerPixelY = zonesLayer->rasterUnitsPerPixelY();
150 mLayerWidth = zonesLayer->width();
151 mLayerHeight = zonesLayer->height();
152 mExtent = zonesLayer->extent();
153
154 // add projector if necessary
155 if ( layer->crs() != zonesLayer->crs() )
156 {
157 mProjector = std::make_unique<QgsRasterProjector>();
158 mProjector->setInput( mSourceDataProvider.get() );
159 mProjector->setCrs( layer->crs(), zonesLayer->crs(), context.transformContext() );
160 mSourceInterface = mProjector.get();
161 }
162 break;
163 }
164
165 return true;
166}
167
168QVariantMap QgsRasterLayerZonalStatsAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
169{
170 QString areaUnit = QgsUnitTypes::toAbbreviatedString( QgsUnitTypes::distanceToAreaUnit( mCrs.mapUnits() ) );
171
172 QString tableDest;
173 std::unique_ptr<QgsFeatureSink> sink;
174 if ( parameters.contains( u"OUTPUT_TABLE"_s ) && parameters.value( u"OUTPUT_TABLE"_s ).isValid() )
175 {
176 QgsFields outFields;
177 outFields.append( QgsField( u"zone"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
178 outFields.append( QgsField( areaUnit.isEmpty() ? "area" : areaUnit.replace( u"²"_s, "2"_L1 ), QMetaType::Type::Double, QString(), 20, 8 ) );
179 outFields.append( QgsField( u"sum"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
180 outFields.append( QgsField( u"count"_s, QMetaType::Type::LongLong, QString(), 20 ) );
181 outFields.append( QgsField( u"min"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
182 outFields.append( QgsField( u"max"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
183 outFields.append( QgsField( u"mean"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
184
185 sink.reset( parameterAsSink( parameters, u"OUTPUT_TABLE"_s, context, tableDest, outFields, Qgis::WkbType::NoGeometry, QgsCoordinateReferenceSystem() ) );
186 if ( !sink )
187 throw QgsProcessingException( invalidSinkError( parameters, u"OUTPUT_TABLE"_s ) );
188 }
189
190 struct StatCalculator
191 {
192 // only calculate cheap stats-- we cannot calculate stats which require holding values in memory -- because otherwise we'll end
193 // up trying to store EVERY pixel value from the input in memory
195 };
196 QHash<double, StatCalculator> zoneStats;
197 qgssize noDataCount = 0;
198
199 const qgssize layerSize = static_cast<qgssize>( mLayerWidth ) * static_cast<qgssize>( mLayerHeight );
200
201 QgsRasterIterator iter = mRefLayer == Source ? QgsRasterIterator( mSourceInterface )
202 : QgsRasterIterator( mZonesInterface );
203 iter.startRasterRead( mRefLayer == Source ? mBand : mZonesBand, mLayerWidth, mLayerHeight, mExtent );
204
205 int iterLeft = 0;
206 int iterTop = 0;
207 int iterCols = 0;
208 int iterRows = 0;
209 QgsRectangle blockExtent;
210 std::unique_ptr<QgsRasterBlock> rasterBlock;
211 std::unique_ptr<QgsRasterBlock> zonesRasterBlock;
212 bool isNoData = false;
213 while ( true )
214 {
215 int band;
216 if ( mRefLayer == Source )
217 {
218 band = mBand;
219 if ( !iter.readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop, &blockExtent ) )
220 break;
221
222 zonesRasterBlock.reset( mZonesInterface->block( mZonesBand, blockExtent, iterCols, iterRows ) );
223 }
224 else
225 {
226 band = mZonesBand;
227 if ( !iter.readNextRasterPart( mZonesBand, iterCols, iterRows, zonesRasterBlock, iterLeft, iterTop, &blockExtent ) )
228 break;
229
230 rasterBlock.reset( mSourceInterface->block( mBand, blockExtent, iterCols, iterRows ) );
231 }
232 if ( !zonesRasterBlock || !rasterBlock )
233 continue;
234
235 feedback->setProgress( 100 * iter.progress( band ) );
236 if ( !rasterBlock->isValid() || rasterBlock->isEmpty() || !zonesRasterBlock->isValid() || zonesRasterBlock->isEmpty() )
237 continue;
238
239 for ( int row = 0; row < iterRows; row++ )
240 {
241 if ( feedback->isCanceled() )
242 break;
243
244 for ( int column = 0; column < iterCols; column++ )
245 {
246 const double value = rasterBlock->valueAndNoData( row, column, isNoData );
247 if ( mHasNoDataValue && isNoData )
248 {
249 noDataCount += 1;
250 continue;
251 }
252 const double zone = zonesRasterBlock->valueAndNoData( row, column, isNoData );
253 if ( mZonesHasNoDataValue && isNoData )
254 {
255 noDataCount += 1;
256 continue;
257 }
258 zoneStats[zone].s.addValue( value );
259 }
260 }
261 }
262
263 QVariantMap outputs;
264 outputs.insert( u"EXTENT"_s, mExtent.toString() );
265 outputs.insert( u"CRS_AUTHID"_s, mCrs.authid() );
266 outputs.insert( u"WIDTH_IN_PIXELS"_s, mLayerWidth );
267 outputs.insert( u"HEIGHT_IN_PIXELS"_s, mLayerHeight );
268 outputs.insert( u"TOTAL_PIXEL_COUNT"_s, layerSize );
269 outputs.insert( u"NODATA_PIXEL_COUNT"_s, noDataCount );
270
271 const double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
272
273 for ( auto it = zoneStats.begin(); it != zoneStats.end(); ++it )
274 {
275 QgsFeature f;
276 it->s.finalize();
277 f.setAttributes( QgsAttributes() << it.key() << it->s.count() * pixelArea << it->s.sum() << it->s.count() << it->s.min() << it->s.max() << it->s.mean() );
278 if ( !sink->addFeature( f, QgsFeatureSink::FastInsert ) )
279 throw QgsProcessingException( writeFeatureError( sink.get(), parameters, u"OUTPUT_TABLE"_s ) );
280 sink->finalize();
281 }
282 outputs.insert( u"OUTPUT_TABLE"_s, tableDest );
283
284 return outputs;
285}
286
287
@ Vector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
Definition qgis.h:3610
@ Mean
Mean of values.
Definition qgis.h:6153
@ Max
Max of values.
Definition qgis.h:6158
@ Min
Min of values.
Definition qgis.h:6157
@ Sum
Sum of values.
Definition qgis.h:6152
@ Count
Count.
Definition qgis.h:6150
@ NoGeometry
No geometry.
Definition qgis.h:298
@ Advanced
Parameter is an advanced parameter which should be hidden from users by default.
Definition qgis.h:3834
Represents a coordinate reference system (CRS).
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:55
void setProgress(double progress)
Sets the current progress for the feedback object.
Definition qgsfeedback.h:63
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
Container of fields for a vector layer.
Definition qgsfields.h:46
bool append(const QgsField &field, Qgis::FieldOrigin origin=Qgis::FieldOrigin::Provider, int originIndex=-1)
Appends a field.
Definition qgsfields.cpp:76
virtual QgsRectangle extent() const
Returns the extent of the layer.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:90
Contains information about the context in which a processing algorithm is executed.
QgsCoordinateTransformContext transformContext() const
Returns the coordinate transform context.
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
A numeric output for processing algorithms.
A string output for processing algorithms.
A raster band parameter for Processing algorithms.
A feature sink output for processing algorithms.
A raster layer parameter for processing algorithms.
QgsRasterDataProvider * clone() const override=0
Clone itself, create deep copy.
virtual bool sourceHasNoDataValue(int bandNo) const
Returns true if source band has no data value.
Iterator for sequentially processing raster cells.
bool readNextRasterPart(int bandNumber, int &nCols, int &nRows, QgsRasterBlock **block, int &topLeftCol, int &topLeftRow)
Fetches next part of raster data, caller takes ownership of the block and caller should delete the bl...
double progress(int bandNumber, double currentBlockProgress=-1) const
Returns the raster iteration progress as a fraction from 0 to 1.0, for the specified bandNumber.
void startRasterRead(int bandNumber, qgssize nCols, qgssize nRows, const QgsRectangle &extent, QgsRasterBlockFeedback *feedback=nullptr)
Start reading of raster band.
Represents a raster layer.
int height() const
Returns the height of the (unclipped) raster.
int bandCount() const
Returns the number of bands in this layer.
double rasterUnitsPerPixelX() const
Returns the number of raster units per each raster pixel in X axis.
QgsRasterDataProvider * dataProvider() override
Returns the source data provider.
double rasterUnitsPerPixelY() const
Returns the number of raster units per each raster pixel in Y axis.
int width() const
Returns the width of the (unclipped) raster.
A rectangle specified with double values.
static Q_INVOKABLE QString toAbbreviatedString(Qgis::DistanceUnit unit)
Returns a translated abbreviation representing a distance unit.
static Q_INVOKABLE Qgis::AreaUnit distanceToAreaUnit(Qgis::DistanceUnit distanceUnit)
Converts a distance unit to its corresponding area unit, e.g., meters to square meters.
unsigned long long qgssize
Qgssize is used instead of size_t, because size_t is stdlib type, unknown by SIP, and it would be har...
Definition qgis.h:7458