QGIS API Documentation 4.1.0-Master (376402f9aeb)
Loading...
Searching...
No Matches
qgsrasterrendererregistry.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsrasterrendererregistry.cpp
3 -----------------------------
4 begin : January 2012
5 copyright : (C) 2012 by Marco Hugentobler
6 email : marco at sourcepole dot ch
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
20#include "qgsapplication.h"
21#include "qgscolorrampimpl.h"
28#include "qgsrastershader.h"
32#include "qgssettingstree.h"
36
37#include <QIcon>
38#include <QString>
39
40using namespace Qt::StringLiterals;
41
43 = new QgsSettingsEntryInteger( u"default-red-band"_s, QgsSettingsTree::sTreeRaster, 1, u"Default band number assigned to the red channel when creating an RGB renderer for a multi-band raster."_s );
45 = new QgsSettingsEntryInteger( u"default-green-band"_s, QgsSettingsTree::sTreeRaster, 2, u"Default band number assigned to the green channel when creating an RGB renderer for a multi-band raster."_s );
47 = new QgsSettingsEntryInteger( u"default-blue-band"_s, QgsSettingsTree::sTreeRaster, 3, u"Default band number assigned to the blue channel when creating an RGB renderer for a multi-band raster."_s );
49 = new QgsSettingsEntryBool( u"use-standard-deviation"_s, QgsSettingsTree::sTreeRaster, false, u"If true, newly created raster renderers use a standard-deviation based contrast enhancement by default."_s );
51 = new QgsSettingsEntryDouble( u"default-standard-deviation"_s, QgsSettingsTree::sTreeRaster, 2.0, u"Default standard deviation multiplier used to compute min/max values for raster contrast enhancement when \"use-standard-deviation\" is enabled."_s );
52
62
64{
65 return QgsApplication::getThemeIcon( QString( "styleicons/%1.svg" ).arg( name ) );
66}
67
69{
70 // insert items in a particular order, which is returned in renderersList()
71 insert( QgsRasterRendererRegistryEntry( u"multibandcolor"_s, QObject::tr( "Multiband color" ), QgsMultiBandColorRenderer::create, nullptr, Qgis::RasterRendererCapability::UsesMultipleBands ) );
72 insert( QgsRasterRendererRegistryEntry( u"paletted"_s, QObject::tr( "Paletted/Unique values" ), QgsPalettedRasterRenderer::create, nullptr ) );
73 insert( QgsRasterRendererRegistryEntry( u"singlebandgray"_s, QObject::tr( "Singleband gray" ), QgsSingleBandGrayRenderer::create, nullptr ) );
74 insert( QgsRasterRendererRegistryEntry( u"singlebandpseudocolor"_s, QObject::tr( "Singleband pseudocolor" ), QgsSingleBandPseudoColorRenderer::create, nullptr ) );
75 insert( QgsRasterRendererRegistryEntry( u"singlebandcolordata"_s, QObject::tr( "Singleband color data" ), QgsSingleBandColorDataRenderer::create, nullptr ) );
76 insert( QgsRasterRendererRegistryEntry( u"singlecolor"_s, QObject::tr( "Single color" ), QgsRasterSingleColorRenderer::create, nullptr ) );
77 insert( QgsRasterRendererRegistryEntry( u"hillshade"_s, QObject::tr( "Hillshade" ), QgsHillshadeRenderer::create, nullptr ) );
78 insert( QgsRasterRendererRegistryEntry( u"contour"_s, QObject::tr( "Contours" ), QgsRasterContourRenderer::create, nullptr ) );
79}
80
82{
83 mEntries.insert( entry.name, entry );
84 mSortedEntries.append( entry.name );
85}
86
88{
89 if ( !mEntries.contains( rendererName ) )
90 {
91 return;
92 }
93 mEntries[rendererName].widgetCreateFunction = func;
94}
95
96bool QgsRasterRendererRegistry::rendererData( const QString &rendererName, QgsRasterRendererRegistryEntry &data ) const
97{
98 const QHash< QString, QgsRasterRendererRegistryEntry >::const_iterator it = mEntries.find( rendererName );
99 if ( it == mEntries.constEnd() )
100 {
101 return false;
102 }
103 data = it.value();
104 return true;
105}
106
108{
109 return mSortedEntries;
110}
111
112QList< QgsRasterRendererRegistryEntry > QgsRasterRendererRegistry::entries() const
113{
114 QList< QgsRasterRendererRegistryEntry > result;
115
116 QHash< QString, QgsRasterRendererRegistryEntry >::const_iterator it = mEntries.constBegin();
117 for ( ; it != mEntries.constEnd(); ++it )
118 {
119 result.push_back( it.value() );
120 }
121 return result;
122}
123
125{
126 const QHash< QString, QgsRasterRendererRegistryEntry >::const_iterator it = mEntries.constFind( rendererName );
127 if ( it != mEntries.constEnd() )
128 {
129 return it.value().capabilities;
130 }
132}
133
135{
136 if ( !provider || provider->bandCount() < 1 )
137 {
138 return nullptr;
139 }
140
141 std::unique_ptr< QgsRasterRenderer > renderer;
142 switch ( drawingStyle )
143 {
145 {
146 const int grayBand = 1; //reasonable default
147
148 // first preference -- use attribute table to generate classes
149 if ( provider->attributeTable( grayBand ) )
150 {
151 std::unique_ptr<QgsColorRamp> ramp;
152 if ( !provider->attributeTable( grayBand )->hasColor() )
153 {
154 ramp = std::make_unique< QgsRandomColorRamp >();
155 }
157 if ( !classes.empty() )
158 {
159 renderer = std::make_unique< QgsPalettedRasterRenderer >( provider, grayBand, classes );
160 }
161 }
162
163 // second preference -- use raster color table to generate classes
164 if ( !renderer )
165 {
167 if ( !classes.empty() )
168 {
169 renderer = std::make_unique< QgsPalettedRasterRenderer >( provider, grayBand, classes );
170 }
171 }
172
173 // last preference -- just fallback to single band gray renderer if we couldn't determine color palette
174 if ( !renderer )
175 {
176 renderer = std::make_unique< QgsSingleBandGrayRenderer >( provider, grayBand );
177
178 QgsContrastEnhancement *ce = new QgsContrastEnhancement( ( Qgis::DataType ) ( provider->dataType( grayBand ) ) );
179
180 // Default contrast enhancement is set from QgsRasterLayer, it has already setContrastEnhancementAlgorithm(). Default enhancement must only be set if default style was not loaded (to avoid stats calculation).
181 qgis::down_cast< QgsSingleBandGrayRenderer * >( renderer.get() )->setContrastEnhancement( ce );
182 }
183 }
184 break;
185
188 {
189 const int grayBand = 1;
190
191 // If the raster band has an attribute table try to use it.
192 QString ratErrorMessage;
193 if ( QgsRasterAttributeTable *rat = provider->attributeTable( grayBand ); rat && rat->isValid( &ratErrorMessage ) )
194 {
195 renderer.reset( rat->createRenderer( provider, grayBand ) );
196 }
197
198 if ( !ratErrorMessage.isEmpty() )
199 {
200 QgsDebugMsgLevel( u"Invalid RAT from band 1, RAT was not used to create the renderer: %1."_s.arg( ratErrorMessage ), 2 );
201 }
202
203 if ( !renderer )
204 {
205 renderer = std::make_unique< QgsSingleBandGrayRenderer >( provider, grayBand );
206
207 QgsContrastEnhancement *ce = new QgsContrastEnhancement( ( Qgis::DataType ) ( provider->dataType( grayBand ) ) );
208
209 // Default contrast enhancement is set from QgsRasterLayer, it has already setContrastEnhancementAlgorithm(). Default enhancement must only be set if default style was not loaded (to avoid stats calculation).
210 qgis::down_cast< QgsSingleBandGrayRenderer * >( renderer.get() )->setContrastEnhancement( ce );
211 }
212 break;
213 }
214
216 {
217 const int bandNo = 1;
218 double minValue = 0;
219 double maxValue = 0;
220 // TODO: avoid calculating statistics if not necessary (default style loaded)
221 minMaxValuesForBand( bandNo, provider, minValue, maxValue );
222 QgsRasterShader *shader = new QgsRasterShader( minValue, maxValue );
223 renderer = std::make_unique< QgsSingleBandPseudoColorRenderer >( provider, bandNo, shader );
224 break;
225 }
227 {
228 int redBand = settingsDefaultRedBand->value();
229 if ( redBand < 0 || redBand > provider->bandCount() )
230 {
231 redBand = -1;
232 }
233 int greenBand = settingsDefaultGreenBand->value();
234 if ( greenBand < 0 || greenBand > provider->bandCount() )
235 {
236 greenBand = -1;
237 }
238 int blueBand = settingsDefaultBlueBand->value();
239 if ( blueBand < 0 || blueBand > provider->bandCount() )
240 {
241 blueBand = -1;
242 }
243
244 renderer = std::make_unique< QgsMultiBandColorRenderer >( provider, redBand, greenBand, blueBand );
245 break;
246 }
248 {
249 renderer = std::make_unique< QgsSingleBandColorDataRenderer >( provider, 1 );
250 break;
251 }
252 default:
253 return nullptr;
254 }
255
256 auto tr = std::make_unique< QgsRasterTransparency >();
257 const int bandCount = renderer->usesBands().size();
258 if ( bandCount == 1 )
259 {
260 tr->setTransparentSingleValuePixelList( {} );
261 }
262 else if ( bandCount == 3 )
263 {
264 tr->setTransparentThreeValuePixelList( {} );
265 }
266 renderer->setRasterTransparency( tr.release() );
267 return renderer.release();
268}
269
270bool QgsRasterRendererRegistry::minMaxValuesForBand( int band, QgsRasterDataProvider *provider, double &minValue, double &maxValue ) const
271{
272 if ( !provider )
273 {
274 return false;
275 }
276
277 minValue = 0;
278 maxValue = 0;
279
280 if ( settingsUseStandardDeviation->value() )
281 {
282 const QgsRasterBandStats stats = provider->bandStatistics( band, Qgis::RasterBandStatistic::Mean | Qgis::RasterBandStatistic::StdDev );
283
284 const double stdDevFactor = settingsDefaultStandardDeviation->value();
285 const double diff = stdDevFactor * stats.stdDev;
286 minValue = stats.mean - diff;
287 maxValue = stats.mean + diff;
288 }
289 else
290 {
291 const QgsRasterBandStats stats = provider->bandStatistics( band, Qgis::RasterBandStatistic::Min | Qgis::RasterBandStatistic::Max );
292 minValue = stats.minimumValue;
293 maxValue = stats.maximumValue;
294 }
295 return true;
296}
RasterDrawingStyle
Raster drawing styles.
Definition qgis.h:5092
@ SingleBandGray
A single band image drawn as a range of gray colors.
Definition qgis.h:5094
@ MultiBandSingleBandGray
A layer containing 2 or more bands, but a single band drawn as a range of gray colors.
Definition qgis.h:5100
@ SingleBandColorData
ARGB values rendered directly.
Definition qgis.h:5103
@ MultiBandColor
A layer containing 2 or more bands, mapped to RGB color space. In the case of a multiband with only t...
Definition qgis.h:5102
@ PalettedColor
A "Palette" image drawn using color table.
Definition qgis.h:5096
@ SingleBandPseudoColor
A single band image drawn using a pseudocolor algorithm.
Definition qgis.h:5095
QFlags< RasterRendererCapability > RasterRendererCapabilities
Raster renderer capabilities.
Definition qgis.h:1653
@ StdDev
Standard deviation.
Definition qgis.h:6460
DataType
Raster data types.
Definition qgis.h:393
@ UsesMultipleBands
The renderer utilizes multiple raster bands for color data (note that alpha bands are not considered ...
Definition qgis.h:1644
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
Handles contrast enhancement and clipping.
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
Factory method to create a new renderer.
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
static QgsPalettedRasterRenderer::MultiValueClassData rasterAttributeTableToClassData(const QgsRasterAttributeTable *attributeTable, int classificationColumn=-1, QgsColorRamp *ramp=nullptr)
Reads and returns classes from the Raster Attribute Table attributeTable, optionally classifying the ...
QList< QgsPalettedRasterRenderer::Class > ClassData
Map of value to class properties.
QList< QgsPalettedRasterRenderer::MultiValueClass > MultiValueClassData
Map of multi value to class properties.
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
static QgsPalettedRasterRenderer::ClassData colorTableToClassData(const QList< QgsColorRampShader::ColorRampItem > &table)
Converts a raster color table to paletted renderer class data.
Represents a Raster Attribute Table (RAT).
bool hasColor() const
Returns true if the Raster Attribute Table has color RGBA information.
bool isValid(QString *errorMessage=nullptr) const
Returns true if the Raster Attribute Table is valid, optionally reporting validity checks results in ...
double mean
The mean cell value for the band. NO_DATA values are excluded.
double stdDev
The standard deviation of the cell values.
double minimumValue
The minimum cell value in the raster band.
double maximumValue
The maximum cell value in the raster band.
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
Creates an instance of the renderer based on definition from XML (used by renderer registry).
Base class for raster data providers.
QgsRasterAttributeTable * attributeTable(int bandNumber) const
Returns the (possibly nullptr) attribute table for the specified bandNumber.
Qgis::DataType dataType(int bandNo) const override=0
Returns data type for the band specified by number.
virtual QList< QgsColorRampShader::ColorRampItem > colorTable(int bandNo) const
Q_DECL_DEPRECATED QgsRasterBandStats bandStatistics(int bandNo, int stats, const QgsRectangle &extent=QgsRectangle(), int sampleSize=0, QgsRasterBlockFeedback *feedback=nullptr)
Returns the band statistics.
virtual int bandCount() const =0
Gets number of bands.
static const QgsSettingsEntryInteger * settingsDefaultGreenBand
void insertWidgetFunction(const QString &rendererName, QgsRasterRendererWidgetCreateFunc func)
Sets the widget creation function for a renderer.
QgsRasterRendererRegistry()
Constructor for QgsRasterRendererRegistry.
QList< QgsRasterRendererRegistryEntry > entries() const
Returns the list of registered renderers.
void insert(const QgsRasterRendererRegistryEntry &entry)
Inserts a new entry into the registry.
static const QgsSettingsEntryBool * settingsUseStandardDeviation
static const QgsSettingsEntryDouble * settingsDefaultStandardDeviation
bool rendererData(const QString &rendererName, QgsRasterRendererRegistryEntry &data) const
Retrieves renderer data from the registry.
static const QgsSettingsEntryInteger * settingsDefaultRedBand
QgsRasterRenderer * defaultRendererForDrawingStyle(Qgis::RasterDrawingStyle drawingStyle, QgsRasterDataProvider *provider) const
Creates a default renderer for a raster drawing style (considering user options such as default contr...
Qgis::RasterRendererCapabilities rendererCapabilities(const QString &rendererName) const
Returns the capabilities for the renderer with the specified name.
QStringList renderersList() const
Returns a list of the names of registered renderers.
static const QgsSettingsEntryInteger * settingsDefaultBlueBand
Raster renderer pipe that applies colors to a raster.
Interface for all raster shaders.
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
Creates an instance of the renderer based on definition from XML (used by the renderer registry).
A boolean settings entry.
A double settings entry.
An integer settings entry.
static QgsSettingsTreeNode * sTreeRaster
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
static QgsRasterRenderer * create(const QDomElement &elem, QgsRasterInterface *input)
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63
QgsRasterRendererWidget *(* QgsRasterRendererWidgetCreateFunc)(QgsRasterLayer *, const QgsRectangle &extent)
QgsRasterRenderer *(* QgsRasterRendererCreateFunc)(const QDomElement &, QgsRasterInterface *input)
Registry for raster renderer entries.
QgsRasterRendererWidgetCreateFunc widgetCreateFunction
Qgis::RasterRendererCapabilities capabilities
Renderer capabilities.
QgsRasterRendererCreateFunc rendererCreateFunction