25 QString QgsRasterLayerZonalStatsAlgorithm::name()
const
27 return QStringLiteral(
"rasterlayerzonalstats" );
30 QString QgsRasterLayerZonalStatsAlgorithm::displayName()
const
32 return QObject::tr(
"Raster layer zonal statistics" );
35 QStringList QgsRasterLayerZonalStatsAlgorithm::tags()
const
37 return QObject::tr(
"count,area,statistics,stats,zones,categories,minimum,maximum,mean,sum,total" ).split(
',' );
40 QString QgsRasterLayerZonalStatsAlgorithm::group()
const
42 return QObject::tr(
"Raster analysis" );
45 QString QgsRasterLayerZonalStatsAlgorithm::groupId()
const
47 return QStringLiteral(
"rasteranalysis" );
50 void QgsRasterLayerZonalStatsAlgorithm::initAlgorithm(
const QVariantMap & )
53 QObject::tr(
"Input layer" ) ) );
55 QObject::tr(
"Band number" ), 1, QStringLiteral(
"INPUT" ) ) );
57 QObject::tr(
"Zones layer" ) ) );
59 QObject::tr(
"Zones band number" ), 1, QStringLiteral(
"ZONES" ) ) );
61 std::unique_ptr< QgsProcessingParameterEnum > refParam = qgis::make_unique< QgsProcessingParameterEnum >( QStringLiteral(
"REF_LAYER" ), QObject::tr(
"Reference layer" ),
62 QStringList() << QObject::tr(
"Input layer" ) << QObject::tr(
"Zones layer" ),
false, 0 );
64 addParameter( refParam.release() );
74 addOutput(
new QgsProcessingOutputNumber( QStringLiteral(
"NODATA_PIXEL_COUNT" ), QObject::tr(
"NODATA pixel count" ) ) );
77 QString QgsRasterLayerZonalStatsAlgorithm::shortDescription()
const
79 return QObject::tr(
"Calculates statistics for a raster layer's values, categorized by zones defined in another raster layer." );
82 QString QgsRasterLayerZonalStatsAlgorithm::shortHelpString()
const
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 including in the calculated statistics." );
90 QgsRasterLayerZonalStatsAlgorithm *QgsRasterLayerZonalStatsAlgorithm::createInstance()
const
92 return new QgsRasterLayerZonalStatsAlgorithm();
97 mRefLayer =
static_cast< RefLayer
>( parameterAsEnum( parameters, QStringLiteral(
"REF_LAYER" ), context ) );
99 QgsRasterLayer *layer = parameterAsRasterLayer( parameters, QStringLiteral(
"INPUT" ), context );
100 int band = parameterAsInt( parameters, QStringLiteral(
"BAND" ), context );
105 mBand = parameterAsInt( parameters, QStringLiteral(
"BAND" ), 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 )
112 QgsRasterLayer *zonesLayer = parameterAsRasterLayer( parameters, QStringLiteral(
"ZONES" ), context );
117 mZonesBand = parameterAsInt( parameters, QStringLiteral(
"ZONES_BAND" ), context );
118 if ( mZonesBand < 1 || mZonesBand > zonesLayer->
bandCount() )
119 throw QgsProcessingException( QObject::tr(
"Invalid band number for ZONES_BAND (%1): Valid values for input raster are 1 to %2" ).arg( mZonesBand )
124 mSourceInterface = mSourceDataProvider.get();
126 mZonesInterface = mZonesDataProvider.get();
134 mLayerWidth = layer->
width();
135 mLayerHeight = layer->
height();
136 mExtent = layer->
extent();
139 if ( layer->
crs() != zonesLayer->
crs() )
141 mProjector = qgis::make_unique< QgsRasterProjector >();
142 mProjector->setInput( mZonesDataProvider.get() );
144 mZonesInterface = mProjector.get();
149 mCrs = zonesLayer->
crs();
152 mLayerWidth = zonesLayer->
width();
153 mLayerHeight = zonesLayer->
height();
154 mExtent = zonesLayer->
extent();
157 if ( layer->
crs() != zonesLayer->
crs() )
159 mProjector = qgis::make_unique< QgsRasterProjector >();
160 mProjector->setInput( mSourceDataProvider.get() );
162 mSourceInterface = mProjector.get();
175 std::unique_ptr< QgsFeatureSink > sink;
176 if ( parameters.contains( QStringLiteral(
"OUTPUT_TABLE" ) ) && parameters.value( QStringLiteral(
"OUTPUT_TABLE" ) ).isValid() )
179 outFields.
append(
QgsField( QStringLiteral(
"zone" ), QVariant::Double, QString(), 20, 8 ) );
180 outFields.
append(
QgsField( areaUnit.replace( QStringLiteral(
"²" ), QStringLiteral(
"2" ) ), QVariant::Double, QString(), 20, 8 ) );
181 outFields.
append(
QgsField( QStringLiteral(
"sum" ), QVariant::Double, QString(), 20, 8 ) );
182 outFields.
append(
QgsField( QStringLiteral(
"count" ), QVariant::LongLong, QString(), 20 ) );
183 outFields.
append(
QgsField( QStringLiteral(
"min" ), QVariant::Double, QString(), 20, 8 ) );
184 outFields.
append(
QgsField( QStringLiteral(
"max" ), QVariant::Double, QString(), 20, 8 ) );
185 outFields.
append(
QgsField( QStringLiteral(
"mean" ), QVariant::Double, QString(), 20, 8 ) );
192 struct StatCalculator
198 QHash<double, StatCalculator > zoneStats;
204 int nbBlocksWidth =
static_cast< int>( std::ceil( 1.0 * mLayerWidth / maxWidth ) );
205 int nbBlocksHeight =
static_cast< int >( std::ceil( 1.0 * mLayerHeight / maxHeight ) );
206 int nbBlocks = nbBlocksWidth * nbBlocksHeight;
210 iter.
startRasterRead( mRefLayer == Source ? mBand : mZonesBand, mLayerWidth, mLayerHeight, mExtent );
217 std::unique_ptr< QgsRasterBlock > rasterBlock;
218 std::unique_ptr< QgsRasterBlock > zonesRasterBlock;
219 bool isNoData =
false;
222 if ( mRefLayer == Source )
224 if ( !iter.
readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop, &blockExtent ) )
227 zonesRasterBlock.reset( mZonesInterface->block( mZonesBand, blockExtent, iterCols, iterRows ) );
231 if ( !iter.
readNextRasterPart( mZonesBand, iterCols, iterRows, zonesRasterBlock, iterLeft, iterTop, &blockExtent ) )
234 rasterBlock.reset( mSourceInterface->block( mBand, blockExtent, iterCols, iterRows ) );
236 if ( !zonesRasterBlock || !rasterBlock )
239 feedback->
setProgress( 100 * ( ( iterTop / maxHeight * nbBlocksWidth ) + iterLeft / maxWidth ) / nbBlocks );
240 if ( !rasterBlock->isValid() || rasterBlock->isEmpty() || !zonesRasterBlock->isValid() || zonesRasterBlock->isEmpty() )
243 for (
int row = 0; row < iterRows; row++ )
248 for (
int column = 0; column < iterCols; column++ )
250 double value = rasterBlock->valueAndNoData( row, column, isNoData );
251 if ( mHasNoDataValue && isNoData )
256 double zone = zonesRasterBlock->valueAndNoData( row, column, isNoData );
257 if ( mZonesHasNoDataValue && isNoData )
262 zoneStats[ zone ].s.addValue( value );
268 outputs.insert( QStringLiteral(
"EXTENT" ), mExtent.toString() );
269 outputs.insert( QStringLiteral(
"CRS_AUTHID" ), mCrs.authid() );
270 outputs.insert( QStringLiteral(
"WIDTH_IN_PIXELS" ), mLayerWidth );
271 outputs.insert( QStringLiteral(
"HEIGHT_IN_PIXELS" ), mLayerHeight );
272 outputs.insert( QStringLiteral(
"TOTAL_PIXEL_COUNT" ), layerSize );
273 outputs.insert( QStringLiteral(
"NODATA_PIXEL_COUNT" ), noDataCount );
275 double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
277 for (
auto it = zoneStats.begin(); it != zoneStats.end(); ++it )
282 it->s.min() << it->s.max() << it->s.mean() );
285 outputs.insert( QStringLiteral(
"OUTPUT_TABLE" ), tableDest );