QGIS API Documentation 3.28.14-Firenze (exported)
Loading...
Searching...
No Matches
qgsalgorithmrasterlayeruniquevalues.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsalgorithmrasterlayeruniquevalues.cpp
3 ---------------------
4 begin : April 2017
5 copyright : (C) 2017 by Mathieu Pellerin
6 email : nirvn dot asia 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#include "qgsstringutils.h"
20#include <QTextStream>
21
23
24QString QgsRasterLayerUniqueValuesReportAlgorithm::name() const
25{
26 return QStringLiteral( "rasterlayeruniquevaluesreport" );
27}
28
29QString QgsRasterLayerUniqueValuesReportAlgorithm::displayName() const
30{
31 return QObject::tr( "Raster layer unique values report" );
32}
33
34QStringList QgsRasterLayerUniqueValuesReportAlgorithm::tags() const
35{
36 return QObject::tr( "count,area,statistics" ).split( ',' );
37}
38
39QString QgsRasterLayerUniqueValuesReportAlgorithm::group() const
40{
41 return QObject::tr( "Raster analysis" );
42}
43
44QString QgsRasterLayerUniqueValuesReportAlgorithm::groupId() const
45{
46 return QStringLiteral( "rasteranalysis" );
47}
48
49void QgsRasterLayerUniqueValuesReportAlgorithm::initAlgorithm( const QVariantMap & )
50{
51 addParameter( new QgsProcessingParameterRasterLayer( QStringLiteral( "INPUT" ),
52 QObject::tr( "Input layer" ) ) );
53 addParameter( new QgsProcessingParameterBand( QStringLiteral( "BAND" ),
54 QObject::tr( "Band number" ), 1, QStringLiteral( "INPUT" ) ) );
55 addParameter( new QgsProcessingParameterFileDestination( QStringLiteral( "OUTPUT_HTML_FILE" ),
56 QObject::tr( "Unique values report" ), QObject::tr( "HTML files (*.html)" ), QVariant(), true ) );
57 addParameter( new QgsProcessingParameterFeatureSink( QStringLiteral( "OUTPUT_TABLE" ),
58 QObject::tr( "Unique values table" ), QgsProcessing::TypeVector, QVariant(), true, false ) );
59
60 addOutput( new QgsProcessingOutputString( QStringLiteral( "EXTENT" ), QObject::tr( "Extent" ) ) );
61 addOutput( new QgsProcessingOutputString( QStringLiteral( "CRS_AUTHID" ), QObject::tr( "CRS authority identifier" ) ) );
62 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "WIDTH_IN_PIXELS" ), QObject::tr( "Width in pixels" ) ) );
63 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "HEIGHT_IN_PIXELS" ), QObject::tr( "Height in pixels" ) ) );
64 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "TOTAL_PIXEL_COUNT" ), QObject::tr( "Total pixel count" ) ) );
65 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "NODATA_PIXEL_COUNT" ), QObject::tr( "NODATA pixel count" ) ) );
66}
67
68QString QgsRasterLayerUniqueValuesReportAlgorithm::shortHelpString() const
69{
70 return QObject::tr( "This algorithm returns the count and area of each unique value in a given raster layer." );
71}
72
73QgsRasterLayerUniqueValuesReportAlgorithm *QgsRasterLayerUniqueValuesReportAlgorithm::createInstance() const
74{
75 return new QgsRasterLayerUniqueValuesReportAlgorithm();
76}
77
78bool QgsRasterLayerUniqueValuesReportAlgorithm::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
79{
80 QgsRasterLayer *layer = parameterAsRasterLayer( parameters, QStringLiteral( "INPUT" ), context );
81 const int band = parameterAsInt( parameters, QStringLiteral( "BAND" ), context );
82
83 if ( !layer )
84 throw QgsProcessingException( invalidRasterError( parameters, QStringLiteral( "INPUT" ) ) );
85
86 mBand = parameterAsInt( parameters, QStringLiteral( "BAND" ), context );
87 if ( mBand < 1 || mBand > layer->bandCount() )
88 throw QgsProcessingException( QObject::tr( "Invalid band number for BAND (%1): Valid values for input raster are 1 to %2" ).arg( mBand )
89 .arg( layer->bandCount() ) );
90
91 mInterface.reset( layer->dataProvider()->clone() );
92 mHasNoDataValue = layer->dataProvider()->sourceHasNoDataValue( band );
93 mLayerWidth = layer->width();
94 mLayerHeight = layer->height();
95 mExtent = layer->extent();
96 mCrs = layer->crs();
97 mRasterUnitsPerPixelX = layer->rasterUnitsPerPixelX();
98 mRasterUnitsPerPixelY = layer->rasterUnitsPerPixelY();
99 mSource = layer->source();
100
101 return true;
102}
103
104QVariantMap QgsRasterLayerUniqueValuesReportAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
105{
106 const QString outputFile = parameterAsFileOutput( parameters, QStringLiteral( "OUTPUT_HTML_FILE" ), context );
107
108 QString areaUnit = QgsUnitTypes::toAbbreviatedString( QgsUnitTypes::distanceToAreaUnit( mCrs.mapUnits() ) );
109
110 QString tableDest;
111 std::unique_ptr< QgsFeatureSink > sink;
112 if ( parameters.contains( QStringLiteral( "OUTPUT_TABLE" ) ) && parameters.value( QStringLiteral( "OUTPUT_TABLE" ) ).isValid() )
113 {
114 QgsFields outFields;
115 outFields.append( QgsField( QStringLiteral( "value" ), QVariant::Double, QString(), 20, 8 ) );
116 outFields.append( QgsField( QStringLiteral( "count" ), QVariant::LongLong, QString(), 20 ) );
117 outFields.append( QgsField( areaUnit.replace( QStringLiteral( "²" ), QStringLiteral( "2" ) ), QVariant::Double, QString(), 20, 8 ) );
118 sink.reset( parameterAsSink( parameters, QStringLiteral( "OUTPUT_TABLE" ), context, tableDest, outFields, QgsWkbTypes::NoGeometry, QgsCoordinateReferenceSystem() ) );
119 if ( !sink )
120 throw QgsProcessingException( invalidSinkError( parameters, QStringLiteral( "OUTPUT_TABLE" ) ) );
121 }
122
123 QHash< double, qgssize > uniqueValues;
124 qgssize noDataCount = 0;
125
126 const qgssize layerSize = static_cast< qgssize >( mLayerWidth ) * static_cast< qgssize >( mLayerHeight );
129 const int nbBlocksWidth = std::ceil( 1.0 * mLayerWidth / maxWidth );
130 const int nbBlocksHeight = std::ceil( 1.0 * mLayerHeight / maxHeight );
131 const int nbBlocks = nbBlocksWidth * nbBlocksHeight;
132
133 QgsRasterIterator iter( mInterface.get() );
134 iter.startRasterRead( mBand, mLayerWidth, mLayerHeight, mExtent );
135
136 int iterLeft = 0;
137 int iterTop = 0;
138 int iterCols = 0;
139 int iterRows = 0;
140 bool isNoData = false;
141 std::unique_ptr< QgsRasterBlock > rasterBlock;
142 while ( iter.readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop ) )
143 {
144 feedback->setProgress( 100 * ( ( iterTop / maxHeight * nbBlocksWidth ) + iterLeft / maxWidth ) / nbBlocks );
145 for ( int row = 0; row < iterRows; row++ )
146 {
147 if ( feedback->isCanceled() )
148 break;
149 for ( int column = 0; column < iterCols; column++ )
150 {
151 const double value = rasterBlock->valueAndNoData( row, column, isNoData );
152 if ( mHasNoDataValue && isNoData )
153 {
154 noDataCount++;
155 }
156 else
157 {
158 uniqueValues[ value ]++;
159 }
160 }
161 }
162 if ( feedback->isCanceled() )
163 break;
164 }
165
166 QMap< double, qgssize > sortedUniqueValues;
167 for ( auto it = uniqueValues.constBegin(); it != uniqueValues.constEnd(); ++it )
168 {
169 sortedUniqueValues.insert( it.key(), it.value() );
170 }
171
172 QVariantMap outputs;
173 outputs.insert( QStringLiteral( "EXTENT" ), mExtent.toString() );
174 outputs.insert( QStringLiteral( "CRS_AUTHID" ), mCrs.authid() );
175 outputs.insert( QStringLiteral( "WIDTH_IN_PIXELS" ), mLayerWidth );
176 outputs.insert( QStringLiteral( "HEIGHT_IN_PIXELS" ), mLayerHeight );
177 outputs.insert( QStringLiteral( "TOTAL_PIXEL_COUNT" ), layerSize );
178 outputs.insert( QStringLiteral( "NODATA_PIXEL_COUNT" ), noDataCount );
179
180 const double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
181
182 if ( !outputFile.isEmpty() )
183 {
184 QFile file( outputFile );
185 if ( file.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
186 {
187 const QString encodedAreaUnit = QgsStringUtils::ampersandEncode( areaUnit );
188
189 QTextStream out( &file );
190#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
191 out.setCodec( "UTF-8" );
192#endif
193 out << QStringLiteral( "<html><head><meta http-equiv=\"Content-Type\" content=\"text/html;charset=utf-8\"/></head><body>\n" );
194 out << QStringLiteral( "<p>%1: %2 (%3 %4)</p>\n" ).arg( QObject::tr( "Analyzed file" ), mSource, QObject::tr( "band" ) ).arg( mBand );
195 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Extent" ), mExtent.toString() );
196 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Projection" ), mCrs.userFriendlyIdentifier() );
197 out << QObject::tr( "<p>%1: %2 (%3 %4)</p>\n" ).arg( QObject::tr( "Width in pixels" ) ).arg( mLayerWidth ).arg( QObject::tr( "units per pixel" ) ).arg( mRasterUnitsPerPixelX );
198 out << QObject::tr( "<p>%1: %2 (%3 %4)</p>\n" ).arg( QObject::tr( "Height in pixels" ) ).arg( mLayerHeight ).arg( QObject::tr( "units per pixel" ) ).arg( mRasterUnitsPerPixelY );
199 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Total pixel count" ) ).arg( layerSize );
200 if ( mHasNoDataValue )
201 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "NODATA pixel count" ) ).arg( noDataCount );
202 out << QStringLiteral( "<table><tr><td>%1</td><td>%2</td><td>%3 (%4)</td></tr>\n" ).arg( QObject::tr( "Value" ), QObject::tr( "Pixel count" ), QObject::tr( "Area" ), encodedAreaUnit );
203
204 for ( auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
205 {
206 const double area = it.value() * pixelArea;
207 out << QStringLiteral( "<tr><td>%1</td><td>%2</td><td>%3</td></tr>\n" ).arg( it.key() ).arg( it.value() ).arg( QString::number( area, 'g', 16 ) );
208 }
209 out << QStringLiteral( "</table>\n</body></html>" );
210 outputs.insert( QStringLiteral( "OUTPUT_HTML_FILE" ), outputFile );
211 }
212 }
213
214 if ( sink )
215 {
216 for ( auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
217 {
218 QgsFeature f;
219 const double area = it.value() * pixelArea;
220 f.setAttributes( QgsAttributes() << it.key() << it.value() << area );
221 if ( !sink->addFeature( f, QgsFeatureSink::FastInsert ) )
222 throw QgsProcessingException( writeFeatureError( sink.get(), parameters, QStringLiteral( "OUTPUT_TABLE" ) ) );
223 }
224 outputs.insert( QStringLiteral( "OUTPUT_TABLE" ), tableDest );
225 }
226
227 return outputs;
228}
229
230
232
233
234
A vector of attributes.
This class represents a coordinate reference system (CRS).
@ FastInsert
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:56
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:54
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:51
Container of fields for a vector layer.
Definition qgsfields.h:45
bool append(const QgsField &field, FieldOrigin origin=OriginProvider, int originIndex=-1)
Appends a field. The field must have unique name, otherwise it is rejected (returns false)
Definition qgsfields.cpp:59
virtual QgsRectangle extent() const
Returns the extent of the layer.
QString source() const
Returns the source for the layer.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:79
Contains information about the context in which a processing algorithm is executed.
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 generic file based destination parameter, for specifying the destination path for a file (non-map l...
A raster layer parameter for processing algorithms.
@ TypeVector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
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.
static const int DEFAULT_MAXIMUM_TILE_WIDTH
Default maximum tile width.
static const int DEFAULT_MAXIMUM_TILE_HEIGHT
Default maximum tile height.
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.
static QString ampersandEncode(const QString &string)
Makes a raw string safe for inclusion as a HTML/XML string literal.
static Q_INVOKABLE QgsUnitTypes::AreaUnit distanceToAreaUnit(QgsUnitTypes::DistanceUnit distanceUnit)
Converts a distance unit to its corresponding area unit, e.g., meters to square meters.
static Q_INVOKABLE QString toAbbreviatedString(QgsUnitTypes::DistanceUnit unit)
Returns a translated abbreviation representing a distance unit.
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:3032