32using namespace Qt::StringLiterals;
35 :
QgsZonalStatistics( polygonLayer, rasterLayer ? rasterLayer->dataProvider() : nullptr, rasterLayer ? rasterLayer->crs() :
QgsCoordinateReferenceSystem(), rasterLayer ? rasterLayer->rasterUnitsPerPixelX() : 0, rasterLayer ? rasterLayer->rasterUnitsPerPixelY() : 0, attributePrefix, rasterBand, stats )
40 : mRasterInterface( rasterInterface )
41 , mRasterCrs( rasterCrs )
42 , mCellSizeX( std::fabs( rasterUnitsPerPixelX ) )
43 , mCellSizeY( std::fabs( rasterUnitsPerPixelY ) )
44 , mRasterBand( rasterBand )
45 , mPolygonLayer( polygonLayer )
46 , mAttributePrefix( attributePrefix )
47 , mStatistics( stats )
53 if ( !mRasterInterface )
58 if ( mRasterInterface->bandCount() < mRasterBand )
69 if ( !vectorProvider )
74 QMap<Qgis::ZonalStatistic, int> statFieldIndexes;
77 int oldFieldCount = vectorProvider->
fields().
count();
78 QList<QgsField> newFieldList;
95 if ( mStatistics & stat )
98 QgsField field( fieldName, QMetaType::Type::Double, u
"double precision"_s );
99 newFieldList.push_back( field );
100 statFieldIndexes.insert( stat, oldFieldCount + newFieldList.count() - 1 );
115 int featureCounter = 0;
128 feedback->
setProgress( 100.0 *
static_cast<double>( featureCounter ) / featureCount );
133 QMap<Qgis::ZonalStatistic, QVariant> results =
calculateStatistics( mRasterInterface, featureGeometry, mCellSizeX, mCellSizeY, mRasterBand, mStatistics );
135 if ( results.empty() )
139 for (
const auto &result : results.toStdMap() )
141 changeAttributeMap.insert( statFieldIndexes.value( result.first ), result.second );
144 changeMap.insert( feature.
id(), changeAttributeMap );
148 mPolygonLayer->updateFields();
161QString QgsZonalStatistics::getUniqueFieldName(
const QString &fieldName,
const QList<QgsField> &newFields )
165 if ( !dp->
storageType().contains(
"ESRI Shapefile"_L1 ) )
171 allFields.append( newFields );
172 QString
shortName = fieldName.mid( 0, 10 );
175 for (
const QgsField &field : std::as_const( allFields ) )
190 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 8 ) ).arg( n );
195 for (
const QgsField &field : std::as_const( allFields ) )
202 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 8 ) ).arg( n );
206 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 7 ) ).arg( n );
220 return QObject::tr(
"Count" );
222 return QObject::tr(
"Sum" );
224 return QObject::tr(
"Mean" );
226 return QObject::tr(
"Median" );
228 return QObject::tr(
"St dev" );
230 return QObject::tr(
"Minimum" );
232 return QObject::tr(
"Maximum" );
234 return QObject::tr(
"Minimum point" );
236 return QObject::tr(
"Maximum point" );
238 return QObject::tr(
"Range" );
240 return QObject::tr(
"Minority" );
242 return QObject::tr(
"Majority" );
244 return QObject::tr(
"Variety" );
246 return QObject::tr(
"Variance" );
273 return u
"minpoint"_s;
275 return u
"maxpoint"_s;
279 return u
"minority"_s;
281 return u
"majority"_s;
285 return u
"variance"_s;
297 QMap<int, QVariant> pyResult;
298 for (
auto it = result.constBegin(); it != result.constEnd(); ++it )
300 pyResult.insert(
static_cast<int>( it.key() ), it.value() );
308 QMap<Qgis::ZonalStatistic, QVariant> results;
323 FeatureStats featureStats( statsStoreValues, statsStoreValueCount );
325 int nCellsXProvider = rasterInterface->
xSize();
326 int nCellsYProvider = rasterInterface->
ySize();
328 int nCellsX, nCellsY;
330 QgsRasterAnalysisUtils::cellInfoForBBox( rasterBBox, featureRect, cellSizeX, cellSizeY, nCellsX, nCellsY, nCellsXProvider, nCellsYProvider, rasterBlockExtent );
332 featureStats.reset();
333 QgsRasterAnalysisUtils::statisticsFromMiddlePointTest( rasterInterface, rasterBand, geometry, nCellsX, nCellsY, cellSizeX, cellSizeY, rasterBlockExtent, [&featureStats](
double value,
const QgsPointXY &point ) { featureStats.addValue( value, point ); } );
335 if ( featureStats.count <= 1 )
338 featureStats.reset();
339 QgsRasterAnalysisUtils::statisticsFromPreciseIntersection( rasterInterface, rasterBand, geometry, nCellsX, nCellsY, cellSizeX, cellSizeY, rasterBlockExtent, [&featureStats](
double value,
double weight,
const QgsPointXY &point ) { featureStats.addValue( value, point, weight ); } );
348 if ( featureStats.count > 0 )
350 double mean = featureStats.sum / featureStats.count;
355 std::sort( featureStats.values.begin(), featureStats.values.end() );
356 int size = featureStats.values.count();
357 bool even = ( size % 2 ) < 1;
361 medianValue = ( featureStats.values.at( size / 2 - 1 ) + featureStats.values.at( size / 2 ) ) / 2;
365 medianValue = featureStats.values.at( ( size + 1 ) / 2 - 1 );
371 double sumSquared = 0;
372 for (
int i = 0; i < featureStats.values.count(); ++i )
374 double diff = featureStats.values.at( i ) - mean;
375 sumSquared += diff * diff;
377 double variance = sumSquared / featureStats.values.count();
380 double stdev = std::pow( variance, 0.5 );
398 QList<int> vals = featureStats.valueCount.values();
399 std::sort( vals.begin(), vals.end() );
402 double minorityKey = featureStats.valueCount.key( vals.first() );
407 double majKey = featureStats.valueCount.key( vals.last() );
ZonalStatistic
Statistics to be calculated during a zonal statistics operation.
@ StDev
Standard deviation of pixel values.
@ Mean
Mean of pixel values.
@ Median
Median of pixel values.
@ Max
Max of pixel values.
@ Variance
Variance of pixel values.
@ MinimumPoint
Pixel centroid for minimum pixel value.
@ Min
Min of pixel values.
@ Default
Default statistics.
@ Range
Range of pixel values (max - min).
@ Sum
Sum of pixel values.
@ Minority
Minority of pixel values.
@ All
All statistics. For QGIS 3.x this includes ONLY numeric statistics, but for 4.0 this will be extended...
@ Majority
Majority of pixel values.
@ Variety
Variety (count of distinct) pixel values.
@ MaximumPoint
Pixel centroid for maximum pixel value.
ZonalStatisticResult
Zonal statistics result codes.
@ Canceled
Algorithm was canceled.
@ LayerTypeWrong
Layer is not a polygon layer.
@ RasterBandInvalid
The raster band does not exist on the raster layer.
@ RasterInvalid
Raster layer is invalid.
@ LayerInvalid
Layer is invalid.
QFlags< ZonalStatistic > ZonalStatistics
Statistics to be calculated during a zonal statistics operation.
Represents a coordinate reference system (CRS).
Wrapper for iterator of features from vector data provider or vector layer.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
Wraps a request for features to a vector layer (or directly its vector data provider).
QgsFeatureRequest & setDestinationCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets the destination crs for feature's geometries.
QgsFeatureRequest & setNoAttributes()
Set that no attributes will be fetched.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Base class for feedback objects to be used for cancellation of something running in a worker thread.
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.
QList< QgsField > toList() const
Utility function to return a list of QgsField instances.
A geometry is the spatial representation of a feature.
bool isEmpty() const
Returns true if the geometry is empty (eg a linestring with no vertices, or a collection with no geom...
QgsRectangle boundingBox() const
Returns the bounding box of the geometry.
static QgsProject * instance()
Returns the QgsProject singleton instance.
Base class for processing filters like renderers, reprojector, resampler etc.
virtual int xSize() const
Gets raster size.
virtual int ySize() const
virtual QgsRectangle extent() const
Gets the extent of the interface.
Represents a raster layer.
A rectangle specified with double values.
QgsRectangle intersect(const QgsRectangle &rect) const
Returns the intersection with the given rectangle.
Base class for vector data providers.
long long featureCount() const override=0
Number of features in the layer.
virtual QString storageType() const
Returns the permanent storage type for this layer as a friendly name.
virtual bool changeAttributeValues(const QgsChangedAttributesMap &attr_map)
Changes attribute values of existing features.
QgsFields fields() const override=0
Returns the fields associated with this data provider.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const override=0
Query the provider for features specified in request.
virtual bool addAttributes(const QList< QgsField > &attributes)
Adds new attributes to the provider.
Represents a vector layer which manages a vector based dataset.
QgsVectorDataProvider * dataProvider() final
Returns the layer's data provider, it may be nullptr.
QgsZonalStatistics(QgsVectorLayer *polygonLayer, QgsRasterLayer *rasterLayer, const QString &attributePrefix=QString(), int rasterBand=1, Qgis::ZonalStatistics stats=Qgis::ZonalStatistic::Default)
Convenience constructor for QgsZonalStatistics, using an input raster layer.
Qgis::ZonalStatisticResult calculateStatistics(QgsFeedback *feedback)
Runs the calculation.
static QString displayName(Qgis::ZonalStatistic statistic)
Returns the friendly display name for a statistic.
static QString shortName(Qgis::ZonalStatistic statistic)
Returns a short, friendly display name for a statistic, suitable for use in a field name.
QMap< int, QVariant > QgsAttributeMap
QMap< QgsFeatureId, QgsAttributeMap > QgsChangedAttributesMap