QGIS API Documentation 3.41.0-Master (af5edcb665c)
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 "qgsunittypes.h"
21
22#include <QTextStream>
23
25
26QString QgsRasterLayerUniqueValuesReportAlgorithm::name() const
27{
28 return QStringLiteral( "rasterlayeruniquevaluesreport" );
29}
30
31QString QgsRasterLayerUniqueValuesReportAlgorithm::displayName() const
32{
33 return QObject::tr( "Raster layer unique values report" );
34}
35
36QStringList QgsRasterLayerUniqueValuesReportAlgorithm::tags() const
37{
38 return QObject::tr( "count,area,statistics" ).split( ',' );
39}
40
41QString QgsRasterLayerUniqueValuesReportAlgorithm::group() const
42{
43 return QObject::tr( "Raster analysis" );
44}
45
46QString QgsRasterLayerUniqueValuesReportAlgorithm::groupId() const
47{
48 return QStringLiteral( "rasteranalysis" );
49}
50
51void QgsRasterLayerUniqueValuesReportAlgorithm::initAlgorithm( const QVariantMap & )
52{
53 addParameter( new QgsProcessingParameterRasterLayer( QStringLiteral( "INPUT" ), QObject::tr( "Input layer" ) ) );
54 addParameter( new QgsProcessingParameterBand( QStringLiteral( "BAND" ), QObject::tr( "Band number" ), 1, QStringLiteral( "INPUT" ) ) );
55 addParameter( new QgsProcessingParameterFileDestination( QStringLiteral( "OUTPUT_HTML_FILE" ), QObject::tr( "Unique values report" ), QObject::tr( "HTML files (*.html)" ), QVariant(), true ) );
56 addParameter( new QgsProcessingParameterFeatureSink( QStringLiteral( "OUTPUT_TABLE" ), QObject::tr( "Unique values table" ), Qgis::ProcessingSourceType::Vector, QVariant(), true, false ) );
57
58 addOutput( new QgsProcessingOutputString( QStringLiteral( "EXTENT" ), QObject::tr( "Extent" ) ) );
59 addOutput( new QgsProcessingOutputString( QStringLiteral( "CRS_AUTHID" ), QObject::tr( "CRS authority identifier" ) ) );
60 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "WIDTH_IN_PIXELS" ), QObject::tr( "Width in pixels" ) ) );
61 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "HEIGHT_IN_PIXELS" ), QObject::tr( "Height in pixels" ) ) );
62 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "TOTAL_PIXEL_COUNT" ), QObject::tr( "Total pixel count" ) ) );
63 addOutput( new QgsProcessingOutputNumber( QStringLiteral( "NODATA_PIXEL_COUNT" ), QObject::tr( "NoData pixel count" ) ) );
64}
65
66QString QgsRasterLayerUniqueValuesReportAlgorithm::shortHelpString() const
67{
68 return QObject::tr( "This algorithm returns the count and area of each unique value in a given raster layer. "
69 "The area calculation is done in the area unit of the layer's CRS." );
70}
71
72QgsRasterLayerUniqueValuesReportAlgorithm *QgsRasterLayerUniqueValuesReportAlgorithm::createInstance() const
73{
74 return new QgsRasterLayerUniqueValuesReportAlgorithm();
75}
76
77bool QgsRasterLayerUniqueValuesReportAlgorithm::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
78{
79 QgsRasterLayer *layer = parameterAsRasterLayer( parameters, QStringLiteral( "INPUT" ), context );
80 const int band = parameterAsInt( parameters, QStringLiteral( "BAND" ), context );
81
82 if ( !layer )
83 throw QgsProcessingException( invalidRasterError( parameters, QStringLiteral( "INPUT" ) ) );
84
85 mBand = parameterAsInt( parameters, QStringLiteral( "BAND" ), context );
86 if ( mBand < 1 || mBand > layer->bandCount() )
87 throw QgsProcessingException( QObject::tr( "Invalid band number for BAND (%1): Valid values for input raster are 1 to %2" ).arg( mBand ).arg( layer->bandCount() ) );
88
89 mInterface.reset( layer->dataProvider()->clone() );
90 mHasNoDataValue = layer->dataProvider()->sourceHasNoDataValue( band );
91 mLayerWidth = layer->width();
92 mLayerHeight = layer->height();
93 mExtent = layer->extent();
94 mCrs = layer->crs();
95 mRasterUnitsPerPixelX = layer->rasterUnitsPerPixelX();
96 mRasterUnitsPerPixelY = layer->rasterUnitsPerPixelY();
97 mSource = layer->source();
98
99 return true;
100}
101
102QVariantMap QgsRasterLayerUniqueValuesReportAlgorithm::processAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
103{
104 const QString outputFile = parameterAsFileOutput( parameters, QStringLiteral( "OUTPUT_HTML_FILE" ), context );
105
106 QString areaUnit = QgsUnitTypes::toAbbreviatedString( QgsUnitTypes::distanceToAreaUnit( mCrs.mapUnits() ) );
107
108 QString tableDest;
109 std::unique_ptr<QgsFeatureSink> sink;
110 if ( parameters.contains( QStringLiteral( "OUTPUT_TABLE" ) ) && parameters.value( QStringLiteral( "OUTPUT_TABLE" ) ).isValid() )
111 {
112 QgsFields outFields;
113 outFields.append( QgsField( QStringLiteral( "value" ), QMetaType::Type::Double, QString(), 20, 8 ) );
114 outFields.append( QgsField( QStringLiteral( "count" ), QMetaType::Type::LongLong, QString(), 20 ) );
115 outFields.append( QgsField( areaUnit.replace( QStringLiteral( "²" ), QStringLiteral( "2" ) ), QMetaType::Type::Double, QString(), 20, 8 ) );
116 sink.reset( parameterAsSink( parameters, QStringLiteral( "OUTPUT_TABLE" ), context, tableDest, outFields, Qgis::WkbType::NoGeometry, QgsCoordinateReferenceSystem() ) );
117 if ( !sink )
118 throw QgsProcessingException( invalidSinkError( parameters, QStringLiteral( "OUTPUT_TABLE" ) ) );
119 }
120
121 QHash<double, qgssize> uniqueValues;
122 qgssize noDataCount = 0;
123
124 const qgssize layerSize = static_cast<qgssize>( mLayerWidth ) * static_cast<qgssize>( mLayerHeight );
127 const int nbBlocksWidth = std::ceil( 1.0 * mLayerWidth / maxWidth );
128 const int nbBlocksHeight = std::ceil( 1.0 * mLayerHeight / maxHeight );
129 const int nbBlocks = nbBlocksWidth * nbBlocksHeight;
130
131 QgsRasterIterator iter( mInterface.get() );
132 iter.startRasterRead( mBand, mLayerWidth, mLayerHeight, mExtent );
133
134 int iterLeft = 0;
135 int iterTop = 0;
136 int iterCols = 0;
137 int iterRows = 0;
138 bool isNoData = false;
139 std::unique_ptr<QgsRasterBlock> rasterBlock;
140 while ( iter.readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop ) )
141 {
142 feedback->setProgress( 100 * ( ( iterTop / maxHeight * nbBlocksWidth ) + iterLeft / maxWidth ) / nbBlocks );
143 for ( int row = 0; row < iterRows; row++ )
144 {
145 if ( feedback->isCanceled() )
146 break;
147 for ( int column = 0; column < iterCols; column++ )
148 {
149 const double value = rasterBlock->valueAndNoData( row, column, isNoData );
150 if ( mHasNoDataValue && isNoData )
151 {
152 noDataCount++;
153 }
154 else
155 {
156 uniqueValues[value]++;
157 }
158 }
159 }
160 if ( feedback->isCanceled() )
161 break;
162 }
163
164 QMap<double, qgssize> sortedUniqueValues;
165 for ( auto it = uniqueValues.constBegin(); it != uniqueValues.constEnd(); ++it )
166 {
167 sortedUniqueValues.insert( it.key(), it.value() );
168 }
169
170 QVariantMap outputs;
171 outputs.insert( QStringLiteral( "EXTENT" ), mExtent.toString() );
172 outputs.insert( QStringLiteral( "CRS_AUTHID" ), mCrs.authid() );
173 outputs.insert( QStringLiteral( "WIDTH_IN_PIXELS" ), mLayerWidth );
174 outputs.insert( QStringLiteral( "HEIGHT_IN_PIXELS" ), mLayerHeight );
175 outputs.insert( QStringLiteral( "TOTAL_PIXEL_COUNT" ), layerSize );
176 outputs.insert( QStringLiteral( "NODATA_PIXEL_COUNT" ), noDataCount );
177
178 const double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
179
180 if ( !outputFile.isEmpty() )
181 {
182 QFile file( outputFile );
183 if ( file.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
184 {
185 const QString encodedAreaUnit = QgsStringUtils::ampersandEncode( areaUnit );
186
187 QTextStream out( &file );
188#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
189 out.setCodec( "UTF-8" );
190#endif
191 out << QStringLiteral( "<html><head><meta http-equiv=\"Content-Type\" content=\"text/html;charset=utf-8\"/></head><body>\n" );
192 out << QStringLiteral( "<p>%1: %2 (%3 %4)</p>\n" ).arg( QObject::tr( "Analyzed file" ), mSource, QObject::tr( "band" ) ).arg( mBand );
193 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Extent" ), mExtent.toString() );
194 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Projection" ), mCrs.userFriendlyIdentifier() );
195 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 );
196 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 );
197 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "Total pixel count" ) ).arg( layerSize );
198 if ( mHasNoDataValue )
199 out << QObject::tr( "<p>%1: %2</p>\n" ).arg( QObject::tr( "NoData pixel count" ) ).arg( noDataCount );
200 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 );
201
202 for ( auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
203 {
204 const double area = it.value() * pixelArea;
205 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 ) );
206 }
207 out << QStringLiteral( "</table>\n</body></html>" );
208 outputs.insert( QStringLiteral( "OUTPUT_HTML_FILE" ), outputFile );
209 }
210 }
211
212 if ( sink )
213 {
214 for ( auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
215 {
216 QgsFeature f;
217 const double area = it.value() * pixelArea;
218 f.setAttributes( QgsAttributes() << it.key() << it.value() << area );
219 if ( !sink->addFeature( f, QgsFeatureSink::FastInsert ) )
220 throw QgsProcessingException( writeFeatureError( sink.get(), parameters, QStringLiteral( "OUTPUT_TABLE" ) ) );
221 }
222 sink->finalize();
223 outputs.insert( QStringLiteral( "OUTPUT_TABLE" ), tableDest );
224 }
225
226 return outputs;
227}
228
229
@ Vector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
@ NoGeometry
No geometry.
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:58
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:53
void setProgress(double progress)
Sets the current progress for the feedback object.
Definition qgsfeedback.h:61
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:53
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:70
virtual QgsRectangle extent() const
Returns the extent of the layer.
QString source() const
Returns the source for the layer.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:83
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.
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 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:6614