32using namespace Qt::StringLiterals;
37 rasterLayer ? rasterLayer->dataProvider() : nullptr,
39 rasterLayer ? rasterLayer->rasterUnitsPerPixelX() : 0,
40 rasterLayer ? rasterLayer->rasterUnitsPerPixelY() : 0,
51 double rasterUnitsPerPixelX,
52 double rasterUnitsPerPixelY,
53 const QString &attributePrefix,
57 : mRasterInterface( rasterInterface )
58 , mRasterCrs( rasterCrs )
59 , mCellSizeX( std::fabs( rasterUnitsPerPixelX ) )
60 , mCellSizeY( std::fabs( rasterUnitsPerPixelY ) )
61 , mRasterBand( rasterBand )
62 , mPolygonLayer( polygonLayer )
63 , mAttributePrefix( attributePrefix )
64 , mStatistics( stats )
69 if ( !mRasterInterface )
74 if ( mRasterInterface->bandCount() < mRasterBand )
85 if ( !vectorProvider )
90 QMap<Qgis::ZonalStatistic, int> statFieldIndexes;
93 int oldFieldCount = vectorProvider->
fields().
count();
94 QList<QgsField> newFieldList;
109 if ( mStatistics & stat )
112 QgsField field( fieldName, QMetaType::Type::Double, u
"double precision"_s );
113 newFieldList.push_back( field );
114 statFieldIndexes.insert( stat, oldFieldCount + newFieldList.count() - 1 );
129 int featureCounter = 0;
142 feedback->
setProgress( 100.0 *
static_cast<double>( featureCounter ) / featureCount );
147 QMap<Qgis::ZonalStatistic, QVariant> results =
calculateStatistics( mRasterInterface, featureGeometry, mCellSizeX, mCellSizeY, mRasterBand, mStatistics );
149 if ( results.empty() )
153 for (
const auto &result : results.toStdMap() )
155 changeAttributeMap.insert( statFieldIndexes.value( result.first ), result.second );
158 changeMap.insert( feature.
id(), changeAttributeMap );
162 mPolygonLayer->updateFields();
175QString QgsZonalStatistics::getUniqueFieldName(
const QString &fieldName,
const QList<QgsField> &newFields )
179 if ( !dp->
storageType().contains(
"ESRI Shapefile"_L1 ) )
185 allFields.append( newFields );
186 QString
shortName = fieldName.mid( 0, 10 );
189 for (
const QgsField &field : std::as_const( allFields ) )
204 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 8 ) ).arg( n );
209 for (
const QgsField &field : std::as_const( allFields ) )
216 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 8 ) ).arg( n );
220 shortName = u
"%1_%2"_s.arg( fieldName.mid( 0, 7 ) ).arg( n );
234 return QObject::tr(
"Count" );
236 return QObject::tr(
"Sum" );
238 return QObject::tr(
"Mean" );
240 return QObject::tr(
"Median" );
242 return QObject::tr(
"St dev" );
244 return QObject::tr(
"Minimum" );
246 return QObject::tr(
"Maximum" );
248 return QObject::tr(
"Minimum point" );
250 return QObject::tr(
"Maximum point" );
252 return QObject::tr(
"Range" );
254 return QObject::tr(
"Minority" );
256 return QObject::tr(
"Majority" );
258 return QObject::tr(
"Variety" );
260 return QObject::tr(
"Variance" );
287 return u
"minpoint"_s;
289 return u
"maxpoint"_s;
293 return u
"minority"_s;
295 return u
"majority"_s;
299 return u
"variance"_s;
308QMap<int, QVariant> QgsZonalStatistics::calculateStatisticsInt(
313 QMap<int, QVariant> pyResult;
314 for (
auto it = result.constBegin(); it != result.constEnd(); ++it )
316 pyResult.insert(
static_cast<int>( it.key() ), it.value() );
326 QMap<Qgis::ZonalStatistic, QVariant> results;
341 FeatureStats featureStats( statsStoreValues, statsStoreValueCount );
343 int nCellsXProvider = rasterInterface->
xSize();
344 int nCellsYProvider = rasterInterface->
ySize();
346 int nCellsX, nCellsY;
348 QgsRasterAnalysisUtils::cellInfoForBBox( rasterBBox, featureRect, cellSizeX, cellSizeY, nCellsX, nCellsY, nCellsXProvider, nCellsYProvider, rasterBlockExtent );
350 featureStats.reset();
351 QgsRasterAnalysisUtils::statisticsFromMiddlePointTest( rasterInterface, rasterBand, geometry, nCellsX, nCellsY, cellSizeX, cellSizeY, rasterBlockExtent, [&featureStats](
double value,
const QgsPointXY &point ) {
352 featureStats.addValue( value, point );
355 if ( featureStats.count <= 1 )
358 featureStats.reset();
359 QgsRasterAnalysisUtils::
360 statisticsFromPreciseIntersection( rasterInterface, rasterBand, geometry, nCellsX, nCellsY, cellSizeX, cellSizeY, rasterBlockExtent, [&featureStats](
double value,
double weight,
const QgsPointXY &point ) {
361 featureStats.addValue( value, point, weight );
371 if ( featureStats.count > 0 )
373 double mean = featureStats.sum / featureStats.count;
378 std::sort( featureStats.values.begin(), featureStats.values.end() );
379 int size = featureStats.values.count();
380 bool even = ( size % 2 ) < 1;
384 medianValue = ( featureStats.values.at( size / 2 - 1 ) + featureStats.values.at( size / 2 ) ) / 2;
388 medianValue = featureStats.values.at( ( size + 1 ) / 2 - 1 );
394 double sumSquared = 0;
395 for (
int i = 0; i < featureStats.values.count(); ++i )
397 double diff = featureStats.values.at( i ) - mean;
398 sumSquared += diff * diff;
400 double variance = sumSquared / featureStats.values.count();
403 double stdev = std::pow( variance, 0.5 );
421 QList<int> vals = featureStats.valueCount.values();
422 std::sort( vals.begin(), vals.end() );
425 double minorityKey = featureStats.valueCount.key( vals.first() );
430 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