20 #include <QTextStream> 
   24 QString QgsRasterLayerUniqueValuesReportAlgorithm::name()
 const 
   26   return QStringLiteral( 
"rasterlayeruniquevaluesreport" );
 
   29 QString QgsRasterLayerUniqueValuesReportAlgorithm::displayName()
 const 
   31   return QObject::tr( 
"Raster layer unique values report" );
 
   34 QStringList QgsRasterLayerUniqueValuesReportAlgorithm::tags()
 const 
   36   return QObject::tr( 
"count,area,statistics" ).split( 
',' );
 
   39 QString QgsRasterLayerUniqueValuesReportAlgorithm::group()
 const 
   41   return QObject::tr( 
"Raster analysis" );
 
   44 QString QgsRasterLayerUniqueValuesReportAlgorithm::groupId()
 const 
   46   return QStringLiteral( 
"rasteranalysis" );
 
   49 void QgsRasterLayerUniqueValuesReportAlgorithm::initAlgorithm( 
const QVariantMap & )
 
   52                 QObject::tr( 
"Input layer" ) ) );
 
   54                 QObject::tr( 
"Band number" ), 1, QStringLiteral( 
"INPUT" ) ) );
 
   56                 QObject::tr( 
"Unique values report" ), QObject::tr( 
"HTML files (*.html)" ), QVariant(), 
true ) );
 
   65   addOutput( 
new QgsProcessingOutputNumber( QStringLiteral( 
"NODATA_PIXEL_COUNT" ), QObject::tr( 
"NODATA pixel count" ) ) );
 
   68 QString QgsRasterLayerUniqueValuesReportAlgorithm::shortHelpString()
 const 
   70   return QObject::tr( 
"This algorithm returns the count and area of each unique value in a given raster layer." );
 
   73 QgsRasterLayerUniqueValuesReportAlgorithm *QgsRasterLayerUniqueValuesReportAlgorithm::createInstance()
 const 
   75   return new QgsRasterLayerUniqueValuesReportAlgorithm();
 
   80   QgsRasterLayer *layer = parameterAsRasterLayer( parameters, QStringLiteral( 
"INPUT" ), context );
 
   81   int band = parameterAsInt( parameters, QStringLiteral( 
"BAND" ), context );
 
   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 )
 
   93   mLayerWidth = layer->
width();
 
   94   mLayerHeight = layer->
height();
 
  106   QString outputFile = parameterAsFileOutput( parameters, QStringLiteral( 
"OUTPUT_HTML_FILE" ), context );
 
  111   std::unique_ptr< QgsFeatureSink > sink;
 
  112   if ( parameters.contains( QStringLiteral( 
"OUTPUT_TABLE" ) ) && parameters.value( QStringLiteral( 
"OUTPUT_TABLE" ) ).isValid() )
 
  115     outFields.
append( 
QgsField( QStringLiteral( 
"value" ), QVariant::Double, QString(), 20, 8 ) );
 
  116     outFields.
append( 
QgsField( QStringLiteral( 
"count" ), QVariant::Int, QString(), 20 ) );
 
  117     outFields.
append( 
QgsField( areaUnit.replace( QStringLiteral( 
"²" ), QStringLiteral( 
"2" ) ), QVariant::Double, QString(), 20, 8 ) );
 
  123   QHash< double, qgssize > uniqueValues;
 
  129   int nbBlocksWidth = std::ceil( 1.0 * mLayerWidth / maxWidth );
 
  130   int nbBlocksHeight = std::ceil( 1.0 * mLayerHeight / maxHeight );
 
  131   int nbBlocks = nbBlocksWidth * nbBlocksHeight;
 
  134   iter.startRasterRead( mBand, mLayerWidth, mLayerHeight, mExtent );
 
  140   bool isNoData = 
false;
 
  141   std::unique_ptr< QgsRasterBlock > rasterBlock;
 
  142   while ( iter.readNextRasterPart( mBand, iterCols, iterRows, rasterBlock, iterLeft, iterTop ) )
 
  144     feedback->
setProgress( 100 * ( ( iterTop / maxHeight * nbBlocksWidth ) + iterLeft / maxWidth ) / nbBlocks );
 
  145     for ( 
int row = 0; row < iterRows; row++ )
 
  149       for ( 
int column = 0; column < iterCols; column++ )
 
  151         double value = rasterBlock->valueAndNoData( row, column, isNoData );
 
  152         if ( mHasNoDataValue && isNoData )
 
  158           uniqueValues[ value ]++;
 
  164   QMap< double, qgssize > sortedUniqueValues;
 
  165   for ( 
auto it = uniqueValues.constBegin(); it != uniqueValues.constEnd(); ++it )
 
  167     sortedUniqueValues.insert( it.key(), it.value() );
 
  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 );
 
  178   double pixelArea = mRasterUnitsPerPixelX * mRasterUnitsPerPixelY;
 
  180   if ( !outputFile.isEmpty() )
 
  182     QFile file( outputFile );
 
  183     if ( file.open( QIODevice::WriteOnly | QIODevice::Truncate ) )
 
  187       QTextStream out( &file );
 
  188 #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) 
  189       out.setCodec( 
"UTF-8" );
 
  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 );
 
  202       for ( 
auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
 
  204         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 ) );
 
  207       out << QStringLiteral( 
"</table>\n</body></html>" );
 
  208       outputs.insert( QStringLiteral( 
"OUTPUT_HTML_FILE" ), outputFile );
 
  214     for ( 
auto it = sortedUniqueValues.constBegin(); it != sortedUniqueValues.constEnd(); ++it )
 
  217       double area = it.value() * pixelArea;
 
  221     outputs.insert( QStringLiteral( 
"OUTPUT_TABLE" ), tableDest );
 
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...
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
bool isCanceled() const SIP_HOLDGIL
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, FieldOrigin origin=OriginProvider, int originIndex=-1)
Appends a field. The field must have unique name, otherwise it is rejected (returns false)
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.
@ TypeVector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
virtual bool sourceHasNoDataValue(int bandNo) const
Returns true if source band has no data value.
QgsRasterDataProvider * clone() const override=0
Clone itself, create deep copy.
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...