QGIS API Documentation 3.99.0-Master (d270888f95f)
Loading...
Searching...
No Matches
qgssinglesymbolrenderer.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgssinglesymbolrenderer.cpp
3 ---------------------
4 begin : November 2009
5 copyright : (C) 2009 by Martin Dobias
6 email : wonder dot sk at gmail dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
17
19#include "qgsfeature.h"
20#include "qgsfillsymbol.h"
22#include "qgslinesymbol.h"
23#include "qgslogger.h"
24#include "qgsmarkersymbol.h"
25#include "qgsogcutils.h"
26#include "qgspainteffect.h"
29#include "qgsproperty.h"
30#include "qgssldexportcontext.h"
32#include "qgssymbol.h"
33#include "qgssymbollayer.h"
34#include "qgssymbollayerutils.h"
35#include "qgsvectorlayer.h"
36
37#include <QDomDocument>
38#include <QDomElement>
39#include <QString>
40
41using namespace Qt::StringLiterals;
42
44 : QgsFeatureRenderer( u"singleSymbol"_s )
45 , mSymbol( symbol )
46{
47 Q_ASSERT( symbol );
48}
49
57
59
64
66{
67 Q_UNUSED( context )
68 Q_UNUSED( feature )
69 return mSymbol.get();
70}
71
73{
74 QgsFeatureRenderer::startRender( context, fields );
75
76 if ( !mSymbol )
77 return;
78
79 mSymbol->startRender( context, fields );
80}
81
83{
85
86 if ( !mSymbol )
87 return;
88
89 mSymbol->stopRender( context );
90}
91
92QSet<QString> QgsSingleSymbolRenderer::usedAttributes( const QgsRenderContext &context ) const
93{
94 QSet<QString> attributes;
95 if ( mSymbol )
96 attributes.unite( mSymbol->usedAttributes( context ) );
97 return attributes;
98}
99
101{
102 if ( mSymbol )
103 {
104 QgsStyleSymbolEntity entity( mSymbol.get() );
105 return visitor->visit( QgsStyleEntityVisitorInterface::StyleLeaf( &entity ) );
106 }
107 return true;
108}
109
111{
112 return mSymbol.get();
113}
114
116{
117 Q_ASSERT( s );
118 mSymbol.reset( s );
119}
120
122{
123 return mSymbol ? u"SINGLE: %1"_s.arg( mSymbol->dump() ) : QString();
124}
125
133
134void QgsSingleSymbolRenderer::toSld( QDomDocument &doc, QDomElement &element, const QVariantMap &props ) const
135{
136 QgsSldExportContext context;
137 context.setExtraProperties( props );
138 toSld( doc, element, context );
139}
140
141bool QgsSingleSymbolRenderer::toSld( QDomDocument &doc, QDomElement &element, QgsSldExportContext &context ) const
142{
143 const QVariantMap oldProps = context.extraProperties();
144 QVariantMap newProps = oldProps;
145
146 QDomElement ruleElem = doc.createElement( u"se:Rule"_s );
147 element.appendChild( ruleElem );
148
149 QDomElement nameElem = doc.createElement( u"se:Name"_s );
150 nameElem.appendChild( doc.createTextNode( u"Single symbol"_s ) );
151 ruleElem.appendChild( nameElem );
152
153 QgsSymbolLayerUtils::applyScaleDependency( doc, ruleElem, newProps );
154
155 context.setExtraProperties( newProps );
156 if ( mSymbol )
157 mSymbol->toSld( doc, ruleElem, context );
158
159 context.setExtraProperties( oldProps );
160 return true;
161}
162
164{
165 Q_UNUSED( context )
166 QgsSymbolList lst;
167 lst.append( mSymbol.get() );
168 return lst;
169}
170
171
173{
174 QDomElement symbolsElem = element.firstChildElement( u"symbols"_s );
175 if ( symbolsElem.isNull() )
176 return nullptr;
177
178 QgsSymbolMap symbolMap = QgsSymbolLayerUtils::loadSymbols( symbolsElem, context );
179
180 if ( !symbolMap.contains( u"0"_s ) )
181 return nullptr;
182
183 QgsSingleSymbolRenderer *r = new QgsSingleSymbolRenderer( symbolMap.take( u"0"_s ) );
184
185 // delete symbols if there are any more
187
188 const QDomElement rotationElem = element.firstChildElement( u"rotation"_s );
189 if ( !rotationElem.isNull() && !rotationElem.attribute( u"field"_s ).isEmpty() )
190 {
191 convertSymbolRotation( r->mSymbol.get(), rotationElem.attribute( u"field"_s ) );
192 }
193
194 const QDomElement sizeScaleElem = element.firstChildElement( u"sizescale"_s );
195 if ( !sizeScaleElem.isNull() && !sizeScaleElem.attribute( u"field"_s ).isEmpty() )
196 {
198 QgsSymbolLayerUtils::decodeScaleMethod( sizeScaleElem.attribute( u"scalemethod"_s ) ),
199 sizeScaleElem.attribute( u"field"_s ) );
200 }
201
202 const QDomElement ddsLegendSizeElem = element.firstChildElement( u"data-defined-size-legend"_s );
203 if ( !ddsLegendSizeElem.isNull() )
204 {
205 r->mDataDefinedSizeLegend.reset( QgsDataDefinedSizeLegend::readXml( ddsLegendSizeElem, context ) );
206 }
207
208 // TODO: symbol levels
209 return r;
210}
211
213{
214 // XXX this renderer can handle only one Rule!
215
216 // get the first Rule element
217 const QDomElement ruleElem = element.firstChildElement( u"Rule"_s );
218 if ( ruleElem.isNull() )
219 {
220 QgsDebugError( u"no Rule elements found!"_s );
221 return nullptr;
222 }
223
224 QString label, description;
225 QgsSymbolLayerList layers;
226
227 // retrieve the Rule element child nodes
228 QDomElement childElem = ruleElem.firstChildElement();
229 while ( !childElem.isNull() )
230 {
231 if ( childElem.localName() == "Name"_L1 )
232 {
233 // <se:Name> tag contains the rule identifier,
234 // so prefer title tag for the label property value
235 if ( label.isEmpty() )
236 label = childElem.firstChild().nodeValue();
237 }
238 else if ( childElem.localName() == "Description"_L1 )
239 {
240 // <se:Description> can contains a title and an abstract
241 const QDomElement titleElem = childElem.firstChildElement( u"Title"_s );
242 if ( !titleElem.isNull() )
243 {
244 label = titleElem.firstChild().nodeValue();
245 }
246
247 const QDomElement abstractElem = childElem.firstChildElement( u"Abstract"_s );
248 if ( !abstractElem.isNull() )
249 {
250 description = abstractElem.firstChild().nodeValue();
251 }
252 }
253 else if ( childElem.localName() == "Abstract"_L1 )
254 {
255 // <sld:Abstract> (v1.0)
256 description = childElem.firstChild().nodeValue();
257 }
258 else if ( childElem.localName() == "Title"_L1 )
259 {
260 // <sld:Title> (v1.0)
261 label = childElem.firstChild().nodeValue();
262 }
263 else if ( childElem.localName().endsWith( "Symbolizer"_L1 ) )
264 {
265 // create symbol layers for this symbolizer
266 QgsSymbolLayerUtils::createSymbolLayerListFromSld( childElem, geomType, layers );
267 }
268
269 childElem = childElem.nextSiblingElement();
270 }
271
272 if ( layers.isEmpty() )
273 return nullptr;
274
275 // now create the symbol
276 std::unique_ptr< QgsSymbol > symbol;
277 switch ( geomType )
278 {
280 symbol = std::make_unique< QgsLineSymbol >( layers );
281 break;
282
284 symbol = std::make_unique< QgsFillSymbol >( layers );
285 break;
286
288 symbol = std::make_unique< QgsMarkerSymbol >( layers );
289 break;
290
291 default:
292 QgsDebugError( u"invalid geometry type: found %1"_s.arg( qgsEnumValueToKey( geomType ) ) );
293 return nullptr;
294 }
295
296 // and finally return the new renderer
297 return new QgsSingleSymbolRenderer( symbol.release() );
298}
299
300QDomElement QgsSingleSymbolRenderer::save( QDomDocument &doc, const QgsReadWriteContext &context )
301{
302 QDomElement rendererElem = doc.createElement( RENDERER_TAG_NAME );
303 rendererElem.setAttribute( u"type"_s, u"singleSymbol"_s );
304
306 symbols[u"0"_s] = mSymbol.get();
307 const QDomElement symbolsElem = QgsSymbolLayerUtils::saveSymbols( symbols, u"symbols"_s, doc, context );
308 rendererElem.appendChild( symbolsElem );
309
310 const QDomElement rotationElem = doc.createElement( u"rotation"_s );
311 rendererElem.appendChild( rotationElem );
312
313 const QDomElement sizeScaleElem = doc.createElement( u"sizescale"_s );
314 rendererElem.appendChild( sizeScaleElem );
315
317 {
318 QDomElement ddsLegendElem = doc.createElement( u"data-defined-size-legend"_s );
319 mDataDefinedSizeLegend->writeXml( ddsLegendElem, context );
320 rendererElem.appendChild( ddsLegendElem );
321 }
322
323 saveRendererData( doc, rendererElem, context );
324
325 return rendererElem;
326}
327
329{
331 {
332 const QgsMarkerSymbol *symbol = static_cast<const QgsMarkerSymbol *>( mSymbol.get() );
333 const QgsProperty sizeDD( symbol->dataDefinedSize() );
334 if ( sizeDD && sizeDD.isActive() )
335 {
337 ddSizeLegend.updateFromSymbolAndProperty( static_cast<const QgsMarkerSymbol *>( mSymbol.get() ), sizeDD );
338 return ddSizeLegend.legendSymbolList();
339 }
340 }
341
343 lst << QgsLegendSymbolItem( mSymbol.get(), QString(), u"0"_s );
344 return lst;
345}
346
347QSet< QString > QgsSingleSymbolRenderer::legendKeysForFeature( const QgsFeature &feature, QgsRenderContext &context ) const
348{
349 Q_UNUSED( feature )
350 Q_UNUSED( context )
351 return QSet< QString >() << u"0"_s;
352}
353
354QString QgsSingleSymbolRenderer::legendKeyToExpression( const QString &key, QgsVectorLayer *, bool &ok ) const
355{
356 if ( key == "0"_L1 )
357 {
358 ok = true;
359 return u"TRUE"_s;
360 }
361 else
362 {
363 ok = false;
364 return QString();
365 }
366}
367
369{
370 Q_UNUSED( key )
371 setSymbol( symbol );
372}
373
375{
376 QgsSingleSymbolRenderer *r = nullptr;
377 if ( renderer->type() == "singleSymbol"_L1 )
378 {
379 r = dynamic_cast<QgsSingleSymbolRenderer *>( renderer->clone() );
380 }
381 else if ( renderer->type() == "pointDisplacement"_L1 || renderer->type() == "pointCluster"_L1 )
382 {
383 const QgsPointDistanceRenderer *pointDistanceRenderer = dynamic_cast<const QgsPointDistanceRenderer *>( renderer );
384 if ( pointDistanceRenderer )
385 r = convertFromRenderer( pointDistanceRenderer->embeddedRenderer() );
386 }
387 else if ( renderer->type() == "invertedPolygonRenderer"_L1 )
388 {
389 const QgsInvertedPolygonRenderer *invertedPolygonRenderer = dynamic_cast<const QgsInvertedPolygonRenderer *>( renderer );
390 if ( invertedPolygonRenderer )
391 r = convertFromRenderer( invertedPolygonRenderer->embeddedRenderer() );
392 }
393
394 if ( !r )
395 {
396 QgsRenderContext context;
397 const QgsSymbolList symbols = const_cast<QgsFeatureRenderer *>( renderer )->symbols( context );
398 if ( !symbols.isEmpty() )
399 {
400 r = new QgsSingleSymbolRenderer( symbols.at( 0 )->clone() );
401 }
402 }
403
404 if ( r )
405 {
406 renderer->copyRendererData( r );
407 }
408
409 return r;
410}
411
416
QFlags< FeatureRendererFlag > FeatureRendererFlags
Flags controlling behavior of vector feature renderers.
Definition qgis.h:857
@ AffectsLabeling
If present, indicates that the renderer will participate in the map labeling problem.
Definition qgis.h:848
GeometryType
The geometry types are used to group Qgis::WkbType in a coarse way.
Definition qgis.h:365
@ Point
Points.
Definition qgis.h:366
@ Line
Lines.
Definition qgis.h:367
@ Polygon
Polygons.
Definition qgis.h:368
@ Marker
Marker symbol.
Definition qgis.h:630
@ AffectsLabeling
If present, indicates that the symbol will participate in the map labeling problem.
Definition qgis.h:868
Object that keeps configuration of appearance of marker symbol's data-defined size in legend.
static QgsDataDefinedSizeLegend * readXml(const QDomElement &elem, const QgsReadWriteContext &context) SIP_FACTORY
Creates instance from given element and returns it (caller takes ownership). Returns nullptr on error...
void updateFromSymbolAndProperty(const QgsMarkerSymbol *symbol, const QgsProperty &ddSize)
Updates the list of classes, source symbol and title label from given symbol and property.
QgsLegendSymbolList legendSymbolList() const
Generates legend symbol items according to the configuration.
QgsFeatureRenderer(const QString &type)
virtual void stopRender(QgsRenderContext &context)
Must be called when a render cycle has finished, to allow the renderer to clean up.
QString type() const
void copyRendererData(QgsFeatureRenderer *destRenderer) const
Clones generic renderer data to another renderer.
static void convertSymbolRotation(QgsSymbol *symbol, const QString &field)
Converts old rotation expressions to symbol level data defined angles.
void saveRendererData(QDomDocument &doc, QDomElement &element, const QgsReadWriteContext &context)
Saves generic renderer data into the specified element.
virtual void startRender(QgsRenderContext &context, const QgsFields &fields)
Must be called when a new render cycle is started.
static void convertSymbolSizeScale(QgsSymbol *symbol, Qgis::ScaleMethod method, const QString &field)
Converts old sizeScale expressions to symbol level data defined sizes.
virtual QgsFeatureRenderer * clone() const =0
Create a deep copy of this renderer.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
Container of fields for a vector layer.
Definition qgsfields.h:46
A polygon-only feature renderer used to display features inverted.
Stores information about one class/rule of a vector layer renderer in a unified way that can be used ...
A marker symbol type, for rendering Point and MultiPoint geometries.
const QgsFeatureRenderer * embeddedRenderer() const override
Returns the current embedded renderer (subrenderer) for this feature renderer.
An abstract base class for distance based point renderers (e.g., clusterer and displacement renderers...
const QgsFeatureRenderer * embeddedRenderer() const override
Returns the current embedded renderer (subrenderer) for this feature renderer.
A store for object properties.
bool isActive() const
Returns whether the property is currently active.
A container for the context for various read/write operations on objects.
Contains information about the context of a rendering operation.
QSet< QString > usedAttributes(const QgsRenderContext &context) const override
Returns a list of attributes required by this renderer.
QDomElement save(QDomDocument &doc, const QgsReadWriteContext &context) override
Stores renderer properties to an XML element.
QgsSymbol * symbol() const
Returns the symbol which will be rendered for every feature.
Qgis::FeatureRendererFlags flags() const override
Returns flags associated with the renderer.
QString legendKeyToExpression(const QString &key, QgsVectorLayer *layer, bool &ok) const override
Attempts to convert the specified legend rule key to a QGIS expression matching the features displaye...
void setSymbol(QgsSymbol *s)
Sets the symbol which will be rendered for every feature.
static QgsFeatureRenderer * createFromSld(QDomElement &element, Qgis::GeometryType geomType)
Creates a new single symbol renderer from an SLD element.
std::unique_ptr< QgsSymbol > mSymbol
bool accept(QgsStyleEntityVisitorInterface *visitor) const override
Accepts the specified symbology visitor, causing it to visit all symbols associated with the renderer...
void setLegendSymbolItem(const QString &key, QgsSymbol *symbol) override
Sets the symbol to be used for a legend symbol item.
static QgsFeatureRenderer * create(QDomElement &element, const QgsReadWriteContext &context)
Creates a new single symbol renderer from an XML element, using the supplied read/write context.
~QgsSingleSymbolRenderer() override
QSet< QString > legendKeysForFeature(const QgsFeature &feature, QgsRenderContext &context) const override
Returns legend keys matching a specified feature.
void setDataDefinedSizeLegend(QgsDataDefinedSizeLegend *settings)
Configures appearance of legend when renderer is configured to use data-defined size for marker symbo...
void stopRender(QgsRenderContext &context) override
Must be called when a render cycle has finished, to allow the renderer to clean up.
QgsLegendSymbolList legendSymbolItems() const override
Returns a list of symbology items for the legend.
void startRender(QgsRenderContext &context, const QgsFields &fields) override
Must be called when a new render cycle is started.
QgsDataDefinedSizeLegend * dataDefinedSizeLegend() const
Returns configuration of appearance of legend when using data-defined size for marker symbols.
std::unique_ptr< QgsDataDefinedSizeLegend > mDataDefinedSizeLegend
QgsSymbol * originalSymbolForFeature(const QgsFeature &feature, QgsRenderContext &context) const override
Returns symbol for feature.
static QgsSingleSymbolRenderer * convertFromRenderer(const QgsFeatureRenderer *renderer)
Creates a new single symbol renderer from an existing renderer.
QgsSymbol * symbolForFeature(const QgsFeature &feature, QgsRenderContext &context) const override
To be overridden.
Q_DECL_DEPRECATED void toSld(QDomDocument &doc, QDomElement &element, const QVariantMap &props=QVariantMap()) const override
Used from subclasses to create SLD Rule elements following SLD v1.1 specs.
QgsSingleSymbolRenderer * clone() const override
Create a deep copy of this renderer.
QgsSymbolList symbols(QgsRenderContext &context) const override
Returns list of symbols used by the renderer.
QString dump() const override
Returns debug information about this renderer.
QgsSingleSymbolRenderer(QgsSymbol *symbol)
Constructor for QgsSingleSymbolRenderer.
Holds SLD export options and other information related to SLD export of a QGIS layer style.
void setExtraProperties(const QVariantMap &properties)
Sets the open ended set of properties that can drive/inform the SLD encoding.
QVariantMap extraProperties() const
Returns the open ended set of properties that can drive/inform the SLD encoding.
An interface for classes which can visit style entity (e.g.
virtual bool visit(const QgsStyleEntityVisitorInterface::StyleLeaf &entity)
Called when the visitor will visit a style entity.
A symbol entity for QgsStyle databases.
Definition qgsstyle.h:1398
static void applyScaleDependency(QDomDocument &doc, QDomElement &ruleElem, QVariantMap &props)
Checks if the properties contain scaleMinDenom and scaleMaxDenom, if available, they are added into t...
static bool createSymbolLayerListFromSld(QDomElement &element, Qgis::GeometryType geomType, QList< QgsSymbolLayer * > &layers)
Creates a symbol layer list from a DOM element.
static Qgis::ScaleMethod decodeScaleMethod(const QString &str)
Decodes a symbol scale method from a string.
static void clearSymbolMap(QgsSymbolMap &symbols)
static QgsSymbolMap loadSymbols(QDomElement &element, const QgsReadWriteContext &context)
Reads a collection of symbols from XML and returns them in a map. Caller is responsible for deleting ...
static QDomElement saveSymbols(QgsSymbolMap &symbols, const QString &tagName, QDomDocument &doc, const QgsReadWriteContext &context)
Writes a collection of symbols to XML with specified tagName for the top-level element.
Abstract base class for all rendered symbols.
Definition qgssymbol.h:231
Represents a vector layer which manages a vector based dataset.
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
Definition qgis.h:7091
QList< QgsLegendSymbolItem > QgsLegendSymbolList
#define QgsDebugError(str)
Definition qgslogger.h:59
#define RENDERER_TAG_NAME
Definition qgsrenderer.h:57
QMap< QString, QgsSymbol * > QgsSymbolMap
Definition qgsrenderer.h:52
QList< QgsSymbol * > QgsSymbolList
Definition qgsrenderer.h:51
QList< QgsSymbolLayer * > QgsSymbolLayerList
Definition qgssymbol.h:30
Contains information relating to the style entity currently being visited.