26using namespace Qt::StringLiterals;
30QString QgsRasterLayerUniqueValuesReportAlgorithm::name()
const
32 return u
"rasterlayeruniquevaluesreport"_s;
35QString QgsRasterLayerUniqueValuesReportAlgorithm::displayName()
const
37 return QObject::tr(
"Raster layer unique values report" );
40QStringList QgsRasterLayerUniqueValuesReportAlgorithm::tags()
const
42 return QObject::tr(
"count,area,statistics" ).split(
',' );
45QString QgsRasterLayerUniqueValuesReportAlgorithm::group()
const
47 return QObject::tr(
"Raster analysis" );
50QString QgsRasterLayerUniqueValuesReportAlgorithm::groupId()
const
52 return u
"rasteranalysis"_s;
55void QgsRasterLayerUniqueValuesReportAlgorithm::initAlgorithm(
const QVariantMap & )
70QString QgsRasterLayerUniqueValuesReportAlgorithm::shortHelpString()
const
73 "This algorithm returns the count and area of each unique value in a given raster layer. "
74 "The area calculation is done in the area unit of the layer's CRS."
78QString QgsRasterLayerUniqueValuesReportAlgorithm::shortDescription()
const
80 return QObject::tr(
"Returns the count and area of each unique value in a given raster layer." );
83QgsRasterLayerUniqueValuesReportAlgorithm *QgsRasterLayerUniqueValuesReportAlgorithm::createInstance()
const
85 return new QgsRasterLayerUniqueValuesReportAlgorithm();
90 QgsRasterLayer *layer = parameterAsRasterLayer( parameters, u
"INPUT"_s, context );
91 const int band = parameterAsInt( parameters, u
"BAND"_s, context );
96 mBand = parameterAsInt( parameters, u
"BAND"_s, context );
97 if ( mBand < 1 || mBand > layer->
bandCount() )
98 throw QgsProcessingException( QObject::tr(
"Invalid band number for BAND (%1): Valid values for input raster are 1 to %2" ).arg( mBand ).arg( layer->
bandCount() ) );
102 mLayerWidth = layer->
width();
103 mLayerHeight = layer->
height();
104 mExtent = layer->
extent();
108 mSource = layer->
source();
115 const QString outputFile = parameterAsFileOutput( parameters, u
"OUTPUT_HTML_FILE"_s, context );
120 std::unique_ptr<QgsFeatureSink> sink;
121 if ( parameters.contains( u
"OUTPUT_TABLE"_s ) && parameters.value( u
"OUTPUT_TABLE"_s ).isValid() )
124 outFields.
append(
QgsField( u
"value"_s, QMetaType::Type::Double, QString(), 20, 8 ) );
125 outFields.
append(
QgsField( u
"count"_s, QMetaType::Type::LongLong, QString(), 20 ) );
126 outFields.
append(
QgsField( areaUnit.replace( u
"²"_s,
"2"_L1 ), QMetaType::Type::Double, QString(), 20, 8 ) );
132 QHash<double, qgssize> uniqueValues;
135 const qgssize layerSize =
static_cast<qgssize>( mLayerWidth ) *
static_cast<qgssize>( mLayerHeight );
138 iter.startRasterRead( mBand, mLayerWidth, mLayerHeight, mExtent );
144 bool isNoData =
false;
145 std::unique_ptr<QgsRasterBlock> rasterBlock;
146 while ( iter.readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop ) )
148 feedback->
setProgress( 100 * iter.progress( mBand ) );
149 for (
int row = 0; row < iterRows; row++ )
153 for (
int column = 0; column < iterCols; column++ )
155 const double value = rasterBlock->valueAndNoData( row, column, isNoData );
156 if ( mHasNoDataValue && isNoData )
162 uniqueValues[value]++;
170 QMap<double, qgssize> sortedUniqueValues;
171 for (
auto it = uniqueValues.constBegin(); it != uniqueValues.constEnd(); ++it )
173 sortedUniqueValues.insert( it.key(), it.value() );
177 outputs.insert( u
"EXTENT"_s, mExtent.toString() );
178 outputs.insert( u
"CRS_AUTHID"_s, mCrs.authid() );
179 outputs.insert( u
"WIDTH_IN_PIXELS"_s, mLayerWidth );
180 outputs.insert( u
"HEIGHT_IN_PIXELS"_s, mLayerHeight );
181 outputs.insert( u
"TOTAL_PIXEL_COUNT"_s, layerSize );
182 outputs.insert( u
"NODATA_PIXEL_COUNT"_s, noDataCount );
184 const double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
186 if ( !outputFile.isEmpty() )
188 QFile file( outputFile );
189 if ( file.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
193 QTextStream out( &file );
194 out << u
"<html><head><meta http-equiv=\"Content-Type\" content=\"text/html;charset=utf-8\"/></head><body>\n"_s;
195 out << u
"<p>%1: %2 (%3 %4)</p>\n"_s.arg( QObject::tr(
"Analyzed file" ), mSource, QObject::tr(
"band" ) ).arg( mBand );
196 out << QObject::tr(
"<p>%1: %2</p>\n" ).arg( QObject::tr(
"Extent" ), mExtent.toString() );
197 out << QObject::tr(
"<p>%1: %2</p>\n" ).arg( QObject::tr(
"Projection" ), mCrs.userFriendlyIdentifier() );
198 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 );
199 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 );
200 out << QObject::tr(
"<p>%1: %2</p>\n" ).arg( QObject::tr(
"Total pixel count" ) ).arg( layerSize );
201 if ( mHasNoDataValue )
202 out << QObject::tr(
"<p>%1: %2</p>\n" ).arg( QObject::tr(
"NoData pixel count" ) ).arg( noDataCount );
203 out << u
"<table><tr><td>%1</td><td>%2</td><td>%3 (%4)</td></tr>\n"_s.arg( QObject::tr(
"Value" ), QObject::tr(
"Pixel count" ), QObject::tr(
"Area" ), encodedAreaUnit );
205 for (
auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
207 const double area = it.value() * pixelArea;
208 out << u
"<tr><td>%1</td><td>%2</td><td>%3</td></tr>\n"_s.arg( it.key() ).arg( it.value() ).arg( QString::number( area,
'g', 16 ) );
210 out << u
"</table>\n</body></html>"_s;
211 outputs.insert( u
"OUTPUT_HTML_FILE"_s, outputFile );
217 for (
auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
220 const double area = it.value() * pixelArea;
226 outputs.insert( u
"OUTPUT_TABLE"_s, tableDest );
@ Vector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
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...
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
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.
Container of fields for a vector layer.
bool append(const QgsField &field, Qgis::FieldOrigin origin=Qgis::FieldOrigin::Provider, int originIndex=-1)
Appends a field.
virtual QgsRectangle extent() const
Returns the extent of the layer.
QString source() const
Returns the source for the layer.
QgsCoordinateReferenceSystem crs
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.
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...