22#include <QDomDocument>
26using namespace Qt::StringLiterals;
30 , mColorizeColor( QColor::fromRgb( 255, 128, 128 ) )
56 return mInput->bandCount();
71 return mInput->dataType( bandNo );
96 if (
input->bandCount() < 1 )
117 QgsDebugMsgLevel( u
"width = %1 height = %2 extent = %3"_s.arg( width ).arg( height ).arg(
extent.toString() ), 4 );
126 std::unique_ptr< QgsRasterBlock > inputBlock(
mInput->block( bandNumber,
extent, width, height, feedback ) );
127 if ( !inputBlock || inputBlock->isEmpty() )
133 if ( !mInvertColors && mSaturation == 0 && mGrayscaleMode ==
GrayscaleOff && !mColorizeOn )
136 return inputBlock.release();
139 auto outputBlock = std::make_unique<QgsRasterBlock>();
147 const QRgb myNoDataColor = qRgba( 0, 0, 0, 0 );
150 double alphaFactor = 1.0;
152 const QRgb *inputColorData = inputBlock->colorData();
153 const int imageHeight = inputBlock->image().height();
154 const int imageWidth = inputBlock->image().width();
156 QRgb *outputColorData = outputBlock->colorData();
158 for (
int row = 0; row < height; ++row )
163 for (
int col = 0; col < width; ++col )
167 if ( !inputColorData || row >= imageHeight || col >= imageWidth || inputColorData[i] == myNoDataColor )
169 outputColorData[i] = myNoDataColor;
173 const QRgb inputColor = inputColorData[i];
174 QColor myColor = QColor( inputColor );
177 const int alpha = qAlpha( inputColor );
182 outputColorData[i] = inputColor;
187 myColor.getRgb( &r, &g, &b );
189 if ( mInvertColors || alpha != 255 )
195 alphaFactor = alpha / 255.;
206 myColor = QColor::fromRgb( r, g, b );
209 myColor.getHsl( &h, &s, &l );
212 if ( ( mGrayscaleMode !=
GrayscaleOff ) || ( mSaturationScale != 1 ) )
214 processSaturation( r, g, b, h, s, l );
220 processColorization( r, g, b, h, s, l );
232 outputColorData[i] = qRgba( r, g, b, alpha );
236 return outputBlock.release();
240void QgsHueSaturationFilter::processColorization(
int &r,
int &g,
int &b,
int &h,
int &s,
int &l )
const
249 QColor colorizedColor = QColor::fromHsl( h, s, l );
251 if ( mColorizeStrength == 100 )
254 myColor = colorizedColor;
257 myColor.getRgb( &r, &g, &b );
262 int colorizedR, colorizedG, colorizedB;
263 colorizedColor.getRgb( &colorizedR, &colorizedG, &colorizedB );
266 double p = ( double ) mColorizeStrength / 100.;
267 r = p * colorizedR + ( 1 - p ) * r;
268 g = p * colorizedG + ( 1 - p ) * g;
269 b = p * colorizedB + ( 1 - p ) * b;
272 myColor = QColor::fromRgb( r, g, b );
273 myColor.getHsl( &h, &s, &l );
278void QgsHueSaturationFilter::processSaturation(
int &r,
int &g,
int &b,
int &h,
int &s,
int &l )
const
284 switch ( mGrayscaleMode )
292 myColor = QColor::fromHsl( h, s, l );
293 myColor.getRgb( &r, &g, &b );
299 int luminosity = 0.21 * r + 0.72 * g + 0.07 * b;
300 r = g = b = luminosity;
303 myColor = QColor::fromRgb( r, g, b );
304 myColor.getHsl( &h, &s, &l );
310 int average = ( r + g + b ) / 3;
314 myColor = QColor::fromRgb( r, g, b );
315 myColor.getHsl( &h, &s, &l );
321 if ( mSaturationScale < 1 )
324 s = std::min( (
int )( s * mSaturationScale ), 255 );
330 s = std::min( (
int )( 255. * ( 1 - std::pow( 1 - ( s / 255. ), std::pow( mSaturationScale, 2 ) ) ) ), 255 );
334 myColor = QColor::fromHsl( h, s, l );
335 myColor.getRgb( &r, &g, &b );
343 mSaturation = std::clamp(
saturation, -100, 100 );
346 mSaturationScale = ( ( double ) mSaturation / 100 ) + 1;
354 mColorizeH = mColorizeColor.hue();
355 mColorizeS = mColorizeColor.saturation();
360 if ( parentElem.isNull() )
365 QDomElement filterElem = doc.createElement( u
"huesaturation"_s );
367 filterElem.setAttribute( u
"saturation"_s, QString::number( mSaturation ) );
368 filterElem.setAttribute( u
"grayscaleMode"_s, QString::number( mGrayscaleMode ) );
369 filterElem.setAttribute( u
"invertColors"_s, QString::number( mInvertColors ) );
370 filterElem.setAttribute( u
"colorizeOn"_s, QString::number( mColorizeOn ) );
371 filterElem.setAttribute( u
"colorizeRed"_s, QString::number( mColorizeColor.red() ) );
372 filterElem.setAttribute( u
"colorizeGreen"_s, QString::number( mColorizeColor.green() ) );
373 filterElem.setAttribute( u
"colorizeBlue"_s, QString::number( mColorizeColor.blue() ) );
374 filterElem.setAttribute( u
"colorizeStrength"_s, QString::number( mColorizeStrength ) );
376 parentElem.appendChild( filterElem );
381 if ( filterElem.isNull() )
386 setSaturation( filterElem.attribute( u
"saturation"_s, u
"0"_s ).toInt() );
388 mInvertColors =
static_cast< bool >( filterElem.attribute( u
"invertColors"_s, u
"0"_s ).toInt() );
390 mColorizeOn =
static_cast< bool >( filterElem.attribute( u
"colorizeOn"_s, u
"0"_s ).toInt() );
391 int mColorizeRed = filterElem.attribute( u
"colorizeRed"_s, u
"255"_s ).toInt();
392 int mColorizeGreen = filterElem.attribute( u
"colorizeGreen"_s, u
"128"_s ).toInt();
393 int mColorizeBlue = filterElem.attribute( u
"colorizeBlue"_s, u
"128"_s ).toInt();
394 setColorizeColor( QColor::fromRgb( mColorizeRed, mColorizeGreen, mColorizeBlue ) );
395 mColorizeStrength = filterElem.attribute( u
"colorizeStrength"_s, u
"100"_s ).toInt();
DataType
Raster data types.
@ ARGB32_Premultiplied
Color, alpha, red, green, blue, 4 bytes the same as QImage::Format_ARGB32_Premultiplied.
@ UnknownDataType
Unknown or unspecified type.
@ ARGB32
Color, alpha, red, green, blue, 4 bytes the same as QImage::Format_ARGB32.
bool isCanceled() const
Tells whether the operation has been canceled already.
void setColorizeOn(bool colorizeOn)
void setSaturation(int saturation)
void setGrayscaleMode(QgsHueSaturationFilter::GrayscaleMode grayscaleMode)
QgsHueSaturationFilter(QgsRasterInterface *input=nullptr)
Qgis::DataType dataType(int bandNo) const override
Returns data type for the band specified by number.
QgsRasterBlock * block(int bandNo, const QgsRectangle &extent, int width, int height, QgsRasterBlockFeedback *feedback=nullptr) override
Read block of data using given extent and size.
QgsHueSaturationFilter * clone() const override
Clone itself, create deep copy.
void setInvertColors(bool invertColors)
Sets whether the filter will invert colors.
void setColorizeColor(const QColor &colorizeColor)
void setColorizeStrength(int colorizeStrength)
int bandCount() const override
Gets number of bands.
QColor colorizeColor() const
void readXml(const QDomElement &filterElem) override
Sets base class members from xml. Usually called from create() methods of subclasses.
void writeXml(QDomDocument &doc, QDomElement &parentElem) const override
Write base class members to xml.
bool setInput(QgsRasterInterface *input) override
Set input.
Feedback object tailored for raster block reading.
QgsRasterInterface(QgsRasterInterface *input=nullptr)
QgsRasterInterface * mInput
virtual QgsRectangle extent() const
Gets the extent of the interface.
virtual QgsRasterInterface * input() const
Current input.
A rectangle specified with double values.
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...
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)