QGIS API Documentation 3.41.0-Master (af5edcb665c)
Loading...
Searching...
No Matches
qgslayertreemodellegendnode.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayertreemodellegendnode.cpp
3 --------------------------------------
4 Date : August 2014
5 Copyright : (C) 2014 by Martin Dobias
6 Email : wonder dot sk at gmail dot com
7
8 QgsWMSLegendNode : Sandro Santilli < strk at keybit dot net >
9
10 ***************************************************************************
11 * *
12 * This program is free software; you can redistribute it and/or modify *
13 * it under the terms of the GNU General Public License as published by *
14 * the Free Software Foundation; either version 2 of the License, or *
15 * (at your option) any later version. *
16 * *
17 ***************************************************************************/
18
20#include "moc_qgslayertreemodellegendnode.cpp"
21
23#include "qgslayertreemodel.h"
24#include "qgslegendsettings.h"
25#include "qgsrasterlayer.h"
26#include "qgsrenderer.h"
27#include "qgssymbollayerutils.h"
28#include "qgsimageoperation.h"
29#include "qgsvectorlayer.h"
30#include "qgspointcloudlayer.h"
32#include "qgsrasterrenderer.h"
34#include "qgsexpression.h"
35#include "qgstextrenderer.h"
36#include "qgssettings.h"
37#include "qgsfileutils.h"
38#include "qgsmarkersymbol.h"
39#include "qgsvariantutils.h"
40#include "qgslayertreelayer.h"
41#include "qgstextdocument.h"
43
44#include <QBuffer>
45#include <optional>
46
48 : QObject( parent )
49 , mLayerNode( nodeL )
50 , mEmbeddedInParent( false )
51{
52}
53
55{
56 return qobject_cast<QgsLayerTreeModel *>( parent() );
57}
58
60{
61 return Qt::ItemIsEnabled;
62}
63
64bool QgsLayerTreeModelLegendNode::setData( const QVariant &value, int role )
65{
66 Q_UNUSED( value )
67 Q_UNUSED( role )
68 return false;
69}
70
72{
74 return mLayerNode->patchSize();
75
76 return mUserSize;
77}
78
80{
81 if ( mUserSize == size )
82 return;
83
84 mUserSize = size;
85 emit sizeChanged();
86}
87
89{
91
92 const QStringList lines = settings.evaluateItemText( data( Qt::DisplayRole ).toString(), ctx->context->expressionContext() );
93
94 const QgsTextDocument textDocument = QgsTextDocument::fromTextAndFormat( lines, f );
95 // cppcheck-suppress autoVariables
96 ctx->textDocument = &textDocument;
97
98 std::optional< QgsScopedRenderContextScaleToPixels > scaleToPx( *ctx->context );
99 const double textScaleFactor = QgsTextRenderer::calculateScaleFactorForFormat( *ctx->context, f );
100
101 const QgsTextDocumentMetrics textDocumentMetrics = QgsTextDocumentMetrics::calculateMetrics( textDocument, f, *ctx->context, textScaleFactor );
102 // cppcheck-suppress autoVariables
103 ctx->textDocumentMetrics = &textDocumentMetrics;
104 scaleToPx.reset();
105
106 // itemHeight here is not really item height, it is only for symbol
107 // vertical alignment purpose, i.e. OK take single line height
108 // if there are more lines, those run under the symbol
109 // also note that we explicitly use the first line cap height here, in order to match the Qgis::TextLayoutMode::RectangleCapHeightBased mode
110 // used when rendering the symbol text
111 const double textHeight = textDocumentMetrics.firstLineCapHeight() / ctx->context->scaleFactor();
112 const double itemHeight = std::max( static_cast< double >( ctx && ctx->patchSize.height() > 0 ? ctx->patchSize.height() : settings.symbolSize().height() ), textHeight );
113
114 ItemMetrics im;
115 im.symbolSize = drawSymbol( settings, ctx, itemHeight );
116 im.labelSize = drawSymbolText( settings, ctx, im.symbolSize );
117
118 ctx->textDocument = nullptr;
119 ctx->textDocumentMetrics = nullptr;
120 return im;
121}
122
124{
125 QJsonObject json = exportSymbolToJson( settings, context );
126 const QString text = data( Qt::DisplayRole ).toString();
127 json[ QStringLiteral( "title" ) ] = text;
128 return json;
129}
130
131QSizeF QgsLayerTreeModelLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
132{
133 const QIcon symbolIcon = data( Qt::DecorationRole ).value<QIcon>();
134 if ( symbolIcon.isNull() )
135 return QSizeF();
136
137 QSizeF size = settings.symbolSize();
138 if ( ctx )
139 {
140 if ( ctx->patchSize.width() > 0 )
141 size.setWidth( ctx->patchSize.width( ) );
142 if ( ctx->patchSize.height() > 0 )
143 size.setHeight( ctx->patchSize.height( ) );
144 }
145
146 if ( ctx && ctx->painter && ctx->context )
147 {
148 const QgsScopedRenderContextScaleToPixels scopedScaleToPixels( *( ctx->context ) );
149 const double scaleFactor = ctx->context->scaleFactor();
150 const int width = static_cast<int>( size.width() * scaleFactor );
151 const int height = static_cast<int>( size.height() * scaleFactor );
152 const int y = static_cast<int>( ( ctx->top + ( itemHeight - size.height() ) / 2 ) * scaleFactor );
153 int x = 0;
154
155 switch ( settings.symbolAlignment() )
156 {
157 case Qt::AlignLeft:
158 default:
159 x = static_cast<int>( ctx->columnLeft * scaleFactor );
160 break;
161 case Qt::AlignRight:
162 x = static_cast<int>( ( ctx->columnRight - size.width() ) * scaleFactor );
163 break;
164 }
165 symbolIcon.paint( ctx->painter, x, y, width, height );
166 }
167 return size;
168}
169
171{
172 const QIcon icon = data( Qt::DecorationRole ).value<QIcon>();
173 if ( icon.isNull() )
174 return QJsonObject();
175
176 const QImage image( icon.pixmap( settings.symbolSize().width(), settings.symbolSize().height() ).toImage() );
177 QByteArray byteArray;
178 QBuffer buffer( &byteArray );
179 image.save( &buffer, "PNG" );
180 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
181
182 QJsonObject json;
183 json[ QStringLiteral( "icon" ) ] = base64;
184 return json;
185}
186
187QSizeF QgsLayerTreeModelLegendNode::drawSymbolText( const QgsLegendSettings &settings, ItemContext *ctx, QSizeF symbolSizeMM ) const
188{
189 // we need a full render context here, so make one if we don't already have one
190 std::unique_ptr< QgsRenderContext > tempContext;
191 QgsRenderContext *context = ctx ? ctx->context : nullptr;
192 if ( !context )
193 {
194 tempContext.reset( new QgsRenderContext( QgsRenderContext::fromQPainter( ctx ? ctx->painter : nullptr ) ) );
195 context = tempContext.get();
196 }
197
198 const QgsTextFormat format = settings.style( QgsLegendStyle::SymbolLabel ).textFormat();
199
200 // TODO QGIS 4.0 -- make these all mandatory
201 std::optional< QgsTextDocument > tempDocument;
202 const QgsTextDocument *document = ctx ? ctx->textDocument : nullptr;
203 if ( !document )
204 {
205 const QStringList lines = settings.evaluateItemText( data( Qt::DisplayRole ).toString(), context->expressionContext() );
206 tempDocument.emplace( QgsTextDocument::fromTextAndFormat( lines, format ) );
207 document = &tempDocument.value();
208 }
209
210 std::optional< QgsTextDocumentMetrics > tempMetrics;
211 const QgsTextDocumentMetrics *metrics = ctx ? ctx->textDocumentMetrics : nullptr;
212 if ( !metrics )
213 {
214 tempMetrics.emplace( QgsTextDocumentMetrics::calculateMetrics( *document, format, *context ) );
215 metrics = &tempMetrics.value();
216 }
217
218 const double dotsPerMM = context->scaleFactor();
219 QgsScopedRenderContextScaleToPixels scaleToPx( *context );
220
222 const QSizeF labelSizeMM( documentSize / dotsPerMM );
223
224 double labelXMin = 0.0;
225 double labelXMax = 0.0;
226 double labelYMM = 0.0;
227 if ( ctx && context->painter() )
228 {
229 switch ( settings.symbolAlignment() )
230 {
231 case Qt::AlignLeft:
232 default:
233 labelXMin = ctx->columnLeft + std::max( static_cast< double >( symbolSizeMM.width() ), ctx->maxSiblingSymbolWidth )
236 labelXMax = ctx->columnRight;
237 break;
238
239 case Qt::AlignRight:
240 labelXMin = ctx->columnLeft;
241 // NOTE -- while the below calculations use the flipped margins from the style, that's only done because
242 // those are the only margins we expose and use for now! (and we expose them as generic margins, not side-specific
243 // ones) TODO when/if we expose other margin settings, these should be reversed...
244 labelXMax = ctx->columnRight - std::max( static_cast< double >( symbolSizeMM.width() ), ctx->maxSiblingSymbolWidth )
247 break;
248 }
249
250 labelYMM = ctx->top;
251
252 // Vertical alignment of label with symbol
253 if ( labelSizeMM.height() < symbolSizeMM.height() )
254 labelYMM += ( symbolSizeMM.height() - labelSizeMM.height() ) / 2; // label centered with symbol
255 }
256
257 if ( context->painter() )
258 {
261
262
263 QgsTextRenderer::drawDocument( QRectF( labelXMin * dotsPerMM, std::round( labelYMM * dotsPerMM ),
264 ( labelXMax - labelXMin )* dotsPerMM,
265 std::max( symbolSizeMM.height(), labelSizeMM.height() ) * dotsPerMM ),
266 format, *document, *metrics, *context, halign, Qgis::TextVerticalAlignment::Top,
268 }
269
270 return labelSizeMM;
271}
272
274{
275 checkAll( true );
276}
277
279{
280 checkAll( false );
281}
282
284{
285 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() ) )
286 {
287 if ( !vlayer->renderer() )
288 return;
289
290 const QgsLegendSymbolList symbolList = vlayer->renderer()->legendSymbolItems();
291 for ( const auto &item : symbolList )
292 {
293 vlayer->renderer()->checkLegendSymbolItem( item.ruleKey(), ! vlayer->renderer()->legendSymbolItemChecked( item.ruleKey() ) );
294 }
295
296 emit dataChanged();
297 vlayer->emitStyleChanged();
298 vlayer->triggerRepaint();
299 }
300 else if ( QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( mLayerNode->layer() ) )
301 {
302 if ( !pclayer->renderer() )
303 return;
304
305 const QStringList ruleKeys = pclayer->renderer()->legendRuleKeys();
306 for ( const QString &rule : ruleKeys )
307 {
308 pclayer->renderer()->checkLegendItem( rule, !pclayer->renderer()->legendItemChecked( rule ) );
309 }
310
311 emit dataChanged();
312 pclayer->emitStyleChanged();
313 pclayer->triggerRepaint();
314 }
315}
316
317// -------------------------------------------------------------------------
318
321
323 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
324 , mItem( item )
325 , mSymbolUsesMapUnits( false )
326{
328 mIconSize = QSize( iconSize, iconSize );
329
330 if ( MINIMUM_SIZE < 0 )
331 {
332 // it's FAR too expensive to construct a QgsSettings object for every symbol node, especially for complex
333 // projects. So only read the valid size ranges once, and store them for subsequent use
334 const QgsSettings settings;
335 MINIMUM_SIZE = settings.value( "/qgis/legendsymbolMinimumSize", 0.5 ).toDouble();
336 MAXIMUM_SIZE = settings.value( "/qgis/legendsymbolMaximumSize", 20.0 ).toDouble();
337 }
338
339 updateLabel();
340 if ( QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( nodeLayer->layer() ) )
341 connect( vl, &QgsVectorLayer::symbolFeatureCountMapChanged, this, &QgsSymbolLegendNode::updateLabel );
342
343 connect( nodeLayer, &QObject::destroyed, this, [this]() { mLayerNode = nullptr; } );
344
345 if ( const QgsSymbol *symbol = mItem.symbol() )
346 {
347 mSymbolUsesMapUnits = symbol->usesMapUnits();
348 }
349}
350
352
353Qt::ItemFlags QgsSymbolLegendNode::flags() const
354{
355 if ( mItem.isCheckable() )
356 return Qt::ItemIsEnabled | Qt::ItemIsUserCheckable | Qt::ItemIsSelectable;
357 else
358 return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
359}
360
361
363{
364 const std::unique_ptr<QgsRenderContext> context( createTemporaryRenderContext() );
365 return minimumIconSize( context.get() );
366}
367
369{
371 const int largeIconSize = QgsLayerTreeModel::scaleIconSize( 512 );
372 QSize minSz( iconSize, iconSize );
373 if ( mItem.symbol() && ( mItem.symbol()->type() == Qgis::SymbolType::Marker
374 || mItem.symbol()->type() == Qgis::SymbolType::Line ) )
375 {
376 int maxSize = largeIconSize;
377
378 // unusued width, height variables
379 double width = 0.0;
380 double height = 0.0;
381 bool ok;
382 std::unique_ptr<QgsSymbol> symbol( QgsSymbolLayerUtils::restrictedSizeSymbol( mItem.symbol(), MINIMUM_SIZE, MAXIMUM_SIZE, context, width, height, &ok ) );
383
384 if ( !ok && context )
385 {
386 // It's not possible to get a restricted size symbol, so we restrict
387 // pixmap target size to be sure it would fit MAXIMUM_SIZE
388 maxSize = static_cast<int>( std::round( MAXIMUM_SIZE * context->scaleFactor() ) );
389 }
390
391 const QSize size( mItem.symbol()->type() == Qgis::SymbolType::Marker ? maxSize : minSz.width(),
392 maxSize );
393
394 QgsScreenProperties targetScreen = model() && !model()->targetScreenProperties().isEmpty()
396
399 context, false, nullptr, nullptr, targetScreen ).toImage(),
400 minSz,
401 true ).size() / targetScreen.devicePixelRatio();
402 }
403
404 if ( !mTextOnSymbolLabel.isEmpty() && context )
405 {
406 const double w = QgsTextRenderer::textWidth( *context, mTextOnSymbolTextFormat, QStringList() << mTextOnSymbolLabel );
407 const double h = QgsTextRenderer::textHeight( *context, mTextOnSymbolTextFormat, QStringList() << mTextOnSymbolLabel, Qgis::TextLayoutMode::Point );
408 int wInt = ceil( w ), hInt = ceil( h );
409 if ( wInt > minSz.width() ) minSz.setWidth( wInt );
410 if ( hInt > minSz.height() ) minSz.setHeight( hInt );
411 }
412
413 return minSz;
414}
415
417{
418 return mItem.symbol();
419}
420
422{
423 QString label;
424 if ( mEmbeddedInParent )
425 {
426 const QVariant legendlabel = mLayerNode->customProperty( QStringLiteral( "legend/title-label" ) );
427 const QString layerName = QgsVariantUtils::isNull( legendlabel ) ? mLayerNode->name() : legendlabel.toString();
428 label = mUserLabel.isEmpty() ? layerName : mUserLabel;
429 }
430 else
431 label = mUserLabel.isEmpty() ? mItem.label() : mUserLabel;
432 return label;
433}
434
436{
437 if ( mEmbeddedInParent )
438 {
439 return mLayerNode->patchShape();
440 }
441 else
442 {
443 return mPatchShape;
444 }
445}
446
448{
449 mPatchShape = shape;
450}
451
453{
454 return mCustomSymbol.get();
455}
456
458{
459 mCustomSymbol.reset( symbol );
460}
461
463{
464 if ( !symbol )
465 return;
466
467 std::unique_ptr< QgsSymbol > s( symbol ); // this method takes ownership of symbol
468 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() );
469 if ( !vlayer || !vlayer->renderer() )
470 return;
471
472 mItem.setSymbol( s.get() ); // doesn't transfer ownership
473 vlayer->renderer()->setLegendSymbolItem( mItem.ruleKey(), s.release() ); // DOES transfer ownership!
474
475 mPixmap = QPixmap();
476
477 emit dataChanged();
478 vlayer->triggerRepaint();
479}
480
482{
483 double scale = 0.0;
484 double mupp = 0.0;
485 int dpi = 0;
486 if ( auto *lModel = model() )
487 lModel->legendMapViewData( &mupp, &dpi, &scale );
488
489 if ( qgsDoubleNear( mupp, 0.0 ) || dpi == 0 || qgsDoubleNear( scale, 0.0 ) )
490 return nullptr;
491
492 // setup temporary render context
493 std::unique_ptr<QgsRenderContext> context = std::make_unique<QgsRenderContext>( );
494 context->setScaleFactor( dpi / 25.4 );
495 context->setRendererScale( scale );
496 context->setMapToPixel( QgsMapToPixel( mupp ) );
497 context->setFlag( Qgis::RenderContextFlag::Antialiasing, true );
498 context->setFlag( Qgis::RenderContextFlag::RenderSymbolPreview, true );
499
500 if ( model() && !model()->targetScreenProperties().isEmpty() )
501 {
502 model()->targetScreenProperties().begin()->updateRenderContextForScreen( *context );
503 }
504
505 QgsExpressionContext expContext;
507 context->setExpressionContext( expContext );
508
509 return context.release();
510}
511
512void QgsLayerTreeModelLegendNode::checkAll( bool state )
513{
514 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() ) )
515 {
516 if ( !vlayer->renderer() )
517 return;
518
519 const QgsLegendSymbolList symbolList = vlayer->renderer()->legendSymbolItems();
520 for ( const auto &item : symbolList )
521 {
522 vlayer->renderer()->checkLegendSymbolItem( item.ruleKey(), state );
523 }
524
525 emit dataChanged();
526 vlayer->emitStyleChanged();
527 vlayer->triggerRepaint();
528 }
529 else if ( QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( mLayerNode->layer() ) )
530 {
531 if ( !pclayer->renderer() )
532 return;
533
534 const QStringList ruleKeys = pclayer->renderer()->legendRuleKeys();
535 for ( const QString &rule : ruleKeys )
536 {
537 pclayer->renderer()->checkLegendItem( rule, state );
538 }
539
540 emit dataChanged();
541 pclayer->emitStyleChanged();
542 pclayer->triggerRepaint();
543 }
544}
545
546QVariant QgsSymbolLegendNode::data( int role ) const
547{
548 if ( role == Qt::DisplayRole )
549 {
550 return mLabel;
551 }
552 else if ( role == Qt::EditRole )
553 {
554 return mUserLabel.isEmpty() ? mItem.label() : mUserLabel;
555 }
556 else if ( role == Qt::DecorationRole )
557 {
558 if ( mPixmap.isNull() )
559 {
560 QgsScreenProperties targetScreen = model() && !model()->targetScreenProperties().isEmpty()
562
563 if ( mItem.symbol() )
564 {
565 std::unique_ptr<QgsRenderContext> context( createTemporaryRenderContext() );
566
567 // unusued width, height variables
568 double width = 0.0;
569 double height = 0.0;
570 const std::unique_ptr<QgsSymbol> symbol( QgsSymbolLayerUtils::restrictedSizeSymbol( mItem.symbol(), MINIMUM_SIZE, MAXIMUM_SIZE, context.get(), width, height ) );
571 mPixmap = QgsSymbolLayerUtils::symbolPreviewPixmap( symbol ? symbol.get() : mItem.symbol(), mIconSize, 0, context.get(), false, nullptr, nullptr, targetScreen );
572
573 if ( !mTextOnSymbolLabel.isEmpty() && context )
574 {
575 QPainter painter( &mPixmap );
576 painter.setRenderHint( QPainter::Antialiasing );
577 context->setPainter( &painter );
578 bool isNullSize = false;
579 const QFontMetricsF fm( mTextOnSymbolTextFormat.scaledFont( *context, 1.0, &isNullSize ) );
580 if ( !isNullSize )
581 {
582 const qreal yBaselineVCenter = ( mIconSize.height() + fm.ascent() - fm.descent() ) / 2;
583 QgsTextRenderer::drawText( QPointF( mIconSize.width() / 2, yBaselineVCenter ), 0, Qgis::TextHorizontalAlignment::Center,
584 QStringList() << mTextOnSymbolLabel, *context, mTextOnSymbolTextFormat );
585 }
586 }
587 }
588 else
589 {
590 mPixmap = QPixmap( mIconSize * targetScreen.devicePixelRatio() );
591 mPixmap.fill( Qt::transparent );
592 }
593 }
594 return mPixmap;
595 }
596 else if ( role == Qt::CheckStateRole )
597 {
598 if ( !mItem.isCheckable() )
599 return QVariant();
600
601 if ( QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() ) )
602 {
603 if ( !vlayer->renderer() )
604 return QVariant();
605
606 return vlayer->renderer()->legendSymbolItemChecked( mItem.ruleKey() ) ? Qt::Checked : Qt::Unchecked;
607 }
608 }
609 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::RuleKey ) )
610 {
611 return mItem.ruleKey();
612 }
613 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::ParentRuleKey ) )
614 {
615 return mItem.parentRuleKey();
616 }
617 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ) )
618 {
620 }
621
622 return QVariant();
623}
624
625bool QgsSymbolLegendNode::setData( const QVariant &value, int role )
626{
627 if ( role != Qt::CheckStateRole )
628 return false;
629
630 if ( !mItem.isCheckable() )
631 return false;
632
633 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() );
634 if ( !vlayer || !vlayer->renderer() )
635 return false;
636
637 vlayer->renderer()->checkLegendSymbolItem( mItem.ruleKey(), value == Qt::Checked );
638
639 if ( QgsProject *project = vlayer->project() )
640 project->setDirty( true );
641
642 emit dataChanged();
643 vlayer->emitStyleChanged();
644
645 vlayer->triggerRepaint();
646
647 return true;
648}
649
650QSizeF QgsSymbolLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
651{
652 QgsSymbol *s = mCustomSymbol ? mCustomSymbol.get() : mItem.symbol();
653 if ( !s )
654 {
655 return QSizeF();
656 }
657
658 // setup temporary render context
659 QgsRenderContext *context = nullptr;
660 std::unique_ptr< QgsRenderContext > tempRenderContext;
662 if ( ctx && ctx->context )
663 context = ctx->context;
664 else
665 {
666 tempRenderContext = std::make_unique< QgsRenderContext >();
667 // QGIS 4.0 - make ItemContext compulsory, so we don't have to construct temporary render contexts here
669 tempRenderContext->setScaleFactor( settings.dpi() / 25.4 );
670 tempRenderContext->setRendererScale( settings.mapScale() );
671 tempRenderContext->setFlag( Qgis::RenderContextFlag::Antialiasing, true );
672 tempRenderContext->setMapToPixel( QgsMapToPixel( 1 / ( settings.mmPerMapUnit() * tempRenderContext->scaleFactor() ) ) );
674 tempRenderContext->setForceVectorOutput( true );
675 tempRenderContext->setPainter( ctx ? ctx->painter : nullptr );
676
677 // setup a minimal expression context
678 QgsExpressionContext expContext;
680 tempRenderContext->setExpressionContext( expContext );
681 context = tempRenderContext.get();
682 }
683
684 //Consider symbol size for point markers
685 const bool hasFixedWidth = ctx && ctx->patchSize.width() > 0;
686 const bool hasFixedHeight = ctx && ctx->patchSize.height() > 0;
687 const double desiredHeight = hasFixedHeight ? ctx->patchSize.height() : settings.symbolSize().height();
688 const double desiredWidth = hasFixedWidth ? ctx->patchSize.width() : settings.symbolSize().width();
689 double height = desiredHeight;
690 double width = desiredWidth;
691
692 //Center small marker symbols
693 double widthOffset = 0;
694 double heightOffset = 0;
695
696 const double maxSymbolSize = settings.maximumSymbolSize();
697 const double minSymbolSize = settings.minimumSymbolSize();
698
699 if ( QgsMarkerSymbol *markerSymbol = dynamic_cast<QgsMarkerSymbol *>( s ) )
700 {
701 const double size = markerSymbol->size( *context ) / context->scaleFactor();
702 if ( size > 0 )
703 {
704 if ( !hasFixedHeight )
705 height = size;
706 if ( !hasFixedWidth )
707 width = size;
708 }
709 }
710
711 bool restrictedSizeSymbolOK;
712 double restrictedSymbolWidth = width;
713 double restrictedSymbolHeight = height;
714 const std::unique_ptr<QgsSymbol> minMaxSizeSymbol( QgsSymbolLayerUtils::restrictedSizeSymbol( s, minSymbolSize, maxSymbolSize, context, restrictedSymbolWidth, restrictedSymbolHeight, &restrictedSizeSymbolOK ) );
715 if ( minMaxSizeSymbol )
716 {
717 s = minMaxSizeSymbol.get();
718 if ( !hasFixedHeight )
719 height = restrictedSymbolHeight;
720 if ( !hasFixedWidth )
721 width = restrictedSymbolWidth;
722 }
723
724 if ( s->type() == Qgis::SymbolType::Marker )
725 {
726 if ( width < desiredWidth )
727 {
728 widthOffset = ( desiredWidth - width ) / 2.0;
729 }
730 if ( height < desiredHeight )
731 {
732 heightOffset = ( desiredHeight - height ) / 2.0;
733 }
734 }
735 if ( ctx && ctx->painter )
736 {
737 const double currentYCoord = ctx->top + ( itemHeight - desiredHeight ) / 2;
738 QPainter *p = ctx->painter;
739
740 //setup painter scaling to dots so that raster symbology is drawn to scale
741 const double dotsPerMM = context->scaleFactor();
742
743 int opacity = 255;
744 if ( QgsMapLayer *layer = layerNode()->layer() )
745 opacity = static_cast<int >( std::round( 255 * layer->opacity() ) );
746
747 const QgsScopedQPainterState painterState( p );
748 context->setPainterFlagsUsingContext( p );
749
750 switch ( settings.symbolAlignment() )
751 {
752 case Qt::AlignLeft:
753 default:
754 p->translate( ctx->columnLeft + widthOffset, currentYCoord + heightOffset );
755 break;
756 case Qt::AlignRight:
757 p->translate( ctx->columnRight - widthOffset - width, currentYCoord + heightOffset );
758 break;
759 }
760
761 p->scale( 1.0 / dotsPerMM, 1.0 / dotsPerMM );
763 // QGIS 4.0 -- ctx->context will be mandatory
764 const bool useAdvancedEffects = ctx->context ? ctx->context->flags() & Qgis::RenderContextFlag::UseAdvancedEffects : settings.useAdvancedEffects();
766
767 if ( opacity != 255 && useAdvancedEffects )
768 {
769 // if this is a semi transparent layer, we need to draw symbol to an image (to flatten it first)
770
771 const int maxBleed = static_cast< int >( std::ceil( QgsSymbolLayerUtils::estimateMaxSymbolBleed( s, *context ) ) );
772
773 // create image which is same size as legend rect, in case symbol bleeds outside its allotted space
774 const QSize symbolSize( static_cast< int >( std::round( width * dotsPerMM ) ), static_cast<int >( std::round( height * dotsPerMM ) ) );
775 const QSize tempImageSize( symbolSize.width() + maxBleed * 2, symbolSize.height() + maxBleed * 2 );
776 QImage tempImage = QImage( tempImageSize, QImage::Format_ARGB32 );
777 tempImage.fill( Qt::transparent );
778 QPainter imagePainter( &tempImage );
779 context->setPainterFlagsUsingContext( &imagePainter );
780
781 context->setPainter( &imagePainter );
782 imagePainter.translate( maxBleed, maxBleed );
783 s->drawPreviewIcon( &imagePainter, symbolSize, context, false, nullptr, &patchShape, ctx->screenProperties );
784 imagePainter.translate( -maxBleed, -maxBleed );
785 context->setPainter( ctx->painter );
786 //reduce opacity of image
787 imagePainter.setCompositionMode( QPainter::CompositionMode_DestinationIn );
788 imagePainter.fillRect( tempImage.rect(), QColor( 0, 0, 0, opacity ) );
789 imagePainter.end();
790 //draw rendered symbol image
791 p->drawImage( -maxBleed, -maxBleed, tempImage );
792 }
793 else if ( !restrictedSizeSymbolOK )
794 {
795 // if there is no restricted symbol size (because of geometry generator mainly) we need to ensure
796 // that there is no drawing outside the given size
797 const int maxBleed = static_cast< int >( std::ceil( QgsSymbolLayerUtils::estimateMaxSymbolBleed( s, *context ) ) );
798 const QSize symbolSize( static_cast< int >( std::round( width * dotsPerMM ) ), static_cast<int >( std::round( height * dotsPerMM ) ) );
799 const QSize maxSize( symbolSize.width() + maxBleed * 2, symbolSize.height() + maxBleed * 2 );
800 p->save();
801 p->setClipRect( -maxBleed, -maxBleed, maxSize.width(), maxSize.height(), Qt::IntersectClip );
802 s->drawPreviewIcon( p, symbolSize, context, false, nullptr, &patchShape, ctx->screenProperties );
803 p->restore();
804 }
805 else
806 {
807 s->drawPreviewIcon( p, QSize( static_cast< int >( std::round( width * dotsPerMM ) ), static_cast< int >( std::round( height * dotsPerMM ) ) ), context, false, nullptr, &patchShape, ctx->screenProperties );
808 }
809
810 if ( !mTextOnSymbolLabel.isEmpty() )
811 {
812 bool isNullSize = false;
813 const QFontMetricsF fm( mTextOnSymbolTextFormat.scaledFont( *context, 1.0, &isNullSize ) );
814 if ( !isNullSize )
815 {
816 const qreal yBaselineVCenter = ( height * dotsPerMM + fm.ascent() - fm.descent() ) / 2;
817 QgsTextRenderer::drawText( QPointF( width * dotsPerMM / 2, yBaselineVCenter ), 0, Qgis::TextHorizontalAlignment::Center,
818 QStringList() << mTextOnSymbolLabel, *context, mTextOnSymbolTextFormat );
819 }
820 }
821 }
822
823 return QSizeF( std::max( width + 2 * widthOffset, static_cast< double >( desiredWidth ) ),
824 std::max( height + 2 * heightOffset, static_cast< double >( desiredHeight ) ) );
825}
826
827QJsonObject QgsSymbolLegendNode::exportSymbolToJson( const QgsLegendSettings &settings, const QgsRenderContext &context ) const
828{
829 QJsonObject json;
830 if ( mItem.scaleMaxDenom() > 0 )
831 {
832 json[ QStringLiteral( "scaleMaxDenom" ) ] = mItem.scaleMaxDenom();
833 }
834 if ( mItem.scaleMinDenom() > 0 )
835 {
836 json[ QStringLiteral( "scaleMinDenom" ) ] = mItem.scaleMinDenom();
837 }
838
839 const QgsSymbol *s = mCustomSymbol ? mCustomSymbol.get() : mItem.symbol();
840 if ( !s )
841 {
842 return json;
843 }
844
845
847 // QGIS 4.0 - use render context directly here, and note in the dox that the context must be correctly setup
849 ctx.setScaleFactor( settings.dpi() / 25.4 );
850 ctx.setRendererScale( settings.mapScale() );
851 ctx.setMapToPixel( QgsMapToPixel( 1 / ( settings.mmPerMapUnit() * ctx.scaleFactor() ) ) );
852 ctx.setForceVectorOutput( true );
855
857
858 // ensure that a minimal expression context is available
859 QgsExpressionContext expContext = context.expressionContext();
861 ctx.setExpressionContext( expContext );
862
863 const QPixmap pix = QgsSymbolLayerUtils::symbolPreviewPixmap( mItem.symbol(), minimumIconSize(), 0, &ctx );
864 QImage img( pix.toImage().convertToFormat( QImage::Format_ARGB32_Premultiplied ) );
865
866 int opacity = 255;
867 if ( QgsMapLayer *layer = layerNode()->layer() )
868 opacity = ( 255 * layer->opacity() );
869
870 if ( opacity != 255 )
871 {
872 QPainter painter;
873 painter.begin( &img );
874 painter.setCompositionMode( QPainter::CompositionMode_DestinationIn );
875 painter.fillRect( pix.rect(), QColor( 0, 0, 0, opacity ) );
876 painter.end();
877 }
878
879 QByteArray byteArray;
880 QBuffer buffer( &byteArray );
881 img.save( &buffer, "PNG" );
882 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
883
884 json[ QStringLiteral( "icon" ) ] = base64;
885 return json;
886}
887
889{
891 updateLabel();
892}
893
894
896{
897 if ( mSymbolUsesMapUnits )
898 {
899 mPixmap = QPixmap();
900 emit dataChanged();
901 }
902}
903
905{
906 if ( mIconSize == sz )
907 return;
908
909 mIconSize = sz;
910 mPixmap = QPixmap();
911 emit dataChanged();
912}
913
914void QgsSymbolLegendNode::updateLabel()
915{
916 if ( !mLayerNode )
917 return;
918
919 const bool showFeatureCount = mLayerNode->customProperty( QStringLiteral( "showFeatureCount" ), 0 ).toBool();
920 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() );
921 if ( !mLayerNode->labelExpression().isEmpty() )
922 mLabel = "[%" + mLayerNode->labelExpression() + "%]";
923 else
924 mLabel = symbolLabel();
925
926 if ( showFeatureCount && vl )
927 {
928 const bool estimatedCount = vl->dataProvider() ? QgsDataSourceUri( vl->dataProvider()->dataSourceUri() ).useEstimatedMetadata() : false;
929 const qlonglong count = mEmbeddedInParent ? vl->featureCount() : vl->featureCount( mItem.ruleKey() ) ;
930
931 // if you modify this line, please update QgsLayerTreeModel::data (DisplayRole)
932 mLabel += QStringLiteral( " [%1%2]" ).arg(
933 estimatedCount ? QStringLiteral( "≈" ) : QString(),
934 count != -1 ? QLocale().toString( count ) : tr( "N/A" ) );
935 }
936
937 emit dataChanged();
938}
939
940QString QgsSymbolLegendNode::evaluateLabel( const QgsExpressionContext &context, const QString &label )
941{
942 if ( !mLayerNode )
943 return QString();
944
945 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() );
946
947 if ( vl )
948 {
949 QgsExpressionContext contextCopy = QgsExpressionContext( context );
951 contextCopy.appendScope( symbolScope );
952 contextCopy.appendScope( vl->createExpressionContextScope() );
953
954 if ( label.isEmpty() )
955 {
956 const QString symLabel = symbolLabel();
957 if ( ! mLayerNode->labelExpression().isEmpty() )
958 mLabel = QgsExpression::replaceExpressionText( "[%" + mLayerNode->labelExpression() + "%]", &contextCopy );
959 else if ( symLabel.contains( "[%" ) )
960 mLabel = QgsExpression::replaceExpressionText( symLabel, &contextCopy );
961 return mLabel;
962 }
963 else
964 {
965 QString eLabel;
966 if ( ! mLayerNode->labelExpression().isEmpty() )
967 eLabel = QgsExpression::replaceExpressionText( label + "[%" + mLayerNode->labelExpression() + "%]", &contextCopy );
968 else if ( label.contains( "[%" ) )
969 eLabel = QgsExpression::replaceExpressionText( label, &contextCopy );
970 return eLabel;
971 }
972 }
973 return mLabel;
974}
975
977{
978 QgsVectorLayer *vl = qobject_cast<QgsVectorLayer *>( mLayerNode->layer() );
979
980 QgsExpressionContextScope *scope = new QgsExpressionContextScope( tr( "Symbol scope" ) );
981 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "symbol_label" ), symbolLabel().remove( "[%" ).remove( "%]" ), true ) );
982 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "symbol_id" ), mItem.ruleKey(), true ) );
983 if ( vl )
984 {
985 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "symbol_count" ), QVariant::fromValue( vl->featureCount( mItem.ruleKey() ) ), true ) );
986 }
987 return scope;
988}
989
990// -------------------------------------------------------------------------
991
992
993QgsSimpleLegendNode::QgsSimpleLegendNode( QgsLayerTreeLayer *nodeLayer, const QString &label, const QIcon &icon, QObject *parent, const QString &key )
994 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
995 , mLabel( label )
996 , mIcon( icon )
997 , mKey( key )
998{
999}
1000
1001QVariant QgsSimpleLegendNode::data( int role ) const
1002{
1003 if ( role == Qt::DisplayRole || role == Qt::EditRole )
1004 return mUserLabel.isEmpty() ? mLabel : mUserLabel;
1005 else if ( role == Qt::DecorationRole )
1006 return mIcon;
1007 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::RuleKey ) && !mKey.isEmpty() )
1008 return mKey;
1009 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ) )
1011 else
1012 return QVariant();
1013}
1014
1015
1016// -------------------------------------------------------------------------
1017
1018QgsImageLegendNode::QgsImageLegendNode( QgsLayerTreeLayer *nodeLayer, const QImage &img, QObject *parent )
1019 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
1020 , mImage( img )
1021{
1022}
1023
1024QVariant QgsImageLegendNode::data( int role ) const
1025{
1026 if ( role == Qt::DecorationRole )
1027 {
1028 return QPixmap::fromImage( mImage );
1029 }
1030 else if ( role == Qt::SizeHintRole )
1031 {
1032 return mImage.size();
1033 }
1034 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ) )
1035 {
1037 }
1038 return QVariant();
1039}
1040
1041QSizeF QgsImageLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
1042{
1043 Q_UNUSED( itemHeight )
1044
1045 if ( ctx && ctx->painter && ctx->context )
1046 {
1047 const QgsScopedRenderContextScaleToPixels scopedScaleToPixels( *( ctx->context ) );
1048 const double scaleFactor = ctx->context->scaleFactor();
1049 const double imgWidth = settings.wmsLegendSize().width() * scaleFactor;
1050 const double imgHeight = settings.wmsLegendSize().height() * scaleFactor;
1051
1052 const QImage scaledImg = mImage.scaled( QSizeF( imgWidth, imgHeight ).toSize(), Qt::KeepAspectRatio, Qt::SmoothTransformation );
1053 switch ( settings.symbolAlignment() )
1054 {
1055 case Qt::AlignLeft:
1056 default:
1057 ctx->painter->drawImage( QPointF( ctx->columnLeft * scaleFactor, ctx->top * scaleFactor ), scaledImg );
1058 break;
1059
1060 case Qt::AlignRight:
1061 ctx->painter->drawImage( QPointF( ctx->columnRight * scaleFactor - imgWidth, ctx->top * scaleFactor ), scaledImg );
1062 break;
1063 }
1064 }
1065 return settings.wmsLegendSize();
1066}
1067
1069{
1070 QByteArray byteArray;
1071 QBuffer buffer( &byteArray );
1072 mImage.save( &buffer, "PNG" );
1073 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
1074
1075 QJsonObject json;
1076 json[ QStringLiteral( "icon" ) ] = base64;
1077 return json;
1078}
1079
1080// -------------------------------------------------------------------------
1081
1082QgsRasterSymbolLegendNode::QgsRasterSymbolLegendNode( QgsLayerTreeLayer *nodeLayer, const QColor &color, const QString &label, QObject *parent, bool isCheckable, const QString &ruleKey, const QString &parentRuleKey )
1083 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
1084 , mColor( color )
1085 , mLabel( label )
1086 , mCheckable( isCheckable )
1087 , mRuleKey( ruleKey )
1088 , mParentRuleKey( parentRuleKey )
1089{
1090}
1091
1093{
1094 if ( mCheckable )
1095 return Qt::ItemIsEnabled | Qt::ItemIsUserCheckable | Qt::ItemIsSelectable;
1096 else
1097 return Qt::ItemIsEnabled | Qt::ItemIsSelectable;
1098}
1099
1100QVariant QgsRasterSymbolLegendNode::data( int role ) const
1101{
1102 switch ( role )
1103 {
1104 case Qt::DecorationRole:
1105 {
1106 const int iconSize = QgsLayerTreeModel::scaleIconSize( 16 ); // TODO: configurable?
1107 QPixmap pix( iconSize, iconSize );
1108 pix.fill( mColor );
1109 return QIcon( pix );
1110 }
1111
1112 case Qt::DisplayRole:
1113 case Qt::EditRole:
1114 return mUserLabel.isEmpty() ? mLabel : mUserLabel;
1115
1116 case static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ):
1118
1119 case static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::RuleKey ):
1120 return mRuleKey;
1121
1123 return mParentRuleKey;
1124
1125 case Qt::CheckStateRole:
1126 {
1127 if ( !mCheckable )
1128 return QVariant();
1129
1130 if ( QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( mLayerNode->layer() ) )
1131 {
1132 if ( !pclayer->renderer() )
1133 return QVariant();
1134
1135 return pclayer->renderer()->legendItemChecked( mRuleKey ) ? Qt::Checked : Qt::Unchecked;
1136 }
1137
1138 return QVariant();
1139 }
1140
1141 default:
1142 return QVariant();
1143 }
1144}
1145
1146bool QgsRasterSymbolLegendNode::setData( const QVariant &value, int role )
1147{
1148 if ( role != Qt::CheckStateRole )
1149 return false;
1150
1151 if ( !mCheckable )
1152 return false;
1153
1154 if ( QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( mLayerNode->layer() ) )
1155 {
1156 if ( !pclayer->renderer() )
1157 return false;
1158
1159 pclayer->renderer()->checkLegendItem( mRuleKey, value == Qt::Checked );
1160
1161 emit dataChanged();
1162 pclayer->emitStyleChanged();
1163
1164 pclayer->triggerRepaint();
1165 if ( pclayer->sync3DRendererTo2DRenderer() )
1166 pclayer->convertRenderer3DFromRenderer2D();
1167
1168 return true;
1169 }
1170 else
1171 {
1172 return false;
1173 }
1174}
1175
1176
1177QSizeF QgsRasterSymbolLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
1178{
1179 QSizeF size = settings.symbolSize();
1180 double offsetX = 0;
1181 if ( ctx )
1182 {
1183 if ( ctx->patchSize.width() > 0 )
1184 {
1185 if ( ctx->patchSize.width() < size.width() )
1186 offsetX = ( size.width() - ctx->patchSize.width() ) / 2.0;
1187 size.setWidth( ctx->patchSize.width() );
1188 }
1189 if ( ctx->patchSize.height() > 0 )
1190 {
1191 size.setHeight( ctx->patchSize.height() );
1192 }
1193 }
1194
1195 if ( ctx && ctx->painter )
1196 {
1197 QColor itemColor = mColor;
1198 if ( QgsRasterLayer *rasterLayer = qobject_cast<QgsRasterLayer *>( layerNode()->layer() ) )
1199 {
1200 if ( QgsRasterRenderer *rasterRenderer = rasterLayer->renderer() )
1201 itemColor.setAlpha( rasterRenderer->opacity() * 255.0 );
1202 }
1203 ctx->painter->setBrush( itemColor );
1204
1205 if ( settings.drawRasterStroke() )
1206 {
1207 QPen pen;
1208 pen.setColor( settings.rasterStrokeColor() );
1209 pen.setWidthF( settings.rasterStrokeWidth() );
1210 pen.setJoinStyle( Qt::MiterJoin );
1211 ctx->painter->setPen( pen );
1212 }
1213 else
1214 {
1215 ctx->painter->setPen( Qt::NoPen );
1216 }
1217
1218 switch ( settings.symbolAlignment() )
1219 {
1220 case Qt::AlignLeft:
1221 default:
1222 ctx->painter->drawRect( QRectF( ctx->columnLeft + offsetX, ctx->top + ( itemHeight - size.height() ) / 2,
1223 size.width(), size.height() ) );
1224 break;
1225
1226 case Qt::AlignRight:
1227 ctx->painter->drawRect( QRectF( ctx->columnRight - size.width() - offsetX, ctx->top + ( itemHeight - size.height() ) / 2,
1228 size.width(), size.height() ) );
1229 break;
1230 }
1231 }
1232 return size;
1233}
1234
1236{
1237 QImage img = QImage( settings.symbolSize().toSize(), QImage::Format_ARGB32 );
1238 img.fill( Qt::transparent );
1239
1240 QPainter painter( &img );
1241 painter.setRenderHint( QPainter::Antialiasing );
1242
1243 QColor itemColor = mColor;
1244 if ( QgsRasterLayer *rasterLayer = qobject_cast<QgsRasterLayer *>( layerNode()->layer() ) )
1245 {
1246 if ( QgsRasterRenderer *rasterRenderer = rasterLayer->renderer() )
1247 itemColor.setAlpha( rasterRenderer->opacity() * 255.0 );
1248 }
1249 painter.setBrush( itemColor );
1250
1251 if ( settings.drawRasterStroke() )
1252 {
1253 QPen pen;
1254 pen.setColor( settings.rasterStrokeColor() );
1255 pen.setWidthF( settings.rasterStrokeWidth() );
1256 pen.setJoinStyle( Qt::MiterJoin );
1257 painter.setPen( pen );
1258 }
1259 else
1260 {
1261 painter.setPen( Qt::NoPen );
1262 }
1263
1264 painter.drawRect( QRectF( 0, 0, settings.symbolSize().width(), settings.symbolSize().height() ) );
1265
1266 QByteArray byteArray;
1267 QBuffer buffer( &byteArray );
1268 img.save( &buffer, "PNG" );
1269 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
1270
1271 QJsonObject json;
1272 json[ QStringLiteral( "icon" ) ] = base64;
1273 return json;
1274}
1275
1276// -------------------------------------------------------------------------
1277
1279 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
1280 , mValid( false )
1281{
1282}
1283
1285
1286QImage QgsWmsLegendNode::getLegendGraphic( bool synchronous ) const
1287{
1288 if ( ! mValid && ! mFetcher )
1289 {
1290 // or maybe in presence of a downloader we should just delete it
1291 // and start a new one ?
1292
1293 QgsRasterLayer *layer = qobject_cast<QgsRasterLayer *>( mLayerNode->layer() );
1294
1295 if ( layer && layer->isValid() )
1296 {
1297 const QgsLayerTreeModel *mod = model();
1298 if ( ! mod )
1299 return mImage;
1300 const QgsMapSettings *ms = mod->legendFilterMapSettings();
1301
1302 QgsRasterDataProvider *prov = layer->dataProvider();
1303 if ( ! prov )
1304 return mImage;
1305
1306 Q_ASSERT( ! mFetcher );
1307 mFetcher.reset( prov->getLegendGraphicFetcher( ms ) );
1308 if ( mFetcher )
1309 {
1310 connect( mFetcher.get(), &QgsImageFetcher::finish, this, &QgsWmsLegendNode::getLegendGraphicFinished );
1311 connect( mFetcher.get(), &QgsImageFetcher::error, this, &QgsWmsLegendNode::getLegendGraphicErrored );
1312 connect( mFetcher.get(), &QgsImageFetcher::progress, this, &QgsWmsLegendNode::getLegendGraphicProgress );
1313 mFetcher->start();
1314 if ( synchronous )
1315 {
1316 QEventLoop loop;
1317 // The slots getLegendGraphicFinished and getLegendGraphicErrored will destroy the fetcher
1318 connect( mFetcher.get(), &QObject::destroyed, &loop, &QEventLoop::quit );
1319 loop.exec();
1320 }
1321 }
1322 }
1323 else
1324 {
1325 QgsDebugError( QStringLiteral( "Failed to download legend graphics: layer is not valid." ) );
1326 }
1327 }
1328
1329 return mImage;
1330}
1331
1332QVariant QgsWmsLegendNode::data( int role ) const
1333{
1334 if ( role == Qt::DecorationRole )
1335 {
1336 return QPixmap::fromImage( getLegendGraphic() );
1337 }
1338 else if ( role == Qt::SizeHintRole )
1339 {
1340 return getLegendGraphic().size();
1341 }
1342 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ) )
1343 {
1345 }
1346 return QVariant();
1347}
1348
1349QSizeF QgsWmsLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
1350{
1351 Q_UNUSED( itemHeight )
1352
1353 const QImage image = getLegendGraphic( settings.synchronousLegendRequests() );
1354
1355 double px2mm = 1000. / image.dotsPerMeterX();
1356 double mmWidth = image.width() * px2mm;
1357 double mmHeight = image.height() * px2mm;
1358
1359 QSize targetSize = QSize( mmWidth, mmHeight );
1360 if ( settings.wmsLegendSize().width() < mmWidth )
1361 {
1362 double targetHeight = mmHeight * settings.wmsLegendSize().width() / mmWidth;
1363 targetSize = QSize( settings.wmsLegendSize().width(), targetHeight );
1364 }
1365 else if ( settings.wmsLegendSize().height() < mmHeight )
1366 {
1367 double targetWidth = mmWidth * settings.wmsLegendSize().height() / mmHeight;
1368 targetSize = QSize( targetWidth, settings.wmsLegendSize().height() );
1369 }
1370
1371 if ( ctx && ctx->painter )
1372 {
1373 QImage smoothImage = image.scaled( targetSize / px2mm, Qt::KeepAspectRatio, Qt::SmoothTransformation );
1374
1375 switch ( settings.symbolAlignment() )
1376 {
1377 case Qt::AlignLeft:
1378 default:
1379 ctx->painter->drawImage( QRectF( ctx->columnLeft,
1380 ctx->top,
1381 targetSize.width(),
1382 targetSize.height() ),
1383 smoothImage,
1384 QRectF( QPointF( 0, 0 ), smoothImage.size() ) );
1385 break;
1386
1387 case Qt::AlignRight:
1388 ctx->painter->drawImage( QRectF( ctx->columnRight - settings.wmsLegendSize().width(),
1389 ctx->top,
1390 targetSize.width(),
1391 targetSize.height() ),
1392 smoothImage,
1393 QRectF( QPointF( 0, 0 ), smoothImage.size() ) );
1394 break;
1395 }
1396 }
1397 return targetSize;
1398}
1399
1401{
1402 QByteArray byteArray;
1403 QBuffer buffer( &byteArray );
1404 mImage.save( &buffer, "PNG" );
1405 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
1406
1407 QJsonObject json;
1408 json[ QStringLiteral( "icon" ) ] = base64;
1409 return json;
1410}
1411
1412QImage QgsWmsLegendNode::renderMessage( const QString &msg ) const
1413{
1414 const int fontHeight = 10;
1415 const int margin = fontHeight / 2;
1416 const int nlines = 1;
1417
1418 const int w = 512, h = fontHeight * nlines + margin * ( nlines + 1 );
1419 QImage image( w, h, QImage::Format_ARGB32_Premultiplied );
1420 QPainter painter;
1421 painter.begin( &image );
1422 painter.setPen( QColor( 255, 0, 0 ) );
1423 painter.setFont( QFont( QStringLiteral( "Chicago" ), fontHeight ) );
1424 painter.fillRect( 0, 0, w, h, QColor( 255, 255, 255 ) );
1425 painter.drawText( 0, margin + fontHeight, msg );
1426 //painter.drawText(0,2*(margin+fontHeight),tr("retrying in 5 seconds…"));
1427 painter.end();
1428
1429 return image;
1430}
1431
1432void QgsWmsLegendNode::getLegendGraphicProgress( qint64 cur, qint64 tot )
1433{
1434 const QString msg = tot > 0 ? tr( "Downloading: %1% (%2)" ).arg( static_cast< int >( std::round( 100 * cur / tot ) ) ).arg( QgsFileUtils::representFileSize( tot ) )
1435 : tr( "Downloading: %1" ).arg( QgsFileUtils::representFileSize( cur ) );
1436 mImage = renderMessage( msg );
1437 emit dataChanged();
1438}
1439
1440void QgsWmsLegendNode::getLegendGraphicErrored( const QString & )
1441{
1442 if ( ! mFetcher )
1443 return; // must be coming after finish
1444
1445 mImage = QImage();
1446 emit dataChanged();
1447
1448 mFetcher.reset();
1449
1450 mValid = true; // we consider it valid anyway
1451}
1452
1453void QgsWmsLegendNode::getLegendGraphicFinished( const QImage &image )
1454{
1455 if ( ! mFetcher )
1456 return; // must be coming after error
1457
1458 if ( ! image.isNull() )
1459 {
1460 if ( image != mImage )
1461 {
1462 mImage = image;
1463 setUserPatchSize( mImage.size() );
1464 emit dataChanged();
1465 }
1466 mValid = true; // only if not null I guess
1467 }
1468 mFetcher.reset();
1469}
1470
1472{
1473 // TODO: do this only if this extent != prev extent ?
1474 mValid = false;
1475 emit dataChanged();
1476}
1477
1479{
1480 return getLegendGraphic( true );
1481}
1482
1483// -------------------------------------------------------------------------
1484
1486 : QgsLayerTreeModelLegendNode( nodeLayer, parent )
1487 , mSettings( new QgsDataDefinedSizeLegend( settings ) )
1488{
1489}
1490
1495
1496QVariant QgsDataDefinedSizeLegendNode::data( int role ) const
1497{
1498 if ( role == Qt::DecorationRole )
1499 {
1500 cacheImage();
1501 return QPixmap::fromImage( mImage );
1502 }
1503 else if ( role == Qt::SizeHintRole )
1504 {
1505 cacheImage();
1506 return mImage.size();
1507 }
1508 else if ( role == static_cast< int >( QgsLayerTreeModelLegendNode::CustomRole::NodeType ) )
1509 {
1511 }
1512 return QVariant();
1513}
1514
1516{
1517 // setup temporary render context if none specified
1518 QgsRenderContext *context = nullptr;
1519 std::unique_ptr< QgsRenderContext > tempRenderContext;
1520 if ( ctx && ctx->context )
1521 context = ctx->context;
1522 else
1523 {
1524 tempRenderContext = std::make_unique< QgsRenderContext >();
1525 // QGIS 4.0 - make ItemContext compulsory, so we don't have to construct temporary render contexts here
1527 tempRenderContext->setScaleFactor( settings.dpi() / 25.4 );
1528 tempRenderContext->setRendererScale( settings.mapScale() );
1529 tempRenderContext->setFlag( Qgis::RenderContextFlag::Antialiasing, true );
1530 tempRenderContext->setMapToPixel( QgsMapToPixel( 1 / ( settings.mmPerMapUnit() * tempRenderContext->scaleFactor() ) ) );
1531 tempRenderContext->setForceVectorOutput( true );
1532 tempRenderContext->setPainter( ctx ? ctx->painter : nullptr );
1533 tempRenderContext->setFlag( Qgis::RenderContextFlag::Antialiasing, true );
1535
1536 // setup a minimal expression context
1537 QgsExpressionContext expContext;
1539 tempRenderContext->setExpressionContext( expContext );
1540 context = tempRenderContext.get();
1541 }
1542
1543 if ( context->painter() )
1544 {
1545 context->painter()->save();
1546 context->painter()->translate( ctx->columnLeft, ctx->top );
1547
1548 // scale to pixels
1549 context->painter()->scale( 1 / context->scaleFactor(), 1 / context->scaleFactor() );
1550 }
1551
1552 QgsDataDefinedSizeLegend ddsLegend( *mSettings );
1553 ddsLegend.setFont( settings.style( QgsLegendStyle::SymbolLabel ).textFormat().toQFont() );
1554 ddsLegend.setTextColor( settings.style( QgsLegendStyle::SymbolLabel ).textFormat().color() );
1555
1556 QSizeF contentSize;
1557 double labelXOffset;
1558 ddsLegend.drawCollapsedLegend( *context, &contentSize, &labelXOffset );
1559
1560 if ( context->painter() )
1561 context->painter()->restore();
1562
1563 ItemMetrics im;
1564 im.symbolSize = QSizeF( ( contentSize.width() - labelXOffset ) / context->scaleFactor(), contentSize.height() / context->scaleFactor() );
1565 im.labelSize = QSizeF( labelXOffset / context->scaleFactor(), contentSize.height() / context->scaleFactor() );
1566 return im;
1567}
1568
1569
1570void QgsDataDefinedSizeLegendNode::cacheImage() const
1571{
1572 if ( mImage.isNull() )
1573 {
1574 std::unique_ptr<QgsRenderContext> context( createTemporaryRenderContext() );
1575 if ( !context )
1576 {
1577 context.reset( new QgsRenderContext );
1578 Q_ASSERT( context ); // to make cppcheck happy
1579 context->setScaleFactor( 96 / 25.4 );
1580 }
1581 mImage = mSettings->collapsedLegendImage( *context );
1582 }
1583}
1584
1585QgsVectorLabelLegendNode::QgsVectorLabelLegendNode( QgsLayerTreeLayer *nodeLayer, const QgsPalLayerSettings &labelSettings, QObject *parent ): QgsLayerTreeModelLegendNode( nodeLayer, parent ), mLabelSettings( labelSettings )
1586{
1587}
1588
1592
1593QVariant QgsVectorLabelLegendNode::data( int role ) const
1594{
1595 if ( role == Qt::DisplayRole )
1596 {
1597 return mUserLabel;
1598 }
1599 if ( role == Qt::DecorationRole )
1600 {
1601 const int iconSize = QgsLayerTreeModel::scaleIconSize( 16 );
1602 return QgsPalLayerSettings::labelSettingsPreviewPixmap( mLabelSettings, QSize( iconSize, iconSize ), mLabelSettings.legendString() );
1603 }
1604 return QVariant();
1605}
1606
1607QSizeF QgsVectorLabelLegendNode::drawSymbol( const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight ) const
1608{
1609 Q_UNUSED( itemHeight );
1610 if ( !ctx )
1611 {
1612 return QSizeF( 0, 0 );
1613 }
1614
1615 const QgsRenderContext *renderContext = ctx->context;
1616 if ( renderContext )
1617 {
1618 return drawSymbol( settings, *renderContext, ctx->columnLeft, ctx->top );
1619 }
1620
1621 return QSizeF( 0, 0 );
1622}
1623
1624QSizeF QgsVectorLabelLegendNode::drawSymbol( const QgsLegendSettings &settings, const QgsRenderContext &renderContext, double xOffset, double yOffset ) const
1625{
1626 const QStringList textLines( mLabelSettings.legendString() );
1627 const QgsTextFormat textFormat = mLabelSettings.format();
1628 QgsRenderContext ctx( renderContext );
1629 double textWidth, textHeight;
1630 textWidthHeight( textWidth, textHeight, ctx, textFormat, textLines );
1631 textWidth /= renderContext.scaleFactor();
1632 textHeight /= renderContext.scaleFactor();
1633 const QPointF textPos( renderContext.scaleFactor() * ( xOffset + settings.symbolSize().width() / 2.0 - textWidth / 2.0 ), renderContext.scaleFactor() * ( yOffset + settings.symbolSize().height() / 2.0 + textHeight / 2.0 ) );
1634
1635 const QgsScopedRenderContextScaleToPixels scopedScaleToPixels( ctx );
1636 QgsTextRenderer::drawText( textPos, 0.0, Qgis::TextHorizontalAlignment::Left, textLines, ctx, textFormat );
1637
1638 const double symbolWidth = std::max( textWidth, settings.symbolSize().width() );
1639 const double symbolHeight = std::max( textHeight, settings.symbolSize().height() );
1640 return QSizeF( symbolWidth, symbolHeight );
1641}
1642
1643QJsonObject QgsVectorLabelLegendNode::exportSymbolToJson( const QgsLegendSettings &settings, const QgsRenderContext &context ) const
1644{
1645 Q_UNUSED( settings );
1646
1647 const double mmToPixel = 96.0 / 25.4; //settings.dpi() is deprecated
1648
1649 const QStringList textLines( mLabelSettings.legendString() );
1650 const QgsTextFormat textFormat = mLabelSettings.format();
1651 QgsRenderContext ctx( context );
1652 ctx.setScaleFactor( mmToPixel );
1653
1654 double textWidth, textHeight;
1655 textWidthHeight( textWidth, textHeight, ctx, textFormat, textLines );
1656 const QPixmap previewPixmap = QgsPalLayerSettings::labelSettingsPreviewPixmap( mLabelSettings, QSize( textWidth, textHeight ), mLabelSettings.legendString() );
1657
1658 QByteArray byteArray;
1659 QBuffer buffer( &byteArray );
1660 previewPixmap.save( &buffer, "PNG" );
1661 const QString base64 = QString::fromLatin1( byteArray.toBase64().data() );
1662
1663 QJsonObject json;
1664 json[ QStringLiteral( "icon" ) ] = base64;
1665 return json;
1666}
1667
1668void QgsVectorLabelLegendNode::textWidthHeight( double &width, double &height, QgsRenderContext &ctx, const QgsTextFormat &textFormat, const QStringList &textLines ) const
1669{
1670 QFontMetricsF fm = QgsTextRenderer::fontMetrics( ctx, textFormat );
1671 height = QgsTextRenderer::textHeight( ctx, textFormat, 'A', true );
1672 width = QgsTextRenderer::textWidth( ctx, textFormat, textLines, &fm );
1673}
@ Point
Text at point of origin layout mode.
@ RectangleCapHeightBased
Similar to Rectangle mode, but uses cap height only when calculating font heights for the first line ...
@ Horizontal
Horizontally oriented text.
@ RenderSymbolPreview
The render is for a symbol preview only and map based properties may not be available,...
@ LosslessImageRendering
Render images losslessly whenever possible, instead of the default lossy jpeg rendering used for some...
@ Antialiasing
Use antialiasing while drawing.
@ UseAdvancedEffects
Enable layer opacity and blending effects.
@ Marker
Marker symbol.
@ Line
Line symbol.
TextHorizontalAlignment
Text horizontal alignment.
Definition qgis.h:2803
QgsDataDefinedSizeLegendNode(QgsLayerTreeLayer *nodeLayer, const QgsDataDefinedSizeLegend &settings, QObject *parent=nullptr)
Construct the node using QgsDataDefinedSizeLegend as definition of the node's appearance.
ItemMetrics draw(const QgsLegendSettings &settings, ItemContext *ctx) override
Entry point called from QgsLegendRenderer to do the rendering.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
Object that keeps configuration of appearance of marker symbol's data-defined size in legend.
void setFont(const QFont &font)
Sets font used for rendering of labels - only valid for collapsed legend.
void setTextColor(const QColor &color)
Sets text color for rendering of labels - only valid for collapsed legend.
QImage collapsedLegendImage(QgsRenderContext &context, const QColor &backgroundColor=Qt::transparent, double paddingMM=1) const
Returns output image that would be shown in the legend. Returns invalid image if legend is not config...
void drawCollapsedLegend(QgsRenderContext &context, QSizeF *outputSize SIP_OUT=nullptr, double *labelXOffset SIP_OUT=nullptr) const
Draw the legend if using LegendOneNodeForAll and optionally output size of the legend and x offset of...
virtual QString dataSourceUri(bool expandAuthConfig=false) const
Gets the data source specification.
Class for storing the component parts of a RDBMS data source URI (e.g.
bool useEstimatedMetadata() const
Returns true if estimated metadata should be used for the connection.
Single scope for storing variables and functions for use within a QgsExpressionContext.
void addVariable(const QgsExpressionContextScope::StaticVariable &variable)
Adds a variable into the context scope.
static QList< QgsExpressionContextScope * > globalProjectLayerScopes(const QgsMapLayer *layer)
Creates a list of three scopes: global, layer's project and layer.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
void appendScope(QgsExpressionContextScope *scope)
Appends a scope to the end of the context.
void appendScopes(const QList< QgsExpressionContextScope * > &scopes)
Appends a list of scopes to the end of the context.
static QString replaceExpressionText(const QString &action, const QgsExpressionContext *context, const QgsDistanceArea *distanceArea=nullptr)
This function replaces each expression between [% and %] in the string with the result of its evaluat...
virtual void setLegendSymbolItem(const QString &key, QgsSymbol *symbol)
Sets the symbol to be used for a legend symbol item.
virtual void checkLegendSymbolItem(const QString &key, bool state=true)
Sets whether the legend symbology item with the specified ley should be checked.
static QString representFileSize(qint64 bytes)
Returns the human size from bytes.
void progress(qint64 received, qint64 total)
Emitted to report progress.
void error(const QString &msg)
Emitted when an error occurs.
void finish(const QImage &legend)
Emitted when the download completes.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
QgsImageLegendNode(QgsLayerTreeLayer *nodeLayer, const QImage &img, QObject *parent=nullptr)
Constructor for QgsImageLegendNode.
QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const override
Draws symbol on the left side of the item.
QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const override
Adds a symbol in base64 string within a JSON object with the key "icon".
static QRect nonTransparentImageRect(const QImage &image, QSize minSize=QSize(), bool center=false)
Calculates the non-transparent region of an image.
Layer tree node points to a map layer.
QString labelExpression() const
Returns the expression member of the LayerTreeNode.
QgsLegendPatchShape patchShape() const
Returns the symbol patch shape to use when rendering the legend node symbol.
QString name() const override
Returns the layer's name.
QgsMapLayer * layer() const
Returns the map layer associated with this node.
QSizeF patchSize() const
Returns the user (overridden) size for the legend node.
The QgsLegendRendererItem class is abstract interface for legend items returned from QgsMapLayerLegen...
virtual QVariant data(int role) const =0
Returns data associated with the item. Must be implemented in derived class.
QJsonObject exportToJson(const QgsLegendSettings &settings, const QgsRenderContext &context)
Entry point called from QgsLegendRenderer to do the rendering in a JSON object.
@ SimpleLegend
Simple label with icon legend node type.
@ RasterSymbolLegend
Raster symbol legend node type.
@ ImageLegend
Raster image legend node type.
@ DataDefinedSizeLegend
Marker symbol legend node type.
@ SymbolLegend
Vector symbol legend node type.
void checkAllItems()
Checks all checkable items belonging to the same layer as this node.
@ ParentRuleKey
Rule key of the parent legend node - for legends with tree hierarchy (QString). Added in 2....
@ NodeType
Type of node. Added in 3.16.
@ RuleKey
Rule key of the node (QString)
void uncheckAllItems()
Unchecks all checkable items belonging to the same layer as this node.
QgsLayerTreeModelLegendNode(QgsLayerTreeLayer *nodeL, QObject *parent=nullptr)
Construct the node with pointer to its parent layer node.
virtual void setUserPatchSize(QSizeF size)
Sets the user (overridden) size for the legend node.
void sizeChanged()
Emitted when the size of this node changes.
void dataChanged()
Emitted on internal data change so the layer tree model can forward the signal to views.
QgsRenderContext * createTemporaryRenderContext() const
Returns a temporary context or nullptr if legendMapViewData are not valid.
virtual QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const
Adds a symbol in base64 string within a JSON object with the key "icon".
QgsLayerTreeModel * model() const
Returns pointer to model owning this legend node.
void toggleAllItems()
Toggle all checkable items belonging to the same layer as this node.
virtual QSizeF userPatchSize() const
Returns the user (overridden) size for the legend node.
virtual ItemMetrics draw(const QgsLegendSettings &settings, ItemContext *ctx)
Entry point called from QgsLegendRenderer to do the rendering.
virtual Qt::ItemFlags flags() const
Returns item flags associated with the item. Default implementation returns Qt::ItemIsEnabled.
QgsLayerTreeLayer * layerNode() const
Returns pointer to the parent layer node.
virtual QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const
Draws symbol on the left side of the item.
virtual void setEmbeddedInParent(bool embedded)
virtual bool setData(const QVariant &value, int role)
Sets some data associated with the item. Default implementation does nothing and returns false.
virtual QSizeF drawSymbolText(const QgsLegendSettings &settings, ItemContext *ctx, QSizeF symbolSize) const
Draws label on the right side of the item.
The QgsLayerTreeModel class is model implementation for Qt item views framework.
QSet< QgsScreenProperties > targetScreenProperties() const
Returns the target screen properties to use when generating icons.
static int scaleIconSize(int standardSize)
Scales an layer tree model icon size to compensate for display pixel density, making the icon size hi...
const QgsMapSettings * legendFilterMapSettings() const
Returns the current map settings used for the current legend filter (or nullptr if none is enabled)
QVariant customProperty(const QString &key, const QVariant &defaultValue=QVariant()) const
Read a custom property from layer. Properties are stored in a map and saved in project file.
Represents a patch shape for use in map legends.
The QgsLegendSettings class stores the appearance and layout settings for legend drawing with QgsLege...
QgsLegendStyle style(QgsLegendStyle::Style s) const
Returns the style for a legend component.
bool drawRasterStroke() const
Returns whether a stroke will be drawn around raster symbol items.
QSizeF wmsLegendSize() const
Returns the size (in millimeters) of WMS legend graphics shown in the legend.
double minimumSymbolSize() const
Returns the minimum symbol size (in mm).
double rasterStrokeWidth() const
Returns the stroke width (in millimeters) for the stroke drawn around raster symbol items.
QSizeF symbolSize() const
Returns the default symbol size (in millimeters) used for legend items.
double maximumSymbolSize() const
Returns the maximum symbol size (in mm).
QColor rasterStrokeColor() const
Returns the stroke color for the stroke drawn around raster symbol items.
Q_DECL_DEPRECATED bool useAdvancedEffects() const
Q_DECL_DEPRECATED int dpi() const
bool synchronousLegendRequests() const
Returns whether to request legend graphics synchronously.
Q_DECL_DEPRECATED double mmPerMapUnit() const
QStringList evaluateItemText(const QString &text, const QgsExpressionContext &context) const
Splits a string using the wrap char taking into account handling empty wrap char which means no wrapp...
Qt::AlignmentFlag symbolAlignment() const
Returns the alignment for placement of legend symbols.
Q_DECL_DEPRECATED double mapScale() const
Returns the legend map scale.
Qt::Alignment alignment() const
Returns the alignment for the legend component.
QgsTextFormat & textFormat()
Returns the text format used for rendering this legend component.
@ Right
Right side.
@ Left
Left side.
double margin(Side side) const
Returns the margin (in mm) for the specified side of the component.
@ Symbol
Symbol icon (excluding label)
@ SymbolLabel
Symbol label (excluding icon)
The class stores information about one class/rule of a vector layer renderer in a unified way that ca...
QString parentRuleKey() const
Key of the parent legend node.
int scaleMaxDenom() const
Max scale denominator of the scale range.
void setSymbol(QgsSymbol *s)
Sets the symbol of the item.
int scaleMinDenom() const
Min scale denominator of the scale range.
QgsSymbol * symbol() const
Returns associated symbol. May be nullptr.
QString ruleKey() const
Returns unique identifier of the rule for identification of the item within renderer.
bool isCheckable() const
Returns whether the item is user-checkable - whether renderer supports enabling/disabling it.
QString label() const
Returns text label.
Base class for all map layer types.
Definition qgsmaplayer.h:76
void triggerRepaint(bool deferredUpdate=false)
Will advise the map canvas (and any other interested party) that this layer requires to be repainted.
void emitStyleChanged()
Triggers an emission of the styleChanged() signal.
QgsProject * project() const
Returns the parent project if this map layer is added to a project.
The QgsMapSettings class contains configuration for rendering of the map.
Perform transforms between map coordinates and device coordinates.
A marker symbol type, for rendering Point and MultiPoint geometries.
Contains settings for how a map layer will be labeled.
QString legendString() const
legendString
static QPixmap labelSettingsPreviewPixmap(const QgsPalLayerSettings &settings, QSize size, const QString &previewText=QString(), int padding=0, const QgsScreenProperties &screen=QgsScreenProperties())
Returns a pixmap preview for label settings.
const QgsTextFormat & format() const
Returns the label text formatting settings, e.g., font settings, buffer settings, etc.
Represents a map layer supporting display of point clouds.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
Definition qgsproject.h:107
Base class for raster data providers.
virtual QgsImageFetcher * getLegendGraphicFetcher(const QgsMapSettings *mapSettings)
Returns a new image downloader for the raster legend.
Represents a raster layer.
QgsRasterDataProvider * dataProvider() override
Returns the source data provider.
Raster renderer pipe that applies colors to a raster.
Qt::ItemFlags flags() const override
Returns item flags associated with the item. Default implementation returns Qt::ItemIsEnabled.
bool setData(const QVariant &value, int role) override
Sets some data associated with the item. Default implementation does nothing and returns false.
QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const override
Adds a symbol in base64 string within a JSON object with the key "icon".
QgsRasterSymbolLegendNode(QgsLayerTreeLayer *nodeLayer, const QColor &color, const QString &label, QObject *parent=nullptr, bool isCheckable=false, const QString &ruleKey=QString(), const QString &parentRuleKey=QString())
Constructor for QgsRasterSymbolLegendNode.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const override
Draws symbol on the left side of the item.
Contains information about the context of a rendering operation.
void setForceVectorOutput(bool force)
Sets whether rendering operations should use vector operations instead of any faster raster shortcuts...
double scaleFactor() const
Returns the scaling factor for the render to convert painter units to physical sizes.
void setScaleFactor(double factor)
Sets the scaling factor for the render to convert painter units to physical sizes.
QPainter * painter()
Returns the destination QPainter for the render operation.
void setPainterFlagsUsingContext(QPainter *painter=nullptr) const
Sets relevant flags on a destination painter, using the flags and settings currently defined for the ...
QgsExpressionContext & expressionContext()
Gets the expression context.
void setFlag(Qgis::RenderContextFlag flag, bool on=true)
Enable or disable a particular flag (other flags are not affected)
void setMapToPixel(const QgsMapToPixel &mtp)
Sets the context's map to pixel transform, which transforms between map coordinates and device coordi...
void setPainter(QPainter *p)
Sets the destination QPainter for the render operation.
static QgsRenderContext fromQPainter(QPainter *painter)
Creates a default render context given a pixel based QPainter destination.
void setExpressionContext(const QgsExpressionContext &context)
Sets the expression context.
void setRendererScale(double scale)
Sets the renderer map scale.
Qgis::RenderContextFlags flags() const
Returns combination of flags used for rendering.
Scoped object for saving and restoring a QPainter object's state.
Scoped object for temporary scaling of a QgsRenderContext for pixel based rendering.
Stores properties relating to a screen.
double devicePixelRatio() const
Returns the ratio between physical pixels and device-independent pixels for the screen.
This class is a composition of two QSettings instances:
Definition qgssettings.h:64
QVariant value(const QString &key, const QVariant &defaultValue=QVariant(), Section section=NoSection) const
Returns the value for setting key.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
QgsSimpleLegendNode(QgsLayerTreeLayer *nodeLayer, const QString &label, const QIcon &icon=QIcon(), QObject *parent=nullptr, const QString &key=QString())
Constructor for QgsSimpleLegendNode.
static QPixmap symbolPreviewPixmap(const QgsSymbol *symbol, QSize size, int padding=0, QgsRenderContext *customContext=nullptr, bool selected=false, const QgsExpressionContext *expressionContext=nullptr, const QgsLegendPatchShape *shape=nullptr, const QgsScreenProperties &screen=QgsScreenProperties())
Returns a pixmap preview for a color ramp.
static QgsSymbol * restrictedSizeSymbol(const QgsSymbol *s, double minSize, double maxSize, QgsRenderContext *context, double &width, double &height, bool *ok=nullptr)
Creates a new symbol with size restricted to min/max size if original size is out of min/max range.
static double estimateMaxSymbolBleed(QgsSymbol *symbol, const QgsRenderContext &context)
Returns the maximum estimated bleed for the symbol.
QString evaluateLabel(const QgsExpressionContext &context=QgsExpressionContext(), const QString &label=QString())
Evaluates and returns the text label of the current node.
const QgsSymbol * symbol() const
Returns the symbol used by the legend node.
void setPatchShape(const QgsLegendPatchShape &shape)
Sets the symbol patch shape to use when rendering the legend node symbol.
QgsSymbolLegendNode(QgsLayerTreeLayer *nodeLayer, const QgsLegendSymbolItem &item, QObject *parent=nullptr)
Constructor for QgsSymbolLegendNode.
void setIconSize(QSize sz)
Set the icon size.
QgsExpressionContextScope * createSymbolScope() const
Create an expression context scope containing symbol related variables.
QgsLegendPatchShape patchShape() const
Returns the symbol patch shape to use when rendering the legend node symbol.
QSize minimumIconSize() const
Calculates the minimum icon size to prevent cropping.
QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const override
Draws symbol on the left side of the item.
void invalidateMapBasedData() override
Notification from model that information from associated map view has changed.
bool setData(const QVariant &value, int role) override
Sets some data associated with the item. Default implementation does nothing and returns false.
Qt::ItemFlags flags() const override
Returns item flags associated with the item. Default implementation returns Qt::ItemIsEnabled.
void setCustomSymbol(QgsSymbol *symbol)
Sets the node's custom symbol.
void setEmbeddedInParent(bool embedded) override
QgsSymbol * customSymbol() const
Returns the node's custom symbol.
QString symbolLabel() const
Label of the symbol, user defined label will be used, otherwise will default to the label made by QGI...
void setSymbol(QgsSymbol *symbol)
Sets the symbol to be used by the legend node.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const override
Adds a symbol in base64 string within a JSON object with the key "icon".
Abstract base class for all rendered symbols.
Definition qgssymbol.h:231
bool usesMapUnits() const
Returns true if the symbol has any components which use map unit based sizes.
void drawPreviewIcon(QPainter *painter, QSize size, QgsRenderContext *customContext=nullptr, bool selected=false, const QgsExpressionContext *expressionContext=nullptr, const QgsLegendPatchShape *patchShape=nullptr, const QgsScreenProperties &screen=QgsScreenProperties())
Draws an icon of the symbol that occupies an area given by size using the specified painter.
Qgis::SymbolType type() const
Returns the symbol's type.
Definition qgssymbol.h:294
Contains pre-calculated metrics of a QgsTextDocument.
QSizeF documentSize(Qgis::TextLayoutMode mode, Qgis::TextOrientation orientation) const
Returns the overall size of the document.
double firstLineCapHeight() const
Returns the cap height for the first line of text.
static QgsTextDocumentMetrics calculateMetrics(const QgsTextDocument &document, const QgsTextFormat &format, const QgsRenderContext &context, double scaleFactor=1.0, const QgsTextDocumentRenderContext &documentContext=QgsTextDocumentRenderContext())
Returns precalculated text metrics for a text document, when rendered using the given base format and...
Represents a document consisting of one or more QgsTextBlock objects.
static QgsTextDocument fromTextAndFormat(const QStringList &lines, const QgsTextFormat &format)
Constructor for QgsTextDocument consisting of a set of lines, respecting settings from a text format.
Container for all settings relating to text rendering.
QFont scaledFont(const QgsRenderContext &context, double scaleFactor=1.0, bool *isZeroSize=nullptr) const
Returns a font with the size scaled to match the format's size settings (including units and map unit...
QFont toQFont() const
Returns a QFont matching the relevant settings from this text format.
QColor color() const
Returns the color that text will be rendered in.
static double textWidth(const QgsRenderContext &context, const QgsTextFormat &format, const QStringList &textLines, QFontMetricsF *fontMetrics=nullptr)
Returns the width of a text based on a given format.
static void drawDocument(const QRectF &rect, const QgsTextFormat &format, const QgsTextDocument &document, const QgsTextDocumentMetrics &metrics, QgsRenderContext &context, Qgis::TextHorizontalAlignment horizontalAlignment=Qgis::TextHorizontalAlignment::Left, Qgis::TextVerticalAlignment verticalAlignment=Qgis::TextVerticalAlignment::Top, double rotation=0, Qgis::TextLayoutMode mode=Qgis::TextLayoutMode::Rectangle, Qgis::TextRendererFlags flags=Qgis::TextRendererFlags())
Draws a text document within a rectangle using the specified settings.
static void drawText(const QRectF &rect, double rotation, Qgis::TextHorizontalAlignment alignment, const QStringList &textLines, QgsRenderContext &context, const QgsTextFormat &format, bool drawAsOutlines=true, Qgis::TextVerticalAlignment vAlignment=Qgis::TextVerticalAlignment::Top, Qgis::TextRendererFlags flags=Qgis::TextRendererFlags(), Qgis::TextLayoutMode mode=Qgis::TextLayoutMode::Rectangle)
Draws text within a rectangle using the specified settings.
static QFontMetricsF fontMetrics(QgsRenderContext &context, const QgsTextFormat &format, double scaleFactor=1.0)
Returns the font metrics for the given text format, when rendered in the specified render context.
static double calculateScaleFactorForFormat(const QgsRenderContext &context, const QgsTextFormat &format)
Returns the scale factor used for upscaling font sizes and downscaling destination painter devices.
static double textHeight(const QgsRenderContext &context, const QgsTextFormat &format, const QStringList &textLines, Qgis::TextLayoutMode mode=Qgis::TextLayoutMode::Point, QFontMetricsF *fontMetrics=nullptr, Qgis::TextRendererFlags flags=Qgis::TextRendererFlags(), double maxLineWidth=0)
Returns the height of a text based on a given format.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
QgsVectorLabelLegendNode(QgsLayerTreeLayer *nodeLayer, const QgsPalLayerSettings &labelSettings, QObject *parent=nullptr)
QgsVectorLabelLegendNode.
QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const override
drawSymbol
QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const override
exportSymbolToJson
QVariant data(int role) const override
data Returns data associated with the item
Represents a vector layer which manages a vector based data sets.
QgsExpressionContextScope * createExpressionContextScope() const FINAL
This method needs to be reimplemented in all classes which implement this interface and return an exp...
long long featureCount(const QString &legendKey) const
Number of features rendered with specified legend key.
void symbolFeatureCountMapChanged()
Emitted when the feature count for symbols on this layer has been recalculated.
QgsFeatureRenderer * renderer()
Returns the feature renderer used for rendering the features in the layer in 2D map views.
QgsVectorDataProvider * dataProvider() FINAL
Returns the layer's data provider, it may be nullptr.
QImage getLegendGraphicBlocking() const
Fetches the image from the server and returns it.
QSizeF drawSymbol(const QgsLegendSettings &settings, ItemContext *ctx, double itemHeight) const override
Draws symbol on the left side of the item.
void invalidateMapBasedData() override
Notification from model that information from associated map view has changed.
QVariant data(int role) const override
Returns data associated with the item. Must be implemented in derived class.
~QgsWmsLegendNode() override
QgsWmsLegendNode(QgsLayerTreeLayer *nodeLayer, QObject *parent=nullptr)
Constructor for QgsWmsLegendNode.
QJsonObject exportSymbolToJson(const QgsLegendSettings &settings, const QgsRenderContext &context) const override
Adds a symbol in base64 string within a JSON object with the key "icon".
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6643
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6642
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference)
Definition qgis.h:6066
QList< QgsLegendSymbolItem > QgsLegendSymbolList
#define QgsDebugError(str)
Definition qgslogger.h:38
Single variable definition for use within a QgsExpressionContextScope.
QgsLegendPatchShape patchShape
The patch shape to render for the node.
QgsScreenProperties screenProperties
Destination screen properties.
double maxSiblingSymbolWidth
Largest symbol width, considering all other sibling legend components associated with the current com...
QSizeF patchSize
Symbol patch size to render for the node.
const QgsTextDocument * textDocument
Optional text document.
double columnLeft
Left side of current legend column.
const QgsTextDocumentMetrics * textDocumentMetrics
Optional text document metrics.
double columnRight
Right side of current legend column.
Q_NOWARN_DEPRECATED_POP QgsRenderContext * context
Render context, if available.