QGIS API Documentation 3.43.0-Master (37eec98dbf6)
qgslayoutitemmap.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgslayoutitemmap.cpp
3 ---------------------
4 begin : July 2017
5 copyright : (C) 2017 by Nyall Dawson
6 email : nyall dot dawson at gmail dot com
7 ***************************************************************************/
8/***************************************************************************
9 * *
10 * This program is free software; you can redistribute it and/or modify *
11 * it under the terms of the GNU General Public License as published by *
12 * the Free Software Foundation; either version 2 of the License, or *
13 * (at your option) any later version. *
14 * *
15 ***************************************************************************/
16
17#include "qgslayoutitemmap.h"
18#include "moc_qgslayoutitemmap.cpp"
19#include "qgslayout.h"
22#include "qgslayoututils.h"
23#include "qgslayoutmodel.h"
26#include "qgsannotation.h"
27#include "qgsmapsettingsutils.h"
28#include "qgslayertree.h"
29#include "qgsmaplayerref.h"
32#include "qgsvectorlayer.h"
34#include "qgsapplication.h"
37#include "qgsannotationlayer.h"
39#include "qgsprojoperation.h"
40#include "qgslabelingresults.h"
41#include "qgsvectortileutils.h"
42#include "qgsunittypes.h"
43
44#include <QApplication>
45#include <QPainter>
46#include <QScreen>
47#include <QStyleOptionGraphicsItem>
48#include <QTimer>
49
51 : QgsLayoutItem( layout )
52 , mAtlasClippingSettings( new QgsLayoutItemMapAtlasClippingSettings( this ) )
53 , mItemClippingSettings( new QgsLayoutItemMapItemClipPathSettings( this ) )
54{
55 mBackgroundUpdateTimer = new QTimer( this );
56 mBackgroundUpdateTimer->setSingleShot( true );
57 connect( mBackgroundUpdateTimer, &QTimer::timeout, this, &QgsLayoutItemMap::recreateCachedImageInBackground );
58
60
61 setCacheMode( QGraphicsItem::NoCache );
62
63 connect( this, &QgsLayoutItem::sizePositionChanged, this, [this]
64 {
65 shapeChanged();
66 } );
67
68 mGridStack = std::make_unique< QgsLayoutItemMapGridStack >( this );
69 mOverviewStack = std::make_unique< QgsLayoutItemMapOverviewStack >( this );
70
71 connect( mAtlasClippingSettings, &QgsLayoutItemMapAtlasClippingSettings::changed, this, [this]
72 {
73 refresh();
74 } );
75
76 connect( mItemClippingSettings, &QgsLayoutItemMapItemClipPathSettings::changed, this, [this]
77 {
78 refresh();
79 } );
80
82 {
85 if ( mCrs != crs )
86 {
87 setCrs( crs );
89 }
90 } );
91
92 if ( layout )
93 connectUpdateSlot();
94}
95
97{
98 if ( mPainterJob )
99 {
100 disconnect( mPainterJob.get(), &QgsMapRendererCustomPainterJob::finished, this, &QgsLayoutItemMap::painterJobFinished );
102 mPainterJob->cancel(); // blocks
103 mPainter->end();
104 }
105}
106
111
113{
114 return QgsApplication::getThemeIcon( QStringLiteral( "/mLayoutItemMap.svg" ) );
115}
116
121
123{
124 if ( !mLayout )
125 return;
126
127 QList<QgsLayoutItemMap *> mapsList;
128 mLayout->layoutItems( mapsList );
129
130 int maxId = -1;
131 bool used = false;
132 for ( QgsLayoutItemMap *map : std::as_const( mapsList ) )
133 {
134 if ( map == this )
135 continue;
136
137 if ( map->mMapId == mMapId )
138 used = true;
139
140 maxId = std::max( maxId, map->mMapId );
141 }
142 if ( used )
143 {
144 mMapId = maxId + 1;
145 mLayout->itemsModel()->updateItemDisplayName( this );
146 }
147 updateToolTip();
148}
149
151{
152 if ( !QgsLayoutItem::id().isEmpty() )
153 {
154 return QgsLayoutItem::id();
155 }
156
157 return tr( "Map %1" ).arg( mMapId );
158}
159
164
166{
168
169 mCachedLayerStyleOverridesPresetName.clear();
170
172
173 updateAtlasFeature();
174}
175
177{
178 if ( rect().isEmpty() || !mLayout )
179 return 0;
180
181 QgsScaleCalculator calculator;
182 calculator.setMapUnits( crs().mapUnits() );
183 calculator.setDpi( 25.4 ); //Using mm
184 if ( QgsProject *project = mLayout->project() )
185 {
186 calculator.setMethod( project->scaleMethod() );
187 }
188 double widthInMm = mLayout->convertFromLayoutUnits( rect().width(), Qgis::LayoutUnit::Millimeters ).length();
189 return calculator.calculate( extent(), widthInMm );
190}
191
192void QgsLayoutItemMap::setScale( double scaleDenominator, bool forceUpdate )
193{
194 double currentScaleDenominator = scale();
195
196 if ( qgsDoubleNear( scaleDenominator, currentScaleDenominator ) || qgsDoubleNear( scaleDenominator, 0.0 ) || qgsDoubleNear( currentScaleDenominator, 0 ) )
197 {
198 return;
199 }
200
201 double scaleRatio = scaleDenominator / currentScaleDenominator;
202 mExtent.scale( scaleRatio );
203
204 if ( mAtlasDriven && mAtlasScalingMode == Fixed )
205 {
206 //if map is atlas controlled and set to fixed scaling mode, then scale changes should be treated as permanent
207 //and also apply to the map's original extent (see #9602)
208 //we can't use the scaleRatio calculated earlier, as the scale can vary depending on extent for geographic coordinate systems
209 QgsScaleCalculator calculator;
210 calculator.setMapUnits( crs().mapUnits() );
211 calculator.setDpi( 25.4 ); //QGraphicsView units are mm
212 if ( mLayout && mLayout->project() )
213 {
214 calculator.setMethod( mLayout->project()->scaleMethod() );
215 }
216
217 const double newScale = calculator.calculate( mExtent, rect().width() );
218 if ( !qgsDoubleNear( newScale, 0 ) )
219 {
220 scaleRatio = scaleDenominator / newScale;
221 mExtent.scale( scaleRatio );
222 }
223 }
224
226 if ( forceUpdate )
227 {
228 emit changed();
229 update();
230 }
231 emit extentChanged();
232}
233
235{
236 if ( mExtent == extent )
237 {
238 return;
239 }
240 mExtent = extent;
241
242 //recalculate data defined scale and extents, since that may override extent
243 refreshMapExtents();
244
245 //adjust height, if possible
246 if ( mExtent.isFinite() && !mExtent.isEmpty() )
247 {
248 const QRectF currentRect = rect();
249 const double newHeight = mExtent.width() == 0 ? 0
250 : currentRect.width() * mExtent.height() / mExtent.width();
251 attemptSetSceneRect( QRectF( pos().x(), pos().y(), currentRect.width(), newHeight ) );
252 }
253 update();
254}
255
257{
258 QgsRectangle newExtent = extent;
259 QgsRectangle currentExtent = mExtent;
260 //Make sure the width/height ratio is the same as the current layout map extent.
261 //This is to keep the map item frame size fixed
262 double currentWidthHeightRatio = 1.0;
263 if ( !currentExtent.isEmpty() )
264 currentWidthHeightRatio = currentExtent.width() / currentExtent.height();
265 else
266 currentWidthHeightRatio = rect().width() / rect().height();
267
268 if ( currentWidthHeightRatio != 0 && ! std::isnan( currentWidthHeightRatio ) && !newExtent.isEmpty() )
269 {
270 double newWidthHeightRatio = newExtent.width() / newExtent.height();
271
272 if ( currentWidthHeightRatio < newWidthHeightRatio )
273 {
274 //enlarge height of new extent, ensuring the map center stays the same
275 double newHeight = newExtent.width() / currentWidthHeightRatio;
276 double deltaHeight = newHeight - newExtent.height();
277 newExtent.setYMinimum( newExtent.yMinimum() - deltaHeight / 2 );
278 newExtent.setYMaximum( newExtent.yMaximum() + deltaHeight / 2 );
279 }
280 else
281 {
282 //enlarge width of new extent, ensuring the map center stays the same
283 double newWidth = currentWidthHeightRatio * newExtent.height();
284 double deltaWidth = newWidth - newExtent.width();
285 newExtent.setXMinimum( newExtent.xMinimum() - deltaWidth / 2 );
286 newExtent.setXMaximum( newExtent.xMaximum() + deltaWidth / 2 );
287 }
288 }
289
290 if ( mExtent == newExtent )
291 {
292 return;
293 }
294 mExtent = newExtent;
295
296 //recalculate data defined scale and extents, since that may override extent
297 refreshMapExtents();
298
300 emit changed();
301 emit extentChanged();
302}
303
305{
306 return mExtent;
307}
308
309QPolygonF QgsLayoutItemMap::calculateVisibleExtentPolygon( bool includeClipping ) const
310{
311 QPolygonF poly;
312 mapPolygon( mExtent, poly );
313
314 if ( includeClipping && mItemClippingSettings->isActive() )
315 {
316 const QgsGeometry geom = mItemClippingSettings->clippedMapExtent();
317 if ( !geom.isEmpty() )
318 {
319 poly = poly.intersected( geom.asQPolygonF() );
320 }
321 }
322
323 return poly;
324}
325
327{
328 return calculateVisibleExtentPolygon( true );
329}
330
332{
333 if ( mCrs.isValid() )
334 return mCrs;
335 else if ( mLayout && mLayout->project() )
336 return mLayout->project()->crs();
338}
339
341{
342 if ( mCrs == crs )
343 return;
344
345 mCrs = crs;
346 emit crsChanged();
347}
348
349QList<QgsMapLayer *> QgsLayoutItemMap::layers() const
350{
351 return _qgis_listRefToRaw( mLayers );
352}
353
354void QgsLayoutItemMap::setLayers( const QList<QgsMapLayer *> &layers )
355{
356 mGroupLayers.clear();
357
358 QList<QgsMapLayer *> layersCopy { layers };
359
360 // Group layers require special handling because they are just containers for child layers
361 // which are removed/added when group visibility changes,
362 // see issue https://github.com/qgis/QGIS/issues/53379
363 for ( auto it = layersCopy.begin(); it != layersCopy.end(); ++it )
364 {
365 if ( const QgsGroupLayer *groupLayer = qobject_cast<QgsGroupLayer *>( *it ) )
366 {
367 auto existingIt = mGroupLayers.find( groupLayer->id() );
368 if ( existingIt != mGroupLayers.end( ) )
369 {
370 *it = ( *existingIt ).second.get();
371 }
372 else
373 {
374 std::unique_ptr<QgsGroupLayer> groupLayerClone { groupLayer->clone() };
375 mGroupLayers[ groupLayer->id() ] = std::move( groupLayerClone );
376 *it = mGroupLayers[ groupLayer->id() ].get();
377 }
378 }
379 }
380 mLayers = _qgis_listRawToRef( layersCopy );
381}
382
383void QgsLayoutItemMap::setLayerStyleOverrides( const QMap<QString, QString> &overrides )
384{
385 if ( overrides == mLayerStyleOverrides )
386 return;
387
388 mLayerStyleOverrides = overrides;
389 emit layerStyleOverridesChanged(); // associated legends may listen to this
390
391}
392
394{
395 mLayerStyleOverrides.clear();
396 for ( const QgsMapLayerRef &layerRef : std::as_const( mLayers ) )
397 {
398 if ( QgsMapLayer *layer = layerRef.get() )
399 {
400 QgsMapLayerStyle style;
401 style.readFromLayer( layer );
402 mLayerStyleOverrides.insert( layer->id(), style.xmlData() );
403 }
404 }
405}
406
408{
409 if ( mFollowVisibilityPreset == follow )
410 return;
411
412 mFollowVisibilityPreset = follow;
413
414 if ( !mFollowVisibilityPresetName.isEmpty() )
415 emit themeChanged( mFollowVisibilityPreset ? mFollowVisibilityPresetName : QString() );
416}
417
419{
420 if ( name == mFollowVisibilityPresetName )
421 return;
422
423 mFollowVisibilityPresetName = name;
424 if ( mFollowVisibilityPreset )
425 emit themeChanged( mFollowVisibilityPresetName );
426}
427
428void QgsLayoutItemMap::moveContent( double dx, double dy )
429{
430 mLastRenderedImageOffsetX -= dx;
431 mLastRenderedImageOffsetY -= dy;
432 if ( !mDrawing )
433 {
434 transformShift( dx, dy );
435 mExtent.setXMinimum( mExtent.xMinimum() + dx );
436 mExtent.setXMaximum( mExtent.xMaximum() + dx );
437 mExtent.setYMinimum( mExtent.yMinimum() + dy );
438 mExtent.setYMaximum( mExtent.yMaximum() + dy );
439
440 //in case data defined extents are set, these override the calculated values
441 refreshMapExtents();
442
444 emit changed();
445 emit extentChanged();
446 }
447}
448
449void QgsLayoutItemMap::zoomContent( double factor, QPointF point )
450{
451 if ( mDrawing )
452 {
453 return;
454 }
455
456 //find out map coordinates of position
457 double mapX = mExtent.xMinimum() + ( point.x() / rect().width() ) * ( mExtent.xMaximum() - mExtent.xMinimum() );
458 double mapY = mExtent.yMinimum() + ( 1 - ( point.y() / rect().height() ) ) * ( mExtent.yMaximum() - mExtent.yMinimum() );
459
460 //find out new center point
461 double centerX = ( mExtent.xMaximum() + mExtent.xMinimum() ) / 2;
462 double centerY = ( mExtent.yMaximum() + mExtent.yMinimum() ) / 2;
463
464 centerX = mapX + ( centerX - mapX ) * ( 1.0 / factor );
465 centerY = mapY + ( centerY - mapY ) * ( 1.0 / factor );
466
467 double newIntervalX, newIntervalY;
468
469 if ( factor > 0 )
470 {
471 newIntervalX = ( mExtent.xMaximum() - mExtent.xMinimum() ) / factor;
472 newIntervalY = ( mExtent.yMaximum() - mExtent.yMinimum() ) / factor;
473 }
474 else //no need to zoom
475 {
476 return;
477 }
478
479 mExtent.setXMaximum( centerX + newIntervalX / 2 );
480 mExtent.setXMinimum( centerX - newIntervalX / 2 );
481 mExtent.setYMaximum( centerY + newIntervalY / 2 );
482 mExtent.setYMinimum( centerY - newIntervalY / 2 );
483
484 if ( mAtlasDriven && mAtlasScalingMode == Fixed )
485 {
486 //if map is atlas controlled and set to fixed scaling mode, then scale changes should be treated as permanent
487 //and also apply to the map's original extent (see #9602)
488 //we can't use the scaleRatio calculated earlier, as the scale can vary depending on extent for geographic coordinate systems
489 QgsScaleCalculator calculator;
490 calculator.setMapUnits( crs().mapUnits() );
491 calculator.setDpi( 25.4 ); //QGraphicsView units are mm
492 if ( mLayout && mLayout->project() )
493 {
494 calculator.setMethod( mLayout->project()->scaleMethod() );
495 }
496 const double newScale = calculator.calculate( mExtent, rect().width() );
497 if ( !qgsDoubleNear( newScale, 0 ) )
498 {
499 const double scaleRatio = scale() / newScale ;
500 mExtent.scale( scaleRatio );
501 }
502 }
503
504 //recalculate data defined scale and extents, since that may override zoom
505 refreshMapExtents();
506
508 emit changed();
509 emit extentChanged();
510}
511
513{
514 const QList< QgsMapLayer * > layers = layersToRender();
515 for ( QgsMapLayer *layer : layers )
516 {
517 if ( layer->dataProvider() && layer->providerType() == QLatin1String( "wms" ) )
518 {
519 return true;
520 }
521 }
522 return false;
523}
524
526{
527 if ( blendMode() != QPainter::CompositionMode_SourceOver )
528 return true;
529
530 // we MUST force the whole layout to render as a raster if any map item
531 // uses blend modes, and we are not drawing on a solid opaque background
532 // because in this case the map item needs to be rendered as a raster, but
533 // it also needs to interact with items below it
534
535 // WARNING -- modifying this logic? Then ALSO update containsAdvancedEffects accordingly!
536
537 // BIG WARNING -- we CANNOT just check containsAdvancedEffects here, as that method MUST
538 // return true if the map item has transparency.
539 // BUT, we **DO NOT HAVE TO** force the WHOLE layout to be rasterized if a map item
540 // is semi-opaque, as we have logic in QgsLayoutItemMap::paint to automatically render the
541 // map to a temporary image surface. I.e, we can get away with just rasterising the map
542 // alone and leaving the rest of the content as vector.
543
544 // SO this logic is a COPY of containsAdvancedEffects, without the opacity check
545
546 auto containsAdvancedEffectsIgnoreItemOpacity = [this]()-> bool
547 {
549 return true;
550
551 //check easy things first
552
553 // WARNING -- modifying this logic? Then ALSO update containsAdvancedEffects accordingly!
554
555 //overviews
556 if ( mOverviewStack->containsAdvancedEffects() )
557 {
558 return true;
559 }
560
561 // WARNING -- modifying this logic? Then ALSO update containsAdvancedEffects accordingly!
562
563 //grids
564 if ( mGridStack->containsAdvancedEffects() )
565 {
566 return true;
567 }
568
569 // WARNING -- modifying this logic? Then ALSO update containsAdvancedEffects accordingly!
570
573 return ( !QgsMapSettingsUtils::containsAdvancedEffects( ms ).isEmpty() );
574 // WARNING -- modifying this logic? Then ALSO update requiresRasterization accordingly!
575 };
576
577 if ( !containsAdvancedEffectsIgnoreItemOpacity() )
578 return false;
579
580 // WARNING -- modifying this logic? Then ALSO update containsAdvancedEffects accordingly!
581
582 if ( hasBackground() && qgsDoubleNear( backgroundColor().alphaF(), 1.0 ) )
583 return false;
584
585 return true;
586}
587
589{
590 if ( QgsLayoutItem::containsAdvancedEffects() || mEvaluatedOpacity < 1.0 )
591 return true;
592
593 //check easy things first
594
595 // WARNING -- modifying this logic? Then ALSO update requiresRasterization accordingly!
596
597 //overviews
598 if ( mOverviewStack->containsAdvancedEffects() )
599 {
600 return true;
601 }
602
603 // WARNING -- modifying this logic? Then ALSO update requiresRasterization accordingly!
604
605 //grids
606 if ( mGridStack->containsAdvancedEffects() )
607 {
608 return true;
609 }
610
611 // WARNING -- modifying this logic? Then ALSO update requiresRasterization accordingly!
612
615 return ( !QgsMapSettingsUtils::containsAdvancedEffects( ms ).isEmpty() );
616 // WARNING -- modifying this logic? Then ALSO update requiresRasterization accordingly!
617}
618
619void QgsLayoutItemMap::setMapRotation( double rotation )
620{
621 mMapRotation = rotation;
622 mEvaluatedMapRotation = mMapRotation;
624 emit mapRotationChanged( rotation );
625 emit changed();
626}
627
629{
630 return valueType == QgsLayoutObject::EvaluatedValue ? mEvaluatedMapRotation : mMapRotation;
631}
632
634{
635 mAtlasDriven = enabled;
636
637 if ( !enabled )
638 {
639 //if not enabling the atlas, we still need to refresh the map extents
640 //so that data defined extents and scale are recalculated
641 refreshMapExtents();
642 }
643}
644
646{
647 if ( valueType == QgsLayoutObject::EvaluatedValue )
648 {
649 //evaluate data defined atlas margin
650
651 //start with user specified margin
652 double margin = mAtlasMargin;
654
655 bool ok = false;
657 if ( ok )
658 {
659 //divide by 100 to convert to 0 -> 1.0 range
660 margin = ddMargin / 100;
661 }
662 return margin;
663 }
664 else
665 {
666 return mAtlasMargin;
667 }
668}
669
671{
672 if ( mGridStack->size() < 1 )
673 {
674 QgsLayoutItemMapGrid *grid = new QgsLayoutItemMapGrid( tr( "Grid %1" ).arg( 1 ), this );
675 mGridStack->addGrid( grid );
676 }
677 return mGridStack->grid( 0 );
678}
679
681{
682 if ( mOverviewStack->size() < 1 )
683 {
684 QgsLayoutItemMapOverview *overview = new QgsLayoutItemMapOverview( tr( "Overview %1" ).arg( 1 ), this );
685 mOverviewStack->addOverview( overview );
686 }
687 return mOverviewStack->overview( 0 );
688}
689
691{
692 double frameBleed = QgsLayoutItem::estimatedFrameBleed();
693
694 // Check if any of the grids are enabled
695 if ( mGridStack )
696 {
697 for ( int i = 0; i < mGridStack->size(); ++i )
698 {
699 const QgsLayoutItemMapGrid *grid = qobject_cast<QgsLayoutItemMapGrid *>( mGridStack->item( i ) );
700 if ( grid->mEvaluatedEnabled )
701 {
702 // Grid bleed is the grid frame width + grid frame offset + half the pen width
703 frameBleed = std::max( frameBleed, grid->mEvaluatedGridFrameWidth + grid->mEvaluatedGridFrameMargin + grid->mEvaluatedGridFrameLineThickness / 2.0 );
704 }
705 }
706 }
707
708 return frameBleed;
709}
710
714
715bool QgsLayoutItemMap::writePropertiesToElement( QDomElement &mapElem, QDomDocument &doc, const QgsReadWriteContext &context ) const
716{
717 if ( mKeepLayerSet )
718 {
719 mapElem.setAttribute( QStringLiteral( "keepLayerSet" ), QStringLiteral( "true" ) );
720 }
721 else
722 {
723 mapElem.setAttribute( QStringLiteral( "keepLayerSet" ), QStringLiteral( "false" ) );
724 }
725
726 if ( mDrawAnnotations )
727 {
728 mapElem.setAttribute( QStringLiteral( "drawCanvasItems" ), QStringLiteral( "true" ) );
729 }
730 else
731 {
732 mapElem.setAttribute( QStringLiteral( "drawCanvasItems" ), QStringLiteral( "false" ) );
733 }
734
735 //extent
736 QDomElement extentElem = doc.createElement( QStringLiteral( "Extent" ) );
737 extentElem.setAttribute( QStringLiteral( "xmin" ), qgsDoubleToString( mExtent.xMinimum() ) );
738 extentElem.setAttribute( QStringLiteral( "xmax" ), qgsDoubleToString( mExtent.xMaximum() ) );
739 extentElem.setAttribute( QStringLiteral( "ymin" ), qgsDoubleToString( mExtent.yMinimum() ) );
740 extentElem.setAttribute( QStringLiteral( "ymax" ), qgsDoubleToString( mExtent.yMaximum() ) );
741 mapElem.appendChild( extentElem );
742
743 if ( mCrs.isValid() )
744 {
745 QDomElement crsElem = doc.createElement( QStringLiteral( "crs" ) );
746 mCrs.writeXml( crsElem, doc );
747 mapElem.appendChild( crsElem );
748 }
749
750 // follow map theme
751 mapElem.setAttribute( QStringLiteral( "followPreset" ), mFollowVisibilityPreset ? QStringLiteral( "true" ) : QStringLiteral( "false" ) );
752 mapElem.setAttribute( QStringLiteral( "followPresetName" ), mFollowVisibilityPresetName );
753
754 //map rotation
755 mapElem.setAttribute( QStringLiteral( "mapRotation" ), QString::number( mMapRotation ) );
756
757 //layer set
758 QDomElement layerSetElem = doc.createElement( QStringLiteral( "LayerSet" ) );
759 for ( const QgsMapLayerRef &layerRef : mLayers )
760 {
761 if ( !layerRef )
762 continue;
763 QDomElement layerElem = doc.createElement( QStringLiteral( "Layer" ) );
764 QString layerId;
765 const auto it = std::find_if( mGroupLayers.cbegin(), mGroupLayers.cend(), [ &layerRef ]( const std::pair<const QString, std::unique_ptr<QgsGroupLayer>> &groupLayer ) -> bool
766 {
767 return groupLayer.second.get() == layerRef.get();
768 } );
769
770 if ( it != mGroupLayers.end() )
771 {
772 layerId = it->first;
773 }
774 else
775 {
776 layerId = layerRef.layerId;
777 }
778
779 QDomText layerIdText = doc.createTextNode( layerId );
780 layerElem.appendChild( layerIdText );
781
782 layerElem.setAttribute( QStringLiteral( "name" ), layerRef.name );
783 layerElem.setAttribute( QStringLiteral( "source" ), layerRef.source );
784 layerElem.setAttribute( QStringLiteral( "provider" ), layerRef.provider );
785
786 if ( it != mGroupLayers.end() )
787 {
788 const auto childLayers { it->second->childLayers() };
789 QDomElement childLayersElement = doc.createElement( QStringLiteral( "childLayers" ) );
790 for ( const QgsMapLayer *childLayer : std::as_const( childLayers ) )
791 {
792 QDomElement childElement = doc.createElement( QStringLiteral( "child" ) );
793 childElement.setAttribute( QStringLiteral( "layerid" ), childLayer->id() );
794 childLayersElement.appendChild( childElement );
795 }
796 layerElem.appendChild( childLayersElement );
797 }
798 layerSetElem.appendChild( layerElem );
799 }
800 mapElem.appendChild( layerSetElem );
801
802 // override styles
803 if ( mKeepLayerStyles )
804 {
805 QDomElement stylesElem = doc.createElement( QStringLiteral( "LayerStyles" ) );
806 for ( auto styleIt = mLayerStyleOverrides.constBegin(); styleIt != mLayerStyleOverrides.constEnd(); ++styleIt )
807 {
808 QDomElement styleElem = doc.createElement( QStringLiteral( "LayerStyle" ) );
809
810 QgsMapLayerRef ref( styleIt.key() );
811 ref.resolve( mLayout->project() );
812
813 styleElem.setAttribute( QStringLiteral( "layerid" ), ref.layerId );
814 styleElem.setAttribute( QStringLiteral( "name" ), ref.name );
815 styleElem.setAttribute( QStringLiteral( "source" ), ref.source );
816 styleElem.setAttribute( QStringLiteral( "provider" ), ref.provider );
817
818 QgsMapLayerStyle style( styleIt.value() );
819 style.writeXml( styleElem );
820 stylesElem.appendChild( styleElem );
821 }
822 mapElem.appendChild( stylesElem );
823 }
824
825 //grids
826 mGridStack->writeXml( mapElem, doc, context );
827
828 //overviews
829 mOverviewStack->writeXml( mapElem, doc, context );
830
831 //atlas
832 QDomElement atlasElem = doc.createElement( QStringLiteral( "AtlasMap" ) );
833 atlasElem.setAttribute( QStringLiteral( "atlasDriven" ), mAtlasDriven );
834 atlasElem.setAttribute( QStringLiteral( "scalingMode" ), mAtlasScalingMode );
835 atlasElem.setAttribute( QStringLiteral( "margin" ), qgsDoubleToString( mAtlasMargin ) );
836 mapElem.appendChild( atlasElem );
837
838 mapElem.setAttribute( QStringLiteral( "labelMargin" ), mLabelMargin.encodeMeasurement() );
839 mapElem.setAttribute( QStringLiteral( "mapFlags" ), static_cast< int>( mMapFlags ) );
840
841 QDomElement labelBlockingItemsElem = doc.createElement( QStringLiteral( "labelBlockingItems" ) );
842 for ( const auto &item : std::as_const( mBlockingLabelItems ) )
843 {
844 if ( !item )
845 continue;
846
847 QDomElement blockingItemElem = doc.createElement( QStringLiteral( "item" ) );
848 blockingItemElem.setAttribute( QStringLiteral( "uuid" ), item->uuid() );
849 labelBlockingItemsElem.appendChild( blockingItemElem );
850 }
851 mapElem.appendChild( labelBlockingItemsElem );
852
853 //temporal settings
854 mapElem.setAttribute( QStringLiteral( "isTemporal" ), isTemporal() ? 1 : 0 );
855 if ( isTemporal() )
856 {
857 mapElem.setAttribute( QStringLiteral( "temporalRangeBegin" ), temporalRange().begin().toString( Qt::ISODate ) );
858 mapElem.setAttribute( QStringLiteral( "temporalRangeEnd" ), temporalRange().end().toString( Qt::ISODate ) );
859 }
860
861 mapElem.setAttribute( QStringLiteral( "enableZRange" ), mZRangeEnabled ? 1 : 0 );
862 if ( mZRange.lower() != std::numeric_limits< double >::lowest() )
863 mapElem.setAttribute( QStringLiteral( "zRangeLower" ), qgsDoubleToString( mZRange.lower() ) );
864 if ( mZRange.upper() != std::numeric_limits< double >::max() )
865 mapElem.setAttribute( QStringLiteral( "zRangeUpper" ), qgsDoubleToString( mZRange.upper() ) );
866
867 mAtlasClippingSettings->writeXml( mapElem, doc, context );
868 mItemClippingSettings->writeXml( mapElem, doc, context );
869
870 return true;
871}
872
873bool QgsLayoutItemMap::readPropertiesFromElement( const QDomElement &itemElem, const QDomDocument &doc, const QgsReadWriteContext &context )
874{
875 mUpdatesEnabled = false;
876
877 //extent
878 QDomNodeList extentNodeList = itemElem.elementsByTagName( QStringLiteral( "Extent" ) );
879 if ( !extentNodeList.isEmpty() )
880 {
881 QDomElement extentElem = extentNodeList.at( 0 ).toElement();
882 double xmin, xmax, ymin, ymax;
883 xmin = extentElem.attribute( QStringLiteral( "xmin" ) ).toDouble();
884 xmax = extentElem.attribute( QStringLiteral( "xmax" ) ).toDouble();
885 ymin = extentElem.attribute( QStringLiteral( "ymin" ) ).toDouble();
886 ymax = extentElem.attribute( QStringLiteral( "ymax" ) ).toDouble();
887 setExtent( QgsRectangle( xmin, ymin, xmax, ymax ) );
888 }
889
890 QDomNodeList crsNodeList = itemElem.elementsByTagName( QStringLiteral( "crs" ) );
892 if ( !crsNodeList.isEmpty() )
893 {
894 QDomElement crsElem = crsNodeList.at( 0 ).toElement();
895 crs.readXml( crsElem );
896 }
897 setCrs( crs );
898
899 //map rotation
900 mMapRotation = itemElem.attribute( QStringLiteral( "mapRotation" ), QStringLiteral( "0" ) ).toDouble();
901 mEvaluatedMapRotation = mMapRotation;
902
903 // follow map theme
904 mFollowVisibilityPreset = itemElem.attribute( QStringLiteral( "followPreset" ) ).compare( QLatin1String( "true" ) ) == 0;
905 mFollowVisibilityPresetName = itemElem.attribute( QStringLiteral( "followPresetName" ) );
906
907 //mKeepLayerSet flag
908 QString keepLayerSetFlag = itemElem.attribute( QStringLiteral( "keepLayerSet" ) );
909 if ( keepLayerSetFlag.compare( QLatin1String( "true" ), Qt::CaseInsensitive ) == 0 )
910 {
911 mKeepLayerSet = true;
912 }
913 else
914 {
915 mKeepLayerSet = false;
916 }
917
918 QString drawCanvasItemsFlag = itemElem.attribute( QStringLiteral( "drawCanvasItems" ), QStringLiteral( "true" ) );
919 if ( drawCanvasItemsFlag.compare( QLatin1String( "true" ), Qt::CaseInsensitive ) == 0 )
920 {
921 mDrawAnnotations = true;
922 }
923 else
924 {
925 mDrawAnnotations = false;
926 }
927
928 mLayerStyleOverrides.clear();
929
930 QList<QgsMapLayerRef> layerSet;
931 QDomNodeList layerSetNodeList = itemElem.elementsByTagName( QStringLiteral( "LayerSet" ) );
932 if ( !layerSetNodeList.isEmpty() )
933 {
934 QDomElement layerSetElem = layerSetNodeList.at( 0 ).toElement();
935 QDomNodeList layerIdNodeList = layerSetElem.elementsByTagName( QStringLiteral( "Layer" ) );
936 layerSet.reserve( layerIdNodeList.size() );
937 for ( int i = 0; i < layerIdNodeList.size(); ++i )
938 {
939 QDomElement layerElem = layerIdNodeList.at( i ).toElement();
940 QString layerId = layerElem.text();
941 QString layerName = layerElem.attribute( QStringLiteral( "name" ) );
942 QString layerSource = layerElem.attribute( QStringLiteral( "source" ) );
943 QString layerProvider = layerElem.attribute( QStringLiteral( "provider" ) );
944
945 QgsMapLayerRef ref( layerId, layerName, layerSource, layerProvider );
946 if ( ref.resolveWeakly( mLayout->project() ) )
947 {
948 layerSet << ref;
949 }
950 }
951 }
952
953 setLayers( _qgis_listRefToRaw( layerSet ) );
954
955 // Restore group layers configuration
956 if ( !layerSetNodeList.isEmpty() )
957 {
958 QDomElement layerSetElem = layerSetNodeList.at( 0 ).toElement();
959 QDomNodeList layerIdNodeList = layerSetElem.elementsByTagName( QStringLiteral( "Layer" ) );
960 for ( int i = 0; i < layerIdNodeList.size(); ++i )
961 {
962 QDomElement layerElem = layerIdNodeList.at( i ).toElement();
963 const QString layerId = layerElem.text();
964 const auto it = mGroupLayers.find( layerId );
965 if ( it != mGroupLayers.cend() )
966 {
967 QList<QgsMapLayerRef> childSet;
968 const QDomNodeList childLayersElements = layerElem.elementsByTagName( QStringLiteral( "childLayers" ) );
969 const QDomNodeList children = childLayersElements.at( 0 ).childNodes();
970 for ( int i = 0; i < children.size(); ++i )
971 {
972 const QDomElement childElement = children.at( i ).toElement();
973 const QString id = childElement.attribute( QStringLiteral( "layerid" ) );
974 QgsMapLayerRef layerRef{ id };
975 if ( layerRef.resolveWeakly( mLayout->project() ) )
976 {
977 childSet.push_back( layerRef );
978 }
979 }
980 it->second->setChildLayers( _qgis_listRefToRaw( childSet ) );
981 }
982 }
983 }
984
985
986 // override styles
987 QDomNodeList layerStylesNodeList = itemElem.elementsByTagName( QStringLiteral( "LayerStyles" ) );
988 mKeepLayerStyles = !layerStylesNodeList.isEmpty();
989 if ( mKeepLayerStyles )
990 {
991 QDomElement layerStylesElem = layerStylesNodeList.at( 0 ).toElement();
992 QDomNodeList layerStyleNodeList = layerStylesElem.elementsByTagName( QStringLiteral( "LayerStyle" ) );
993 for ( int i = 0; i < layerStyleNodeList.size(); ++i )
994 {
995 const QDomElement &layerStyleElement = layerStyleNodeList.at( i ).toElement();
996 QString layerId = layerStyleElement.attribute( QStringLiteral( "layerid" ) );
997 QString layerName = layerStyleElement.attribute( QStringLiteral( "name" ) );
998 QString layerSource = layerStyleElement.attribute( QStringLiteral( "source" ) );
999 QString layerProvider = layerStyleElement.attribute( QStringLiteral( "provider" ) );
1000 QgsMapLayerRef ref( layerId, layerName, layerSource, layerProvider );
1001 ref.resolveWeakly( mLayout->project() );
1002
1003 QgsMapLayerStyle style;
1004 style.readXml( layerStyleElement );
1005 mLayerStyleOverrides.insert( ref.layerId, style.xmlData() );
1006 }
1007 }
1008
1009 mDrawing = false;
1010 mNumCachedLayers = 0;
1011 mCacheInvalidated = true;
1012
1013 //overviews
1014 mOverviewStack->readXml( itemElem, doc, context );
1015
1016 //grids
1017 mGridStack->readXml( itemElem, doc, context );
1018
1019 //atlas
1020 QDomNodeList atlasNodeList = itemElem.elementsByTagName( QStringLiteral( "AtlasMap" ) );
1021 if ( !atlasNodeList.isEmpty() )
1022 {
1023 QDomElement atlasElem = atlasNodeList.at( 0 ).toElement();
1024 mAtlasDriven = ( atlasElem.attribute( QStringLiteral( "atlasDriven" ), QStringLiteral( "0" ) ) != QLatin1String( "0" ) );
1025 if ( atlasElem.hasAttribute( QStringLiteral( "fixedScale" ) ) ) // deprecated XML
1026 {
1027 mAtlasScalingMode = ( atlasElem.attribute( QStringLiteral( "fixedScale" ), QStringLiteral( "0" ) ) != QLatin1String( "0" ) ) ? Fixed : Auto;
1028 }
1029 else if ( atlasElem.hasAttribute( QStringLiteral( "scalingMode" ) ) )
1030 {
1031 mAtlasScalingMode = static_cast<AtlasScalingMode>( atlasElem.attribute( QStringLiteral( "scalingMode" ) ).toInt() );
1032 }
1033 mAtlasMargin = atlasElem.attribute( QStringLiteral( "margin" ), QStringLiteral( "0.1" ) ).toDouble();
1034 }
1035
1036 setLabelMargin( QgsLayoutMeasurement::decodeMeasurement( itemElem.attribute( QStringLiteral( "labelMargin" ), QStringLiteral( "0" ) ) ) );
1037
1038 mMapFlags = static_cast< MapItemFlags>( itemElem.attribute( QStringLiteral( "mapFlags" ), nullptr ).toInt() );
1039
1040 // label blocking items
1041 mBlockingLabelItems.clear();
1042 mBlockingLabelItemUuids.clear();
1043 QDomNodeList labelBlockingNodeList = itemElem.elementsByTagName( QStringLiteral( "labelBlockingItems" ) );
1044 if ( !labelBlockingNodeList.isEmpty() )
1045 {
1046 QDomElement blockingItems = labelBlockingNodeList.at( 0 ).toElement();
1047 QDomNodeList labelBlockingNodeList = blockingItems.childNodes();
1048 for ( int i = 0; i < labelBlockingNodeList.size(); ++i )
1049 {
1050 const QDomElement &itemBlockingElement = labelBlockingNodeList.at( i ).toElement();
1051 const QString itemUuid = itemBlockingElement.attribute( QStringLiteral( "uuid" ) );
1052 mBlockingLabelItemUuids << itemUuid;
1053 }
1054 }
1055
1056 mAtlasClippingSettings->readXml( itemElem, doc, context );
1057 mItemClippingSettings->readXml( itemElem, doc, context );
1058
1060
1061 //temporal settings
1062 setIsTemporal( itemElem.attribute( QStringLiteral( "isTemporal" ) ).toInt() );
1063 if ( isTemporal() )
1064 {
1065 const QDateTime begin = QDateTime::fromString( itemElem.attribute( QStringLiteral( "temporalRangeBegin" ) ), Qt::ISODate );
1066 const QDateTime end = QDateTime::fromString( itemElem.attribute( QStringLiteral( "temporalRangeEnd" ) ), Qt::ISODate );
1067 setTemporalRange( QgsDateTimeRange( begin, end, true, begin == end ) );
1068 }
1069
1070 mZRangeEnabled = itemElem.attribute( QStringLiteral( "enableZRange" ) ).toInt();
1071 bool ok = false;
1072 double zLower = itemElem.attribute( QStringLiteral( "zRangeLower" ) ).toDouble( &ok );
1073 if ( !ok )
1074 {
1075 zLower = std::numeric_limits< double >::lowest();
1076 }
1077 double zUpper = itemElem.attribute( QStringLiteral( "zRangeUpper" ) ).toDouble( &ok );
1078 if ( !ok )
1079 {
1080 zUpper = std::numeric_limits< double >::max();
1081 }
1082 mZRange = QgsDoubleRange( zLower, zUpper );
1083
1084 mUpdatesEnabled = true;
1085 return true;
1086}
1087
1089{
1090 if ( mItemClippingSettings->isActive() )
1091 {
1092 const QgsGeometry g = mItemClippingSettings->clipPathInMapItemCoordinates();
1093 if ( !g.isNull() )
1094 return g.constGet()->asQPainterPath();
1095 }
1096 return QgsLayoutItem::framePath();
1097}
1098
1099void QgsLayoutItemMap::paint( QPainter *painter, const QStyleOptionGraphicsItem *style, QWidget * )
1100{
1101 if ( !mLayout || !painter || !painter->device() || !mUpdatesEnabled )
1102 {
1103 return;
1104 }
1105 if ( !shouldDrawItem() )
1106 {
1107 return;
1108 }
1109
1110 QRectF thisPaintRect = rect();
1111 if ( qgsDoubleNear( thisPaintRect.width(), 0.0 ) || qgsDoubleNear( thisPaintRect.height(), 0 ) )
1112 return;
1113
1114 //TODO - try to reduce the amount of duplicate code here!
1115
1116 if ( mLayout->renderContext().isPreviewRender() )
1117 {
1118 bool renderInProgress = false;
1119 mPreviewDevicePixelRatio = painter->device()->devicePixelRatioF();
1120
1121 QgsScopedQPainterState painterState( painter );
1122 painter->setClipRect( thisPaintRect );
1123 if ( !mCacheFinalImage || mCacheFinalImage->isNull() )
1124 {
1125 // No initial render available - so draw some preview text alerting user
1126 painter->setBrush( QBrush( QColor( 125, 125, 125, 125 ) ) );
1127 painter->drawRect( thisPaintRect );
1128 painter->setBrush( Qt::NoBrush );
1129 QFont messageFont;
1130 messageFont.setPointSize( 12 );
1131 painter->setFont( messageFont );
1132 painter->setPen( QColor( 255, 255, 255, 255 ) );
1133 painter->drawText( thisPaintRect, Qt::AlignCenter | Qt::AlignHCenter, tr( "Rendering map" ) );
1134 if ( mPainterJob && mCacheInvalidated && !mDrawingPreview )
1135 {
1136 // current job was invalidated - start a new one
1137 mPreviewScaleFactor = QgsLayoutUtils::scaleFactorFromItemStyle( style, painter );
1138 mBackgroundUpdateTimer->start( 1 );
1139 }
1140 else if ( !mPainterJob && !mDrawingPreview )
1141 {
1142 // this is the map's very first paint - trigger a cache update
1143 mPreviewScaleFactor = QgsLayoutUtils::scaleFactorFromItemStyle( style, painter );
1144 mBackgroundUpdateTimer->start( 1 );
1145 }
1146 renderInProgress = true;
1147 }
1148 else
1149 {
1150 if ( mCacheInvalidated && !mDrawingPreview )
1151 {
1152 // cache was invalidated - trigger a background update
1153 mPreviewScaleFactor = QgsLayoutUtils::scaleFactorFromItemStyle( style, painter );
1154 mBackgroundUpdateTimer->start( 1 );
1155 renderInProgress = true;
1156 }
1157
1158 //Background color is already included in cached image, so no need to draw
1159
1160 double imagePixelWidth = mCacheFinalImage->width(); //how many pixels of the image are for the map extent?
1161 double scale = rect().width() / imagePixelWidth * mCacheFinalImage->devicePixelRatio();
1162
1163 QgsScopedQPainterState rotatedPainterState( painter );
1164
1165 painter->translate( mLastRenderedImageOffsetX + mXOffset, mLastRenderedImageOffsetY + mYOffset );
1166 painter->setCompositionMode( blendModeForRender() );
1167 painter->scale( scale, scale );
1168 painter->drawImage( 0, 0, *mCacheFinalImage );
1169
1170 //restore rotation
1171 }
1172
1173 painter->setClipRect( thisPaintRect, Qt::NoClip );
1174
1175 mOverviewStack->drawItems( painter, false );
1176 mGridStack->drawItems( painter );
1177 drawAnnotations( painter );
1178 drawMapFrame( painter );
1179
1180 if ( renderInProgress )
1181 {
1182 drawRefreshingOverlay( painter, style );
1183 }
1184 }
1185 else
1186 {
1187 if ( mDrawing )
1188 return;
1189
1190 mDrawing = true;
1191 QPaintDevice *paintDevice = painter->device();
1192 if ( !paintDevice )
1193 return;
1194
1195 QgsRectangle cExtent = extent();
1196 QSizeF size( cExtent.width() * mapUnitsToLayoutUnits(), cExtent.height() * mapUnitsToLayoutUnits() );
1197
1198 if ( mLayout && mLayout->renderContext().flags() & QgsLayoutRenderContext::FlagLosslessImageRendering )
1199 painter->setRenderHint( QPainter::LosslessImageRendering, true );
1200
1201 if ( ( containsAdvancedEffects() || ( blendModeForRender() != QPainter::CompositionMode_SourceOver ) )
1202 && ( !mLayout || !( mLayout->renderContext().flags() & QgsLayoutRenderContext::FlagForceVectorOutput ) ) )
1203 {
1204 // rasterize
1205 double destinationDpi = QgsLayoutUtils::scaleFactorFromItemStyle( style, painter ) * 25.4;
1206 double layoutUnitsInInches = mLayout ? mLayout->convertFromLayoutUnits( 1, Qgis::LayoutUnit::Inches ).length() : 1;
1207 int widthInPixels = static_cast< int >( std::round( boundingRect().width() * layoutUnitsInInches * destinationDpi ) );
1208 int heightInPixels = static_cast< int >( std::round( boundingRect().height() * layoutUnitsInInches * destinationDpi ) );
1209 QImage image = QImage( widthInPixels, heightInPixels, QImage::Format_ARGB32 );
1210
1211 image.fill( Qt::transparent );
1212 image.setDotsPerMeterX( static_cast< int >( std::round( 1000 * destinationDpi / 25.4 ) ) );
1213 image.setDotsPerMeterY( static_cast< int >( std::round( 1000 * destinationDpi / 25.4 ) ) );
1214 double dotsPerMM = destinationDpi / 25.4;
1215 QPainter p( &image );
1216
1217 QPointF tl = -boundingRect().topLeft();
1218 QRect imagePaintRect( static_cast< int >( std::round( tl.x() * dotsPerMM ) ),
1219 static_cast< int >( std::round( tl.y() * dotsPerMM ) ),
1220 static_cast< int >( std::round( thisPaintRect.width() * dotsPerMM ) ),
1221 static_cast< int >( std::round( thisPaintRect.height() * dotsPerMM ) ) );
1222 p.setClipRect( imagePaintRect );
1223
1224 p.translate( imagePaintRect.topLeft() );
1225
1226 // Fill with background color - must be drawn onto the flattened image
1227 // so that layers with opacity or blend modes can correctly interact with it
1228 if ( shouldDrawPart( Background ) )
1229 {
1230 p.scale( dotsPerMM, dotsPerMM );
1231 drawMapBackground( &p );
1232 p.scale( 1.0 / dotsPerMM, 1.0 / dotsPerMM );
1233 }
1234
1235 drawMap( &p, cExtent, imagePaintRect.size(), image.logicalDpiX() );
1236
1237 // important - all other items, overviews, grids etc must be rendered to the
1238 // flattened image, in case these have blend modes must need to interact
1239 // with the map
1240 p.scale( dotsPerMM, dotsPerMM );
1241
1242 if ( shouldDrawPart( OverviewMapExtent ) )
1243 {
1244 mOverviewStack->drawItems( &p, false );
1245 }
1246 if ( shouldDrawPart( Grid ) )
1247 {
1248 mGridStack->drawItems( &p );
1249 }
1250 drawAnnotations( &p );
1251
1252 QgsScopedQPainterState painterState( painter );
1253 painter->setCompositionMode( blendModeForRender() );
1254 painter->scale( 1 / dotsPerMM, 1 / dotsPerMM ); // scale painter from mm to dots
1255 painter->drawImage( QPointF( -tl.x()* dotsPerMM, -tl.y() * dotsPerMM ), image );
1256 painter->scale( dotsPerMM, dotsPerMM );
1257 }
1258 else
1259 {
1260 // Fill with background color
1261 if ( shouldDrawPart( Background ) )
1262 {
1263 drawMapBackground( painter );
1264 }
1265
1266 QgsScopedQPainterState painterState( painter );
1267 painter->setClipRect( thisPaintRect );
1268
1269 if ( shouldDrawPart( Layer ) && !qgsDoubleNear( size.width(), 0.0 ) && !qgsDoubleNear( size.height(), 0.0 ) )
1270 {
1271 QgsScopedQPainterState stagedPainterState( painter );
1272 painter->translate( mXOffset, mYOffset );
1273
1274 double dotsPerMM = paintDevice->logicalDpiX() / 25.4;
1275 size *= dotsPerMM; // output size will be in dots (pixels)
1276 painter->scale( 1 / dotsPerMM, 1 / dotsPerMM ); // scale painter from mm to dots
1277
1278 if ( mCurrentExportPart != NotLayered )
1279 {
1280 if ( !mStagedRendererJob )
1281 {
1282 createStagedRenderJob( cExtent, size, paintDevice->logicalDpiX() );
1283 }
1284
1285 mStagedRendererJob->renderCurrentPart( painter );
1286 }
1287 else
1288 {
1289 drawMap( painter, cExtent, size, paintDevice->logicalDpiX() );
1290 }
1291 }
1292
1293 painter->setClipRect( thisPaintRect, Qt::NoClip );
1294
1295 if ( shouldDrawPart( OverviewMapExtent ) )
1296 {
1297 mOverviewStack->drawItems( painter, false );
1298 }
1299 if ( shouldDrawPart( Grid ) )
1300 {
1301 mGridStack->drawItems( painter );
1302 }
1303 drawAnnotations( painter );
1304 }
1305
1306 if ( shouldDrawPart( Frame ) )
1307 {
1308 drawMapFrame( painter );
1309 }
1310
1311 mDrawing = false;
1312 }
1313}
1314
1316{
1317 const int layerCount = layersToRender().length();
1318 return ( hasBackground() ? 1 : 0 )
1319 + ( layerCount + ( layerCount ? 1 : 0 ) ) // +1 for label layer, if labels present
1320 + ( mGridStack->hasEnabledItems() ? 1 : 0 )
1321 + ( mOverviewStack->hasEnabledItems() ? 1 : 0 )
1322 + ( frameEnabled() ? 1 : 0 );
1323}
1324
1326{
1327 mCurrentExportPart = Start;
1328 // only follow export themes if the map isn't set to follow a fixed theme
1329 mExportThemes = !mFollowVisibilityPreset ? mLayout->renderContext().exportThemes() : QStringList();
1330 mExportThemeIt = mExportThemes.begin();
1331}
1332
1334{
1335 mCurrentExportPart = NotLayered;
1336 mExportThemes.clear();
1337 mExportThemeIt = mExportThemes.begin();
1338}
1339
1341{
1342 switch ( mCurrentExportPart )
1343 {
1344 case Start:
1345 if ( hasBackground() )
1346 {
1347 mCurrentExportPart = Background;
1348 return true;
1349 }
1350 [[fallthrough]];
1351
1352 case Background:
1353 mCurrentExportPart = Layer;
1354 return true;
1355
1356 case Layer:
1357 if ( mStagedRendererJob )
1358 {
1359 if ( mStagedRendererJob->nextPart() )
1360 return true;
1361 else
1362 {
1363 mExportLabelingResults.reset( mStagedRendererJob->takeLabelingResults() );
1364 mStagedRendererJob.reset(); // no more map layer parts
1365 }
1366 }
1367
1368 if ( mExportThemeIt != mExportThemes.end() && ++mExportThemeIt != mExportThemes.end() )
1369 {
1370 // move to next theme and continue exporting map layers
1371 return true;
1372 }
1373
1374 if ( mGridStack->hasEnabledItems() )
1375 {
1376 mCurrentExportPart = Grid;
1377 return true;
1378 }
1379 [[fallthrough]];
1380
1381 case Grid:
1382 for ( int i = 0; i < mOverviewStack->size(); ++i )
1383 {
1384 QgsLayoutItemMapItem *item = mOverviewStack->item( i );
1386 {
1387 mCurrentExportPart = OverviewMapExtent;
1388 return true;
1389 }
1390 }
1391 [[fallthrough]];
1392
1393 case OverviewMapExtent:
1394 if ( frameEnabled() )
1395 {
1396 mCurrentExportPart = Frame;
1397 return true;
1398 }
1399
1400 [[fallthrough]];
1401
1402 case Frame:
1403 if ( isSelected() && !mLayout->renderContext().isPreviewRender() )
1404 {
1405 mCurrentExportPart = SelectionBoxes;
1406 return true;
1407 }
1408 [[fallthrough]];
1409
1410 case SelectionBoxes:
1411 mCurrentExportPart = End;
1412 return false;
1413
1414 case End:
1415 return false;
1416
1417 case NotLayered:
1418 return false;
1419 }
1420 return false;
1421}
1422
1427
1429{
1430 ExportLayerDetail detail;
1431
1432 switch ( mCurrentExportPart )
1433 {
1434 case Start:
1435 break;
1436
1437 case Background:
1438 detail.name = tr( "%1: Background" ).arg( displayName() );
1439 return detail;
1440
1441 case Layer:
1442 if ( !mExportThemes.empty() && mExportThemeIt != mExportThemes.end() )
1443 detail.mapTheme = *mExportThemeIt;
1444
1445 if ( mStagedRendererJob )
1446 {
1447 switch ( mStagedRendererJob->currentStage() )
1448 {
1450 {
1451 detail.mapLayerId = mStagedRendererJob->currentLayerId();
1452 detail.compositionMode = mStagedRendererJob->currentLayerCompositionMode();
1453 detail.opacity = mStagedRendererJob->currentLayerOpacity();
1454 if ( const QgsMapLayer *layer = mLayout->project()->mapLayer( detail.mapLayerId ) )
1455 {
1456 if ( !detail.mapTheme.isEmpty() )
1457 detail.name = QStringLiteral( "%1 (%2): %3" ).arg( displayName(), detail.mapTheme, layer->name() );
1458 else
1459 detail.name = QStringLiteral( "%1: %2" ).arg( displayName(), layer->name() );
1460 }
1461 else if ( mLayout->project()->mainAnnotationLayer()->id() == detail.mapLayerId )
1462 {
1463 // master annotation layer
1464 if ( !detail.mapTheme.isEmpty() )
1465 detail.name = QStringLiteral( "%1 (%2): %3" ).arg( displayName(), detail.mapTheme, tr( "Annotations" ) );
1466 else
1467 detail.name = QStringLiteral( "%1: %2" ).arg( displayName(), tr( "Annotations" ) );
1468 }
1469 else
1470 {
1471 // might be an item based layer
1472 const QList<QgsLayoutItemMapOverview *> res = mOverviewStack->asList();
1473 for ( QgsLayoutItemMapOverview *item : res )
1474 {
1475 if ( !item || !item->enabled() || item->stackingPosition() == QgsLayoutItemMapItem::StackAboveMapLabels )
1476 continue;
1477
1478 if ( item->mapLayer() && detail.mapLayerId == item->mapLayer()->id() )
1479 {
1480 if ( !detail.mapTheme.isEmpty() )
1481 detail.name = QStringLiteral( "%1 (%2): %3" ).arg( displayName(), detail.mapTheme, item->mapLayer()->name() );
1482 else
1483 detail.name = QStringLiteral( "%1: %2" ).arg( displayName(), item->mapLayer()->name() );
1484 break;
1485 }
1486 }
1487 }
1488 return detail;
1489 }
1490
1492 detail.mapLayerId = mStagedRendererJob->currentLayerId();
1493 if ( const QgsMapLayer *layer = mLayout->project()->mapLayer( detail.mapLayerId ) )
1494 {
1495 if ( !detail.mapTheme.isEmpty() )
1496 detail.name = QStringLiteral( "%1 (%2): %3 (Labels)" ).arg( displayName(), detail.mapTheme, layer->name() );
1497 else
1498 detail.name = tr( "%1: %2 (Labels)" ).arg( displayName(), layer->name() );
1499 }
1500 else
1501 {
1502 if ( !detail.mapTheme.isEmpty() )
1503 detail.name = tr( "%1 (%2): Labels" ).arg( displayName(), detail.mapTheme );
1504 else
1505 detail.name = tr( "%1: Labels" ).arg( displayName() );
1506 }
1507 return detail;
1508
1510 break;
1511 }
1512 }
1513 else
1514 {
1515 // we must be on the first layer, not having had a chance to create the render job yet
1516 const QList< QgsMapLayer * > layers = layersToRender();
1517 if ( !layers.isEmpty() )
1518 {
1519 const QgsMapLayer *layer = layers.constLast();
1520 if ( !detail.mapTheme.isEmpty() )
1521 detail.name = QStringLiteral( "%1 (%2): %3" ).arg( displayName(), detail.mapTheme, layer->name() );
1522 else
1523 detail.name = QStringLiteral( "%1: %2" ).arg( displayName(), layer->name() );
1524 detail.mapLayerId = layer->id();
1525 }
1526 }
1527 break;
1528
1529 case Grid:
1530 detail.name = tr( "%1: Grids" ).arg( displayName() );
1531 return detail;
1532
1533 case OverviewMapExtent:
1534 detail.name = tr( "%1: Overviews" ).arg( displayName() );
1535 return detail;
1536
1537 case Frame:
1538 detail.name = tr( "%1: Frame" ).arg( displayName() );
1539 return detail;
1540
1541 case SelectionBoxes:
1542 case End:
1543 case NotLayered:
1544 break;
1545 }
1546
1547 return detail;
1548}
1549
1555
1556void QgsLayoutItemMap::drawMap( QPainter *painter, const QgsRectangle &extent, QSizeF size, double dpi )
1557{
1558 if ( !painter )
1559 {
1560 return;
1561 }
1562 if ( qgsDoubleNear( size.width(), 0.0 ) || qgsDoubleNear( size.height(), 0.0 ) )
1563 {
1564 //don't attempt to draw if size is invalid
1565 return;
1566 }
1567
1568 // render
1569 QgsMapSettings ms( mapSettings( extent, size, dpi, true ) );
1570 if ( shouldDrawPart( OverviewMapExtent ) )
1571 {
1572 ms.setLayers( mOverviewStack->modifyMapLayerList( ms.layers() ) );
1573 }
1574
1575 QgsMapRendererCustomPainterJob job( ms, painter );
1576#ifdef HAVE_SERVER_PYTHON_PLUGINS
1577 job.setFeatureFilterProvider( mLayout->renderContext().featureFilterProvider() );
1578#endif
1579
1580 // Render the map in this thread. This is done because of problems
1581 // with printing to printer on Windows (printing to PDF is fine though).
1582 // Raster images were not displayed - see #10599
1583 job.renderSynchronously();
1584
1585 mExportLabelingResults.reset( job.takeLabelingResults() );
1586
1587 mRenderingErrors = job.errors();
1588}
1589
1590void QgsLayoutItemMap::recreateCachedImageInBackground()
1591{
1592 if ( mPainterJob )
1593 {
1594 disconnect( mPainterJob.get(), &QgsMapRendererCustomPainterJob::finished, this, &QgsLayoutItemMap::painterJobFinished );
1595 QgsMapRendererCustomPainterJob *oldJob = mPainterJob.release();
1596 QPainter *oldPainter = mPainter.release();
1597 QImage *oldImage = mCacheRenderingImage.release();
1598 connect( oldJob, &QgsMapRendererCustomPainterJob::finished, this, [oldPainter, oldJob, oldImage]
1599 {
1600 oldJob->deleteLater();
1601 delete oldPainter;
1602 delete oldImage;
1603 } );
1604 oldJob->cancelWithoutBlocking();
1605 }
1606 else
1607 {
1608 mCacheRenderingImage.reset( nullptr );
1610 }
1611
1612 Q_ASSERT( !mPainterJob );
1613 Q_ASSERT( !mPainter );
1614 Q_ASSERT( !mCacheRenderingImage );
1615
1616 QgsRectangle ext = extent();
1617 double widthLayoutUnits = ext.width() * mapUnitsToLayoutUnits();
1618 double heightLayoutUnits = ext.height() * mapUnitsToLayoutUnits();
1619
1620 int w = static_cast< int >( std::round( widthLayoutUnits * mPreviewScaleFactor ) );
1621 int h = static_cast< int >( std::round( heightLayoutUnits * mPreviewScaleFactor ) );
1622
1623 // limit size of image for better performance
1624 if ( w > 5000 || h > 5000 )
1625 {
1626 if ( w > h )
1627 {
1628 w = 5000;
1629 h = static_cast< int>( std::round( w * heightLayoutUnits / widthLayoutUnits ) );
1630 }
1631 else
1632 {
1633 h = 5000;
1634 w = static_cast< int >( std::round( h * widthLayoutUnits / heightLayoutUnits ) );
1635 }
1636 }
1637
1638 if ( w <= 0 || h <= 0 )
1639 return;
1640
1641 mCacheRenderingImage.reset( new QImage( w * mPreviewDevicePixelRatio, h * mPreviewDevicePixelRatio, QImage::Format_ARGB32 ) );
1642
1643 // set DPI of the image
1644 mCacheRenderingImage->setDotsPerMeterX( static_cast< int >( std::round( 1000 * w / widthLayoutUnits ) ) );
1645 mCacheRenderingImage->setDotsPerMeterY( static_cast< int >( std::round( 1000 * h / heightLayoutUnits ) ) );
1646 mCacheRenderingImage->setDevicePixelRatio( mPreviewDevicePixelRatio );
1647
1648 //start with empty fill to avoid artifacts
1649 mCacheRenderingImage->fill( QColor( 255, 255, 255, 0 ).rgba() );
1650 if ( hasBackground() )
1651 {
1652 //Initially fill image with specified background color. This ensures that layers with blend modes will
1653 //preview correctly
1654 if ( mItemClippingSettings->isActive() )
1655 {
1656 QPainter p( mCacheRenderingImage.get() );
1657 const QPainterPath path = framePath();
1658 p.setPen( Qt::NoPen );
1659 p.setBrush( backgroundColor() );
1660 p.scale( mCacheRenderingImage->width() / widthLayoutUnits, mCacheRenderingImage->height() / heightLayoutUnits );
1661 p.drawPath( path );
1662 p.end();
1663 }
1664 else
1665 {
1666 mCacheRenderingImage->fill( backgroundColor().rgba() );
1667 }
1668 }
1669
1670 mCacheInvalidated = false;
1671 mPainter.reset( new QPainter( mCacheRenderingImage.get() ) );
1672 QgsMapSettings settings( mapSettings( ext, QSizeF( w, h ), mCacheRenderingImage->logicalDpiX(), true ) );
1673
1674 if ( shouldDrawPart( OverviewMapExtent ) )
1675 {
1676 settings.setLayers( mOverviewStack->modifyMapLayerList( settings.layers() ) );
1677 }
1678
1679 mPainterJob.reset( new QgsMapRendererCustomPainterJob( settings, mPainter.get() ) );
1680 connect( mPainterJob.get(), &QgsMapRendererCustomPainterJob::finished, this, &QgsLayoutItemMap::painterJobFinished );
1681 mPainterJob->start();
1682
1683 // from now on we can accept refresh requests again
1684 // this must be reset only after the job has been started, because
1685 // some providers (yes, it's you WCS and AMS!) during preparation
1686 // do network requests and start an internal event loop, which may
1687 // end up calling refresh() and would schedule another refresh,
1688 // deleting the one we have just started.
1689
1690 // ^^ that comment was directly copied from a similar fix in QgsMapCanvas. And
1691 // with little surprise, both those providers are still badly behaved and causing
1692 // annoying bugs for us to deal with...
1693 mDrawingPreview = false;
1694}
1695
1697{
1698 return mMapFlags;
1699}
1700
1702{
1703 mMapFlags = mapFlags;
1704}
1705
1706QgsMapSettings QgsLayoutItemMap::mapSettings( const QgsRectangle &extent, QSizeF size, double dpi, bool includeLayerSettings ) const
1707{
1708 QgsExpressionContext expressionContext = createExpressionContext();
1709 QgsCoordinateReferenceSystem renderCrs = crs();
1710
1711 QgsMapSettings jobMapSettings;
1712 jobMapSettings.setDestinationCrs( renderCrs );
1713 jobMapSettings.setExtent( extent );
1714 jobMapSettings.setOutputSize( size.toSize() );
1715 jobMapSettings.setOutputDpi( dpi );
1716 if ( layout()->renderContext().isPreviewRender() )
1717 {
1718 jobMapSettings.setDpiTarget( layout()->renderContext().dpi() );
1719 jobMapSettings.setDevicePixelRatio( mPainter ? mPainter->device()->devicePixelRatioF() : 1.0 );
1720 }
1721 jobMapSettings.setBackgroundColor( Qt::transparent );
1722 jobMapSettings.setRotation( mEvaluatedMapRotation );
1723 if ( mLayout )
1724 {
1725 jobMapSettings.setEllipsoid( mLayout->project()->ellipsoid() );
1726 jobMapSettings.setElevationShadingRenderer( mLayout->project()->elevationShadingRenderer() );
1727 jobMapSettings.setScaleMethod( mLayout->project()->scaleMethod() );
1728 }
1729
1730 if ( includeLayerSettings )
1731 {
1732 //set layers to render
1733 QList<QgsMapLayer *> layers = layersToRender( &expressionContext );
1734
1735 if ( !mLayout->project()->mainAnnotationLayer()->isEmpty() )
1736 {
1737 // render main annotation layer above all other layers
1738 layers.insert( 0, mLayout->project()->mainAnnotationLayer() );
1739 }
1740
1741 jobMapSettings.setLayers( layers );
1742 jobMapSettings.setLayerStyleOverrides( layerStyleOverridesToRender( expressionContext ) );
1743 }
1744
1745 if ( !mLayout->renderContext().isPreviewRender() )
1746 {
1747 //if outputting layout, we disable optimisations like layer simplification by default, UNLESS the context specifically tells us to use them
1749 jobMapSettings.setSimplifyMethod( mLayout->renderContext().simplifyMethod() );
1750 jobMapSettings.setMaskSettings( mLayout->renderContext().maskSettings() );
1752 }
1753 else
1754 {
1755 // preview render - always use optimization
1757 // in a preview render we disable vector masking, as that is considerably slower vs raster masking
1758 jobMapSettings.setFlag( Qgis::MapSettingsFlag::ForceRasterMasks, true );
1760 }
1761
1762 jobMapSettings.setExpressionContext( expressionContext );
1763
1764 // layout-specific overrides of flags
1765 jobMapSettings.setFlag( Qgis::MapSettingsFlag::ForceVectorOutput, true ); // force vector output (no caching of marker images etc.)
1769 jobMapSettings.setFlag( Qgis::MapSettingsFlag::DrawEditingInfo, false );
1770 jobMapSettings.setSelectionColor( mLayout->renderContext().selectionColor() );
1775 jobMapSettings.setTransformContext( mLayout->project()->transformContext() );
1776 jobMapSettings.setPathResolver( mLayout->project()->pathResolver() );
1777
1778 QgsLabelingEngineSettings labelSettings = mLayout->project()->labelingEngineSettings();
1779
1780 // override project "show partial labels" setting with this map's setting
1784 jobMapSettings.setLabelingEngineSettings( labelSettings );
1785
1786 // override the default text render format inherited from the labeling engine settings using the layout's render context setting
1787 jobMapSettings.setTextRenderFormat( mLayout->renderContext().textRenderFormat() );
1788
1789 QgsGeometry labelBoundary;
1790 if ( mEvaluatedLabelMargin.length() > 0 )
1791 {
1792 QPolygonF visiblePoly = jobMapSettings.visiblePolygon();
1793 visiblePoly.append( visiblePoly.at( 0 ) ); //close polygon
1794 const double layoutLabelMargin = mLayout->convertToLayoutUnits( mEvaluatedLabelMargin );
1795 const double layoutLabelMarginInMapUnits = layoutLabelMargin / rect().width() * jobMapSettings.extent().width();
1796 QgsGeometry mapBoundaryGeom = QgsGeometry::fromQPolygonF( visiblePoly );
1797 mapBoundaryGeom = mapBoundaryGeom.buffer( -layoutLabelMarginInMapUnits, 0 );
1798 labelBoundary = mapBoundaryGeom;
1799 }
1800
1801 if ( !mBlockingLabelItems.isEmpty() )
1802 {
1803 jobMapSettings.setLabelBlockingRegions( createLabelBlockingRegions( jobMapSettings ) );
1804 }
1805
1806 for ( QgsRenderedFeatureHandlerInterface *handler : std::as_const( mRenderedFeatureHandlers ) )
1807 {
1808 jobMapSettings.addRenderedFeatureHandler( handler );
1809 }
1810
1811 if ( isTemporal() )
1812 jobMapSettings.setTemporalRange( temporalRange() );
1813
1814 if ( mZRangeEnabled )
1815 {
1816 jobMapSettings.setZRange( mZRange );
1817 }
1818
1819 if ( mAtlasClippingSettings->enabled() && mLayout->reportContext().feature().isValid() )
1820 {
1821 QgsGeometry clipGeom( mLayout->reportContext().currentGeometry( jobMapSettings.destinationCrs() ) );
1822 QgsMapClippingRegion region( clipGeom );
1823 region.setFeatureClip( mAtlasClippingSettings->featureClippingType() );
1824 region.setRestrictedLayers( mAtlasClippingSettings->layersToClip() );
1825 region.setRestrictToLayers( mAtlasClippingSettings->restrictToLayers() );
1826 jobMapSettings.addClippingRegion( region );
1827
1828 if ( mAtlasClippingSettings->forceLabelsInsideFeature() )
1829 {
1830 if ( !labelBoundary.isEmpty() )
1831 {
1832 labelBoundary = clipGeom.intersection( labelBoundary );
1833 }
1834 else
1835 {
1836 labelBoundary = clipGeom;
1837 }
1838 }
1839 }
1840
1841 if ( mItemClippingSettings->isActive() )
1842 {
1843 const QgsGeometry clipGeom = mItemClippingSettings->clippedMapExtent();
1844 if ( !clipGeom.isEmpty() )
1845 {
1846 jobMapSettings.addClippingRegion( mItemClippingSettings->toMapClippingRegion() );
1847
1848 if ( mItemClippingSettings->forceLabelsInsideClipPath() )
1849 {
1850 const double layoutLabelMargin = mLayout->convertToLayoutUnits( mEvaluatedLabelMargin );
1851 const double layoutLabelMarginInMapUnits = layoutLabelMargin / rect().width() * jobMapSettings.extent().width();
1852 QgsGeometry mapBoundaryGeom = clipGeom;
1853 mapBoundaryGeom = mapBoundaryGeom.buffer( -layoutLabelMarginInMapUnits, 0 );
1854 if ( !labelBoundary.isEmpty() )
1855 {
1856 labelBoundary = mapBoundaryGeom.intersection( labelBoundary );
1857 }
1858 else
1859 {
1860 labelBoundary = mapBoundaryGeom;
1861 }
1862 }
1863 }
1864 }
1865
1866 if ( !labelBoundary.isNull() )
1867 jobMapSettings.setLabelBoundaryGeometry( labelBoundary );
1868
1869 return jobMapSettings;
1870}
1871
1873{
1874 assignFreeId();
1875
1876 mBlockingLabelItems.clear();
1877 for ( const QString &uuid : std::as_const( mBlockingLabelItemUuids ) )
1878 {
1879 QgsLayoutItem *item = mLayout->itemByUuid( uuid, true );
1880 if ( item )
1881 {
1882 addLabelBlockingItem( item );
1883 }
1884 }
1885
1886 mOverviewStack->finalizeRestoreFromXml();
1887 mGridStack->finalizeRestoreFromXml();
1888 mItemClippingSettings->finalizeRestoreFromXml();
1889}
1890
1891void QgsLayoutItemMap::setMoveContentPreviewOffset( double xOffset, double yOffset )
1892{
1893 mXOffset = xOffset;
1894 mYOffset = yOffset;
1895}
1896
1898{
1899 return mCurrentRectangle;
1900}
1901
1903{
1905
1906 //Can't utilize QgsExpressionContextUtils::mapSettingsScope as we don't always
1907 //have a QgsMapSettings object available when the context is required, so we manually
1908 //add the same variables here
1909 QgsExpressionContextScope *scope = new QgsExpressionContextScope( tr( "Map Settings" ) );
1910
1911 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_id" ), id(), true ) );
1912 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_rotation" ), mMapRotation, true ) );
1913 const double mapScale = scale();
1914 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_scale" ), mapScale, true ) );
1915
1916 scope->setVariable( QStringLiteral( "zoom_level" ), !qgsDoubleNear( mapScale, 0 ) ? QgsVectorTileUtils::scaleToZoomLevel( mapScale, 0, 99999 ) : 0, true );
1917 scope->setVariable( QStringLiteral( "vector_tile_zoom" ), !qgsDoubleNear( mapScale, 0 ) ? QgsVectorTileUtils::scaleToZoom( mapScale ) : 0, true );
1918
1919 QgsRectangle currentExtent( extent() );
1920 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_extent" ), QVariant::fromValue( QgsGeometry::fromRect( currentExtent ) ), true ) );
1921 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_extent_width" ), currentExtent.width(), true ) );
1922 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_extent_height" ), currentExtent.height(), true ) );
1923 QgsGeometry centerPoint = QgsGeometry::fromPointXY( currentExtent.center() );
1924 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_extent_center" ), QVariant::fromValue( centerPoint ), true ) );
1925
1927 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs" ), mapCrs.authid(), true ) );
1928 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_definition" ), mapCrs.toProj(), true ) );
1929 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_description" ), mapCrs.description(), true ) );
1930 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_units" ), QgsUnitTypes::toString( mapCrs.mapUnits() ), true ) );
1931 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_acronym" ), mapCrs.projectionAcronym(), true ) );
1932 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_projection" ), mapCrs.operation().description(), true ) );
1933 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_ellipsoid" ), mapCrs.ellipsoidAcronym(), true ) );
1934 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_proj4" ), mapCrs.toProj(), true ) );
1935 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_crs_wkt" ), mapCrs.toWkt( Qgis::CrsWktVariant::Preferred ), true ) );
1936
1937 QVariantList layersIds;
1938 QVariantList layers;
1939 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_layer_ids" ), layersIds, true ) );
1940 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_layers" ), layers, true ) );
1941
1942 context.appendScope( scope );
1943
1944 // The scope map_layer_ids and map_layers variables have been added to the context, only now we can
1945 // call layersToRender (just in case layersToRender relies on evaluating an expression which uses
1946 // other variables contained within the map settings scope
1947 const QList<QgsMapLayer *> layersInMap = layersToRender( &context );
1948
1949 layersIds.reserve( layersInMap.count() );
1950 layers.reserve( layersInMap.count() );
1951 for ( QgsMapLayer *layer : layersInMap )
1952 {
1953 layersIds << layer->id();
1954 layers << QVariant::fromValue<QgsWeakMapLayerPointer>( QgsWeakMapLayerPointer( layer ) );
1955 }
1956 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_layer_ids" ), layersIds, true ) );
1957 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_layers" ), layers, true ) );
1958
1959 scope->addFunction( QStringLiteral( "is_layer_visible" ), new QgsExpressionContextUtils::GetLayerVisibility( layersInMap, scale() ) );
1960
1961 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_start_time" ), isTemporal() ? temporalRange().begin() : QVariant(), true ) );
1962 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_end_time" ), isTemporal() ? temporalRange().end() : QVariant(), true ) );
1963 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_interval" ), isTemporal() ? QgsInterval( temporalRange().end() - temporalRange().begin() ) : QVariant(), true ) );
1964
1965 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_z_range_lower" ), mZRangeEnabled ? mZRange.lower() : QVariant(), true ) );
1966 scope->addVariable( QgsExpressionContextScope::StaticVariable( QStringLiteral( "map_z_range_upper" ), mZRangeEnabled ? mZRange.upper() : QVariant(), true ) );
1967
1968#if 0 // not relevant here! (but left so as to respect all the dangerous warnings in QgsExpressionContextUtils::mapSettingsScope)
1969 if ( mapSettings.frameRate() >= 0 )
1970 scope->setVariable( QStringLiteral( "frame_rate" ), mapSettings.frameRate(), true );
1971 if ( mapSettings.currentFrame() >= 0 )
1972 scope->setVariable( QStringLiteral( "frame_number" ), mapSettings.currentFrame(), true );
1973#endif
1974
1975 return context;
1976}
1977
1979{
1980 double extentWidth = extent().width();
1981 if ( extentWidth <= 0 )
1982 {
1983 return 1;
1984 }
1985 return rect().width() / extentWidth;
1986}
1987
1989{
1990 double dx = mXOffset;
1991 double dy = mYOffset;
1992 transformShift( dx, dy );
1993 QPolygonF poly = calculateVisibleExtentPolygon( false );
1994 poly.translate( -dx, -dy );
1995 return poly;
1996}
1997
1999{
2000 if ( !mBlockingLabelItems.contains( item ) )
2001 mBlockingLabelItems.append( item );
2002
2003 connect( item, &QgsLayoutItem::sizePositionChanged, this, &QgsLayoutItemMap::invalidateCache, Qt::UniqueConnection );
2004}
2005
2007{
2008 mBlockingLabelItems.removeAll( item );
2009 if ( item )
2011}
2012
2014{
2015 return mBlockingLabelItems.contains( item );
2016}
2017
2019{
2020 return mPreviewLabelingResults.get();
2021}
2022
2024{
2025 // NOTE: if visitEnter returns false it means "don't visit the item", not "abort all further visitations"
2027 return true;
2028
2029 if ( mOverviewStack )
2030 {
2031 for ( int i = 0; i < mOverviewStack->size(); ++i )
2032 {
2033 if ( mOverviewStack->item( i )->accept( visitor ) )
2034 return false;
2035 }
2036 }
2037
2038 if ( mGridStack )
2039 {
2040 for ( int i = 0; i < mGridStack->size(); ++i )
2041 {
2042 if ( mGridStack->item( i )->accept( visitor ) )
2043 return false;
2044 }
2045 }
2046
2048 return false;
2049
2050 return true;
2051}
2052
2054{
2055 mRenderedFeatureHandlers.append( handler );
2056}
2057
2059{
2060 mRenderedFeatureHandlers.removeAll( handler );
2061}
2062
2063QPointF QgsLayoutItemMap::mapToItemCoords( QPointF mapCoords ) const
2064{
2065 QPolygonF mapPoly = transformedMapPolygon();
2066 if ( mapPoly.empty() )
2067 {
2068 return QPointF( 0, 0 );
2069 }
2070
2071 QgsRectangle tExtent = transformedExtent();
2072 QgsPointXY rotationPoint( ( tExtent.xMaximum() + tExtent.xMinimum() ) / 2.0, ( tExtent.yMaximum() + tExtent.yMinimum() ) / 2.0 );
2073 double dx = mapCoords.x() - rotationPoint.x();
2074 double dy = mapCoords.y() - rotationPoint.y();
2075 QgsLayoutUtils::rotate( -mEvaluatedMapRotation, dx, dy );
2076 QgsPointXY backRotatedCoords( rotationPoint.x() + dx, rotationPoint.y() + dy );
2077
2078 QgsRectangle unrotatedExtent = transformedExtent();
2079 double xItem = rect().width() * ( backRotatedCoords.x() - unrotatedExtent.xMinimum() ) / unrotatedExtent.width();
2080 double yItem = rect().height() * ( 1 - ( backRotatedCoords.y() - unrotatedExtent.yMinimum() ) / unrotatedExtent.height() );
2081 return QPointF( xItem, yItem );
2082}
2083
2085{
2087 QgsRectangle newExtent = mExtent;
2088 if ( qgsDoubleNear( mEvaluatedMapRotation, 0.0 ) )
2089 {
2090 extent = newExtent;
2091 }
2092 else
2093 {
2094 QPolygonF poly;
2095 mapPolygon( newExtent, poly );
2096 QRectF bRect = poly.boundingRect();
2097 extent.setXMinimum( bRect.left() );
2098 extent.setXMaximum( bRect.right() );
2099 extent.setYMinimum( bRect.top() );
2100 extent.setYMaximum( bRect.bottom() );
2101 }
2102 return extent;
2103}
2104
2106{
2107 if ( mDrawing )
2108 return;
2109
2110 mCacheInvalidated = true;
2111 update();
2112}
2113
2115{
2116 QRectF rectangle = rect();
2117 double frameExtension = frameEnabled() ? pen().widthF() / 2.0 : 0.0;
2118
2119 double topExtension = 0.0;
2120 double rightExtension = 0.0;
2121 double bottomExtension = 0.0;
2122 double leftExtension = 0.0;
2123
2124 if ( mGridStack )
2125 mGridStack->calculateMaxGridExtension( topExtension, rightExtension, bottomExtension, leftExtension );
2126
2127 topExtension = std::max( topExtension, frameExtension );
2128 rightExtension = std::max( rightExtension, frameExtension );
2129 bottomExtension = std::max( bottomExtension, frameExtension );
2130 leftExtension = std::max( leftExtension, frameExtension );
2131
2132 rectangle.setLeft( rectangle.left() - leftExtension );
2133 rectangle.setRight( rectangle.right() + rightExtension );
2134 rectangle.setTop( rectangle.top() - topExtension );
2135 rectangle.setBottom( rectangle.bottom() + bottomExtension );
2136 if ( rectangle != mCurrentRectangle )
2137 {
2138 prepareGeometryChange();
2139 mCurrentRectangle = rectangle;
2140 }
2141}
2142
2144{
2147 {
2148 bool ok;
2149 const QString crsVar = mDataDefinedProperties.valueAsString( QgsLayoutObject::DataDefinedProperty::MapCrs, context, QString(), &ok );
2150 if ( ok && QgsCoordinateReferenceSystem( crsVar ).isValid() )
2151 {
2152 const QgsCoordinateReferenceSystem newCrs( crsVar );
2153 if ( newCrs.isValid() )
2154 {
2155 setCrs( newCrs );
2156 }
2157 }
2158 }
2159 //updates data defined properties and redraws item to match
2165 {
2166 QgsRectangle beforeExtent = mExtent;
2167 refreshMapExtents( &context );
2168 emit changed();
2169 if ( mExtent != beforeExtent )
2170 {
2171 emit extentChanged();
2172 }
2173 }
2175 {
2176 refreshLabelMargin( false );
2177 }
2179 {
2180 const QString previousTheme = mLastEvaluatedThemeName.isEmpty() ? mFollowVisibilityPresetName : mLastEvaluatedThemeName;
2181 mLastEvaluatedThemeName = mDataDefinedProperties.valueAsString( QgsLayoutObject::DataDefinedProperty::MapStylePreset, context, mFollowVisibilityPresetName );
2182 if ( mLastEvaluatedThemeName != previousTheme )
2183 emit themeChanged( mLastEvaluatedThemeName );
2184 }
2185
2187 {
2188 QDateTime begin = temporalRange().begin();
2189 QDateTime end = temporalRange().end();
2190
2195
2196 setTemporalRange( QgsDateTimeRange( begin, end, true, begin == end ) );
2197 }
2198
2200 {
2201 double zLower = mZRange.lower();
2202 double zUpper = mZRange.upper();
2203
2208
2209 mZRange = QgsDoubleRange( zLower, zUpper );
2210 }
2211
2212 //force redraw
2213 mCacheInvalidated = true;
2214
2216}
2217
2218void QgsLayoutItemMap::layersAboutToBeRemoved( const QList<QgsMapLayer *> &layers )
2219{
2220
2221 if ( !mLayers.isEmpty() || mLayerStyleOverrides.isEmpty() )
2222 {
2223 for ( QgsMapLayer *layer : layers )
2224 {
2225 mLayerStyleOverrides.remove( layer->id() );
2226 }
2227 _qgis_removeLayers( mLayers, layers );
2228 }
2229
2230 for ( QgsMapLayer *layer : std::as_const( layers ) )
2231 {
2232 // Remove groups
2233 if ( mGroupLayers.erase( layer->id() ) == 0 )
2234 {
2235 // Remove group children
2236 for ( auto it = mGroupLayers.begin(); it != mGroupLayers.end(); ++it )
2237 {
2238 QgsGroupLayer *groupLayer = it->second.get();
2239 if ( groupLayer->childLayers().contains( layer ) )
2240 {
2241 QList<QgsMapLayer *> childLayers { groupLayer->childLayers() };
2242 childLayers.removeAll( layer );
2243 groupLayer->setChildLayers( childLayers );
2244 }
2245 }
2246 }
2247 }
2248}
2249
2250void QgsLayoutItemMap::painterJobFinished()
2251{
2252 mPainter->end();
2253 mPreviewLabelingResults.reset( mPainterJob->takeLabelingResults() );
2254 mPainterJob.reset( nullptr );
2255 mPainter.reset( nullptr );
2256 mCacheFinalImage = std::move( mCacheRenderingImage );
2257 mLastRenderedImageOffsetX = 0;
2258 mLastRenderedImageOffsetY = 0;
2260 update();
2261 emit previewRefreshed();
2262}
2263
2264void QgsLayoutItemMap::shapeChanged()
2265{
2266 // keep center as center
2267 QgsPointXY oldCenter = mExtent.center();
2268
2269 double w = rect().width();
2270 double h = rect().height();
2271
2272 // keep same width as before
2273 double newWidth = mExtent.width();
2274 // but scale height to match item's aspect ratio
2275 double newHeight = newWidth * h / w;
2276
2277 mExtent = QgsRectangle::fromCenterAndSize( oldCenter, newWidth, newHeight );
2278
2279 //recalculate data defined scale and extents
2280 refreshMapExtents();
2283 emit changed();
2284 emit extentChanged();
2285}
2286
2287void QgsLayoutItemMap::mapThemeChanged( const QString &theme )
2288{
2289 if ( theme == mCachedLayerStyleOverridesPresetName )
2290 mCachedLayerStyleOverridesPresetName.clear(); // force cache regeneration at next redraw
2291}
2292
2293void QgsLayoutItemMap::currentMapThemeRenamed( const QString &theme, const QString &newTheme )
2294{
2295 if ( theme == mFollowVisibilityPresetName )
2296 {
2297 mFollowVisibilityPresetName = newTheme;
2298 }
2299}
2300
2301void QgsLayoutItemMap::connectUpdateSlot()
2302{
2303 //connect signal from layer registry to update in case of new or deleted layers
2304 QgsProject *project = mLayout->project();
2305 if ( project )
2306 {
2307 // handles updating the stored layer state BEFORE the layers are removed
2308 connect( project, static_cast < void ( QgsProject::* )( const QList<QgsMapLayer *>& layers ) > ( &QgsProject::layersWillBeRemoved ),
2309 this, &QgsLayoutItemMap::layersAboutToBeRemoved );
2310 // redraws the map AFTER layers are removed
2311 connect( project->layerTreeRoot(), &QgsLayerTree::layerOrderChanged, this, [this]
2312 {
2313 if ( layers().isEmpty() )
2314 {
2315 //using project layers, and layer order has changed
2316 invalidateCache();
2317 }
2318 } );
2319
2320 connect( project, &QgsProject::crsChanged, this, [this]
2321 {
2322 if ( !mCrs.isValid() )
2323 {
2324 //using project CRS, which just changed....
2325 invalidateCache();
2326 emit crsChanged();
2327 }
2328 } );
2329
2330 // If project colors change, we need to redraw the map, as layer symbols may rely on project colors
2331 connect( project, &QgsProject::projectColorsChanged, this, [this]
2332 {
2334 } );
2335
2336 connect( project->mapThemeCollection(), &QgsMapThemeCollection::mapThemeChanged, this, &QgsLayoutItemMap::mapThemeChanged );
2337 connect( project->mapThemeCollection(), &QgsMapThemeCollection::mapThemeRenamed, this, &QgsLayoutItemMap::currentMapThemeRenamed );
2338 }
2339 connect( mLayout, &QgsLayout::refreshed, this, &QgsLayoutItemMap::invalidateCache );
2340 connect( &mLayout->renderContext(), &QgsLayoutRenderContext::predefinedScalesChanged, this, [this]
2341 {
2342 if ( mAtlasScalingMode == Predefined )
2343 updateAtlasFeature();
2344 } );
2345}
2346
2348{
2349 QPolygonF thisExtent = calculateVisibleExtentPolygon( false );
2350 QTransform mapTransform;
2351 QPolygonF thisRectPoly = QPolygonF( QRectF( 0, 0, rect().width(), rect().height() ) );
2352 //workaround QT Bug #21329
2353 thisRectPoly.pop_back();
2354 thisExtent.pop_back();
2355
2356 QPolygonF thisItemPolyInLayout = mapToScene( thisRectPoly );
2357
2358 //create transform from layout coordinates to map coordinates
2359 QTransform::quadToQuad( thisItemPolyInLayout, thisExtent, mapTransform );
2360 return mapTransform;
2361}
2362
2364{
2365 mZRangeEnabled = enabled;
2366}
2367
2369{
2370 return mZRangeEnabled;
2371}
2372
2374{
2375 return mZRange;
2376}
2377
2379{
2380 mZRange = range;
2381}
2382
2383QList<QgsLabelBlockingRegion> QgsLayoutItemMap::createLabelBlockingRegions( const QgsMapSettings & ) const
2384{
2385 const QTransform mapTransform = layoutToMapCoordsTransform();
2386 QList< QgsLabelBlockingRegion > blockers;
2387 blockers.reserve( mBlockingLabelItems.count() );
2388 for ( const auto &item : std::as_const( mBlockingLabelItems ) )
2389 {
2390 // invisible items don't block labels!
2391 if ( !item )
2392 continue;
2393
2394 // layout items may be temporarily hidden during layered exports
2395 if ( item->property( "wasVisible" ).isValid() )
2396 {
2397 if ( !item->property( "wasVisible" ).toBool() )
2398 continue;
2399 }
2400 else if ( !item->isVisible() )
2401 continue;
2402
2403 QPolygonF itemRectInMapCoordinates = mapTransform.map( item->mapToScene( item->rect() ) );
2404 itemRectInMapCoordinates.append( itemRectInMapCoordinates.at( 0 ) ); //close polygon
2405 QgsGeometry blockingRegion = QgsGeometry::fromQPolygonF( itemRectInMapCoordinates );
2406 blockers << QgsLabelBlockingRegion( blockingRegion );
2407 }
2408 return blockers;
2409}
2410
2412{
2413 return mLabelMargin;
2414}
2415
2417{
2418 mLabelMargin = margin;
2419 refreshLabelMargin( false );
2420}
2421
2422void QgsLayoutItemMap::updateToolTip()
2423{
2424 setToolTip( displayName() );
2425}
2426
2427QString QgsLayoutItemMap::themeToRender( const QgsExpressionContext &context ) const
2428{
2429 QString presetName;
2430
2431 if ( mFollowVisibilityPreset )
2432 {
2433 presetName = mFollowVisibilityPresetName;
2434 // preset name can be overridden by data-defined one
2436 }
2437 else if ( !mExportThemes.empty() && mExportThemeIt != mExportThemes.end() )
2438 presetName = *mExportThemeIt;
2439 return presetName;
2440}
2441
2442QList<QgsMapLayer *> QgsLayoutItemMap::layersToRender( const QgsExpressionContext *context ) const
2443{
2444 QgsExpressionContext scopedContext;
2445 if ( !context )
2446 scopedContext = createExpressionContext();
2447 const QgsExpressionContext *evalContext = context ? context : &scopedContext;
2448
2449 QList<QgsMapLayer *> renderLayers;
2450
2451 QString presetName = themeToRender( *evalContext );
2452 if ( !presetName.isEmpty() )
2453 {
2454 if ( mLayout->project()->mapThemeCollection()->hasMapTheme( presetName ) )
2455 renderLayers = mLayout->project()->mapThemeCollection()->mapThemeVisibleLayers( presetName );
2456 else // fallback to using map canvas layers
2457 renderLayers = mLayout->project()->mapThemeCollection()->masterVisibleLayers();
2458 }
2459 else if ( !layers().isEmpty() )
2460 {
2461 renderLayers = layers();
2462 }
2463 else
2464 {
2465 renderLayers = mLayout->project()->mapThemeCollection()->masterVisibleLayers();
2466 }
2467
2468 bool ok = false;
2469 QString ddLayers = mDataDefinedProperties.valueAsString( QgsLayoutObject::DataDefinedProperty::MapLayers, *evalContext, QString(), &ok );
2470 if ( ok )
2471 {
2472 renderLayers.clear();
2473
2474 const QStringList layerNames = ddLayers.split( '|' );
2475 //need to convert layer names to layer ids
2476 for ( const QString &name : layerNames )
2477 {
2478 const QList< QgsMapLayer * > matchingLayers = mLayout->project()->mapLayersByName( name );
2479 for ( QgsMapLayer *layer : matchingLayers )
2480 {
2481 renderLayers << layer;
2482 }
2483 }
2484 }
2485
2486 //remove atlas coverage layer if required
2487 if ( mLayout->renderContext().flags() & QgsLayoutRenderContext::FlagHideCoverageLayer )
2488 {
2489 //hiding coverage layer
2490 int removeAt = renderLayers.indexOf( mLayout->reportContext().layer() );
2491 if ( removeAt != -1 )
2492 {
2493 renderLayers.removeAt( removeAt );
2494 }
2495 }
2496
2497 // remove any invalid layers
2498 renderLayers.erase( std::remove_if( renderLayers.begin(), renderLayers.end(), []( QgsMapLayer * layer )
2499 {
2500 return !layer || !layer->isValid();
2501 } ), renderLayers.end() );
2502
2503 return renderLayers;
2504}
2505
2506QMap<QString, QString> QgsLayoutItemMap::layerStyleOverridesToRender( const QgsExpressionContext &context ) const
2507{
2508 QString presetName = themeToRender( context );
2509 if ( !presetName.isEmpty() )
2510 {
2511 if ( mLayout->project()->mapThemeCollection()->hasMapTheme( presetName ) )
2512 {
2513 if ( presetName != mCachedLayerStyleOverridesPresetName )
2514 {
2515 // have to regenerate cache of style overrides
2516 mCachedPresetLayerStyleOverrides = mLayout->project()->mapThemeCollection()->mapThemeStyleOverrides( presetName );
2517 mCachedLayerStyleOverridesPresetName = presetName;
2518 }
2519
2520 return mCachedPresetLayerStyleOverrides;
2521 }
2522 else
2523 return QMap<QString, QString>();
2524 }
2525 else if ( mFollowVisibilityPreset )
2526 {
2527 QString presetName = mFollowVisibilityPresetName;
2528 // data defined preset name?
2530 if ( mLayout->project()->mapThemeCollection()->hasMapTheme( presetName ) )
2531 {
2532 if ( presetName.isEmpty() || presetName != mCachedLayerStyleOverridesPresetName )
2533 {
2534 // have to regenerate cache of style overrides
2535 mCachedPresetLayerStyleOverrides = mLayout->project()->mapThemeCollection()->mapThemeStyleOverrides( presetName );
2536 mCachedLayerStyleOverridesPresetName = presetName;
2537 }
2538
2539 return mCachedPresetLayerStyleOverrides;
2540 }
2541 else
2542 return QMap<QString, QString>();
2543 }
2544 else if ( mKeepLayerStyles )
2545 {
2546 return mLayerStyleOverrides;
2547 }
2548 else
2549 {
2550 return QMap<QString, QString>();
2551 }
2552}
2553
2554QgsRectangle QgsLayoutItemMap::transformedExtent() const
2555{
2556 double dx = mXOffset;
2557 double dy = mYOffset;
2558 transformShift( dx, dy );
2559 return QgsRectangle( mExtent.xMinimum() - dx, mExtent.yMinimum() - dy, mExtent.xMaximum() - dx, mExtent.yMaximum() - dy );
2560}
2561
2562void QgsLayoutItemMap::mapPolygon( const QgsRectangle &extent, QPolygonF &poly ) const
2563{
2564 poly.clear();
2565 if ( qgsDoubleNear( mEvaluatedMapRotation, 0.0 ) )
2566 {
2567 poly << QPointF( extent.xMinimum(), extent.yMaximum() );
2568 poly << QPointF( extent.xMaximum(), extent.yMaximum() );
2569 poly << QPointF( extent.xMaximum(), extent.yMinimum() );
2570 poly << QPointF( extent.xMinimum(), extent.yMinimum() );
2571 //ensure polygon is closed by readding first point
2572 poly << QPointF( poly.at( 0 ) );
2573 return;
2574 }
2575
2576 //there is rotation
2577 QgsPointXY rotationPoint( ( extent.xMaximum() + extent.xMinimum() ) / 2.0, ( extent.yMaximum() + extent.yMinimum() ) / 2.0 );
2578 double dx, dy; //x-, y- shift from rotation point to corner point
2579
2580 //top left point
2581 dx = rotationPoint.x() - extent.xMinimum();
2582 dy = rotationPoint.y() - extent.yMaximum();
2583 QgsLayoutUtils::rotate( mEvaluatedMapRotation, dx, dy );
2584 poly << QPointF( rotationPoint.x() - dx, rotationPoint.y() - dy );
2585
2586 //top right point
2587 dx = rotationPoint.x() - extent.xMaximum();
2588 dy = rotationPoint.y() - extent.yMaximum();
2589 QgsLayoutUtils::rotate( mEvaluatedMapRotation, dx, dy );
2590 poly << QPointF( rotationPoint.x() - dx, rotationPoint.y() - dy );
2591
2592 //bottom right point
2593 dx = rotationPoint.x() - extent.xMaximum();
2594 dy = rotationPoint.y() - extent.yMinimum();
2595 QgsLayoutUtils::rotate( mEvaluatedMapRotation, dx, dy );
2596 poly << QPointF( rotationPoint.x() - dx, rotationPoint.y() - dy );
2597
2598 //bottom left point
2599 dx = rotationPoint.x() - extent.xMinimum();
2600 dy = rotationPoint.y() - extent.yMinimum();
2601 QgsLayoutUtils::rotate( mEvaluatedMapRotation, dx, dy );
2602 poly << QPointF( rotationPoint.x() - dx, rotationPoint.y() - dy );
2603
2604 //ensure polygon is closed by readding first point
2605 poly << QPointF( poly.at( 0 ) );
2606}
2607
2608void QgsLayoutItemMap::transformShift( double &xShift, double &yShift ) const
2609{
2610 double mmToMapUnits = 1.0 / mapUnitsToLayoutUnits();
2611 double dxScaled = xShift * mmToMapUnits;
2612 double dyScaled = - yShift * mmToMapUnits;
2613
2614 QgsLayoutUtils::rotate( mEvaluatedMapRotation, dxScaled, dyScaled );
2615
2616 xShift = dxScaled;
2617 yShift = dyScaled;
2618}
2619
2620void QgsLayoutItemMap::drawAnnotations( QPainter *painter )
2621{
2622 if ( !mLayout || !mLayout->project() || !mDrawAnnotations )
2623 {
2624 return;
2625 }
2626
2627 const QList< QgsAnnotation * > annotations = mLayout->project()->annotationManager()->annotations();
2628 if ( annotations.isEmpty() )
2629 return;
2630
2632 rc.setForceVectorOutput( true );
2634 QList< QgsMapLayer * > layers = layersToRender( &rc.expressionContext() );
2635
2636 for ( QgsAnnotation *annotation : annotations )
2637 {
2638 if ( !annotation || !annotation->isVisible() )
2639 {
2640 continue;
2641 }
2642 if ( annotation->mapLayer() && !layers.contains( annotation->mapLayer() ) )
2643 continue;
2644
2645 drawAnnotation( annotation, rc );
2646 }
2647}
2648
2649void QgsLayoutItemMap::drawAnnotation( const QgsAnnotation *annotation, QgsRenderContext &context )
2650{
2651 if ( !annotation || !annotation->isVisible() || !context.painter() || !context.painter()->device() )
2652 {
2653 return;
2654 }
2655
2656 QgsScopedQPainterState painterState( context.painter() );
2658
2659 double itemX, itemY;
2660 if ( annotation->hasFixedMapPosition() )
2661 {
2662 QPointF mapPos = layoutMapPosForItem( annotation );
2663 itemX = mapPos.x();
2664 itemY = mapPos.y();
2665 }
2666 else
2667 {
2668 itemX = annotation->relativePosition().x() * rect().width();
2669 itemY = annotation->relativePosition().y() * rect().height();
2670 }
2671 context.painter()->translate( itemX, itemY );
2672
2673 //setup painter scaling to dots so that symbology is drawn to scale
2674 double dotsPerMM = context.painter()->device()->logicalDpiX() / 25.4;
2675 context.painter()->scale( 1 / dotsPerMM, 1 / dotsPerMM ); // scale painter from mm to dots
2676
2677 annotation->render( context );
2678}
2679
2680QPointF QgsLayoutItemMap::layoutMapPosForItem( const QgsAnnotation *annotation ) const
2681{
2682 if ( !annotation )
2683 return QPointF( 0, 0 );
2684
2685 double mapX = 0.0;
2686 double mapY = 0.0;
2687
2688 mapX = annotation->mapPosition().x();
2689 mapY = annotation->mapPosition().y();
2690 QgsCoordinateReferenceSystem annotationCrs = annotation->mapPositionCrs();
2691
2692 if ( annotationCrs != crs() )
2693 {
2694 //need to reproject
2695 QgsCoordinateTransform t( annotationCrs, crs(), mLayout->project() );
2696 double z = 0.0;
2697 try
2698 {
2699 t.transformInPlace( mapX, mapY, z );
2700 }
2701 catch ( const QgsCsException & )
2702 {
2703 }
2704 }
2705
2706 return mapToItemCoords( QPointF( mapX, mapY ) );
2707}
2708
2709void QgsLayoutItemMap::drawMapFrame( QPainter *p )
2710{
2711 if ( frameEnabled() && p )
2712 {
2715
2717 }
2718}
2719
2720void QgsLayoutItemMap::drawMapBackground( QPainter *p )
2721{
2722 if ( hasBackground() && p )
2723 {
2726
2728 }
2729}
2730
2731bool QgsLayoutItemMap::shouldDrawPart( QgsLayoutItemMap::PartType part ) const
2732{
2733 if ( mCurrentExportPart == NotLayered )
2734 {
2735 //all parts of the map are visible
2736 return true;
2737 }
2738
2739 switch ( part )
2740 {
2741 case NotLayered:
2742 return true;
2743
2744 case Start:
2745 return false;
2746
2747 case Background:
2748 return mCurrentExportPart == Background && hasBackground();
2749
2750 case Layer:
2751 return mCurrentExportPart == Layer;
2752
2753 case Grid:
2754 return mCurrentExportPart == Grid && mGridStack->hasEnabledItems();
2755
2756 case OverviewMapExtent:
2757 return mCurrentExportPart == OverviewMapExtent && mOverviewStack->hasEnabledItems();
2758
2759 case Frame:
2760 return mCurrentExportPart == Frame && frameEnabled();
2761
2762 case SelectionBoxes:
2763 return mCurrentExportPart == SelectionBoxes && isSelected();
2764
2765 case End:
2766 return false;
2767 }
2768
2769 return false;
2770}
2771
2772void QgsLayoutItemMap::refreshMapExtents( const QgsExpressionContext *context )
2773{
2774 QgsExpressionContext scopedContext;
2775 if ( !context )
2776 scopedContext = createExpressionContext();
2777
2778 bool ok = false;
2779 const QgsExpressionContext *evalContext = context ? context : &scopedContext;
2780
2781
2782 //data defined map extents set?
2783 QgsRectangle newExtent = extent();
2784 bool useDdXMin = false;
2785 bool useDdXMax = false;
2786 bool useDdYMin = false;
2787 bool useDdYMax = false;
2788 double minXD = 0;
2789 double minYD = 0;
2790 double maxXD = 0;
2791 double maxYD = 0;
2792
2794 if ( ok )
2795 {
2796 useDdXMin = true;
2797 newExtent.setXMinimum( minXD );
2798 }
2800 if ( ok )
2801 {
2802 useDdYMin = true;
2803 newExtent.setYMinimum( minYD );
2804 }
2806 if ( ok )
2807 {
2808 useDdXMax = true;
2809 newExtent.setXMaximum( maxXD );
2810 }
2812 if ( ok )
2813 {
2814 useDdYMax = true;
2815 newExtent.setYMaximum( maxYD );
2816 }
2817
2818 if ( newExtent != mExtent )
2819 {
2820 //calculate new extents to fit data defined extents
2821
2822 //Make sure the width/height ratio is the same as in current map extent.
2823 //This is to keep the map item frame and the page layout fixed
2824 double currentWidthHeightRatio = mExtent.width() / mExtent.height();
2825 double newWidthHeightRatio = newExtent.width() / newExtent.height();
2826
2827 if ( currentWidthHeightRatio < newWidthHeightRatio )
2828 {
2829 //enlarge height of new extent, ensuring the map center stays the same
2830 double newHeight = newExtent.width() / currentWidthHeightRatio;
2831 double deltaHeight = newHeight - newExtent.height();
2832 newExtent.setYMinimum( newExtent.yMinimum() - deltaHeight / 2 );
2833 newExtent.setYMaximum( newExtent.yMaximum() + deltaHeight / 2 );
2834 }
2835 else
2836 {
2837 //enlarge width of new extent, ensuring the map center stays the same
2838 double newWidth = currentWidthHeightRatio * newExtent.height();
2839 double deltaWidth = newWidth - newExtent.width();
2840 newExtent.setXMinimum( newExtent.xMinimum() - deltaWidth / 2 );
2841 newExtent.setXMaximum( newExtent.xMaximum() + deltaWidth / 2 );
2842 }
2843
2844 mExtent = newExtent;
2845 }
2846
2847 //now refresh scale, as this potentially overrides extents
2848
2849 //data defined map scale set?
2851 if ( ok )
2852 {
2853 setScale( scaleD, false );
2854 newExtent = mExtent;
2855 }
2856
2857 if ( useDdXMax || useDdXMin || useDdYMax || useDdYMin )
2858 {
2859 //if only one of min/max was set for either x or y, then make sure our extent is locked on that value
2860 //as we can do this without altering the scale
2861 if ( useDdXMin && !useDdXMax )
2862 {
2863 double xMax = mExtent.xMaximum() - ( mExtent.xMinimum() - minXD );
2864 newExtent.setXMinimum( minXD );
2865 newExtent.setXMaximum( xMax );
2866 }
2867 else if ( !useDdXMin && useDdXMax )
2868 {
2869 double xMin = mExtent.xMinimum() - ( mExtent.xMaximum() - maxXD );
2870 newExtent.setXMinimum( xMin );
2871 newExtent.setXMaximum( maxXD );
2872 }
2873 if ( useDdYMin && !useDdYMax )
2874 {
2875 double yMax = mExtent.yMaximum() - ( mExtent.yMinimum() - minYD );
2876 newExtent.setYMinimum( minYD );
2877 newExtent.setYMaximum( yMax );
2878 }
2879 else if ( !useDdYMin && useDdYMax )
2880 {
2881 double yMin = mExtent.yMinimum() - ( mExtent.yMaximum() - maxYD );
2882 newExtent.setYMinimum( yMin );
2883 newExtent.setYMaximum( maxYD );
2884 }
2885
2886 if ( newExtent != mExtent )
2887 {
2888 mExtent = newExtent;
2889 }
2890 }
2891
2892 //lastly, map rotation overrides all
2893 double mapRotation = mMapRotation;
2894
2895 //data defined map rotation set?
2897
2898 if ( !qgsDoubleNear( mEvaluatedMapRotation, mapRotation ) )
2899 {
2900 mEvaluatedMapRotation = mapRotation;
2902 }
2903}
2904
2905void QgsLayoutItemMap::refreshLabelMargin( bool updateItem )
2906{
2907 //data defined label margin set?
2909 mEvaluatedLabelMargin.setLength( labelMargin );
2910 mEvaluatedLabelMargin.setUnits( mLabelMargin.units() );
2911
2912 if ( updateItem )
2913 {
2914 update();
2915 }
2916}
2917
2918void QgsLayoutItemMap::updateAtlasFeature()
2919{
2920 if ( !atlasDriven() || !mLayout->reportContext().layer() )
2921 return; // nothing to do
2922
2923 QgsRectangle bounds = computeAtlasRectangle();
2924 if ( bounds.isNull() )
2925 return;
2926
2927 double xa1 = bounds.xMinimum();
2928 double xa2 = bounds.xMaximum();
2929 double ya1 = bounds.yMinimum();
2930 double ya2 = bounds.yMaximum();
2931 QgsRectangle newExtent = bounds;
2932 QgsRectangle originalExtent = mExtent;
2933
2934 //sanity check - only allow fixed scale mode for point layers
2935 bool isPointLayer = QgsWkbTypes::geometryType( mLayout->reportContext().layer()->wkbType() ) == Qgis::GeometryType::Point;
2936
2937 if ( mAtlasScalingMode == Fixed || mAtlasScalingMode == Predefined || isPointLayer )
2938 {
2939 QgsScaleCalculator calc;
2940 calc.setMapUnits( crs().mapUnits() );
2941 calc.setDpi( 25.4 );
2942 if ( QgsProject *project = mLayout->project() )
2943 {
2944 calc.setMethod( project->scaleMethod() );
2945 }
2946 double originalScale = calc.calculate( originalExtent, rect().width() );
2947 double geomCenterX = ( xa1 + xa2 ) / 2.0;
2948 double geomCenterY = ( ya1 + ya2 ) / 2.0;
2949 QVector<qreal> scales;
2951 if ( !mLayout->reportContext().predefinedScales().empty() ) // remove when deprecated method is removed
2952 scales = mLayout->reportContext().predefinedScales();
2953 else
2954 scales = mLayout->renderContext().predefinedScales();
2956 if ( mAtlasScalingMode == Fixed || scales.isEmpty() || ( isPointLayer && mAtlasScalingMode != Predefined ) )
2957 {
2958 // only translate, keep the original scale (i.e. width x height)
2959 double xMin = geomCenterX - originalExtent.width() / 2.0;
2960 double yMin = geomCenterY - originalExtent.height() / 2.0;
2961 newExtent = QgsRectangle( xMin,
2962 yMin,
2963 xMin + originalExtent.width(),
2964 yMin + originalExtent.height() );
2965
2966 //scale newExtent to match original scale of map
2967 //this is required for geographic coordinate systems, where the scale varies by extent
2968 double newScale = calc.calculate( newExtent, rect().width() );
2969 newExtent.scale( originalScale / newScale );
2970 }
2971 else if ( mAtlasScalingMode == Predefined && !qgsDoubleNear( originalScale, 0 ) )
2972 {
2973 // choose one of the predefined scales
2974 double newWidth = originalExtent.width();
2975 double newHeight = originalExtent.height();
2976 for ( int i = 0; i < scales.size(); i++ )
2977 {
2978 double ratio = scales[i] / originalScale;
2979 newWidth = originalExtent.width() * ratio;
2980 newHeight = originalExtent.height() * ratio;
2981
2982 // compute new extent, centered on feature
2983 double xMin = geomCenterX - newWidth / 2.0;
2984 double yMin = geomCenterY - newHeight / 2.0;
2985 newExtent = QgsRectangle( xMin,
2986 yMin,
2987 xMin + newWidth,
2988 yMin + newHeight );
2989
2990 //scale newExtent to match desired map scale
2991 //this is required for geographic coordinate systems, where the scale varies by extent
2992 const double newScale = calc.calculate( newExtent, rect().width() );
2993 if ( qgsDoubleNear( newScale, 0 ) )
2994 continue;
2995
2996 newExtent.scale( scales[i] / newScale );
2997
2998 if ( ( newExtent.width() >= bounds.width() ) && ( newExtent.height() >= bounds.height() ) )
2999 {
3000 // this is the smallest extent that embeds the feature, stop here
3001 break;
3002 }
3003 }
3004 }
3005 }
3006 else if ( mAtlasScalingMode == Auto )
3007 {
3008 // auto scale
3009
3010 double geomRatio = bounds.width() / bounds.height();
3011 double mapRatio = originalExtent.width() / originalExtent.height();
3012
3013 // geometry height is too big
3014 if ( geomRatio < mapRatio )
3015 {
3016 // extent the bbox's width
3017 double adjWidth = ( mapRatio * bounds.height() - bounds.width() ) / 2.0;
3018 xa1 -= adjWidth;
3019 xa2 += adjWidth;
3020 }
3021 // geometry width is too big
3022 else if ( geomRatio > mapRatio )
3023 {
3024 // extent the bbox's height
3025 double adjHeight = ( bounds.width() / mapRatio - bounds.height() ) / 2.0;
3026 ya1 -= adjHeight;
3027 ya2 += adjHeight;
3028 }
3029 newExtent = QgsRectangle( xa1, ya1, xa2, ya2 );
3030
3031 const double evaluatedAtlasMargin = atlasMargin();
3032 if ( evaluatedAtlasMargin > 0.0 )
3033 {
3034 newExtent.scale( 1 + evaluatedAtlasMargin );
3035 }
3036 }
3037
3038 // set the new extent (and render)
3039 setExtent( newExtent );
3040 emit preparedForAtlas();
3041}
3042
3043QgsRectangle QgsLayoutItemMap::computeAtlasRectangle()
3044{
3045 // QgsGeometry::boundingBox is expressed in the geometry"s native CRS
3046 // We have to transform the geometry to the destination CRS and ask for the bounding box
3047 // Note: we cannot directly take the transformation of the bounding box, since transformations are not linear
3048 QgsGeometry g = mLayout->reportContext().currentGeometry( crs() );
3049 // Rotating the geometry, so the bounding box is correct wrt map rotation
3050 if ( !g.boundingBox().isEmpty() && mEvaluatedMapRotation != 0.0 )
3051 {
3052 QgsPointXY prevCenter = g.boundingBox().center();
3053 g.rotate( mEvaluatedMapRotation, g.boundingBox().center() );
3054 // Rotation center will be still the bounding box center of an unrotated geometry.
3055 // Which means, if the center of bbox moves after rotation, the viewport will
3056 // also be offset, and part of the geometry will fall out of bounds.
3057 // Here we compensate for that roughly: by extending the rotated bounds
3058 // so that its center is the same as the original.
3059 QgsRectangle bounds = g.boundingBox();
3060 double dx = std::max( std::abs( prevCenter.x() - bounds.xMinimum() ),
3061 std::abs( prevCenter.x() - bounds.xMaximum() ) );
3062 double dy = std::max( std::abs( prevCenter.y() - bounds.yMinimum() ),
3063 std::abs( prevCenter.y() - bounds.yMaximum() ) );
3064 QgsPointXY center = g.boundingBox().center();
3065 return QgsRectangle( center.x() - dx, center.y() - dy,
3066 center.x() + dx, center.y() + dy );
3067 }
3068 else
3069 {
3070 return g.boundingBox();
3071 }
3072}
3073
3074void QgsLayoutItemMap::createStagedRenderJob( const QgsRectangle &extent, const QSizeF size, double dpi )
3075{
3076 QgsMapSettings settings = mapSettings( extent, size, dpi, true );
3077 settings.setLayers( mOverviewStack->modifyMapLayerList( settings.layers() ) );
3078
3079 mStagedRendererJob = std::make_unique< QgsMapRendererStagedRenderJob >( settings,
3083 mStagedRendererJob->start();
3084}
3085
3086
3087
3088//
3089// QgsLayoutItemMapAtlasClippingSettings
3090//
3091
3093 : QObject( map )
3094 , mMap( map )
3095{
3096 if ( mMap->layout() && mMap->layout()->project() )
3097 {
3098 connect( mMap->layout()->project(), static_cast < void ( QgsProject::* )( const QList<QgsMapLayer *>& layers ) > ( &QgsProject::layersWillBeRemoved ),
3099 this, &QgsLayoutItemMapAtlasClippingSettings::layersAboutToBeRemoved );
3100 }
3101}
3102
3104{
3105 return mClipToAtlasFeature;
3106}
3107
3109{
3110 if ( enabled == mClipToAtlasFeature )
3111 return;
3112
3113 mClipToAtlasFeature = enabled;
3114 emit changed();
3115}
3116
3121
3123{
3124 if ( mFeatureClippingType == type )
3125 return;
3126
3127 mFeatureClippingType = type;
3128 emit changed();
3129}
3130
3132{
3133 return mForceLabelsInsideFeature;
3134}
3135
3137{
3138 if ( forceInside == mForceLabelsInsideFeature )
3139 return;
3140
3141 mForceLabelsInsideFeature = forceInside;
3142 emit changed();
3143}
3144
3146{
3147 return mRestrictToLayers;
3148}
3149
3151{
3152 if ( mRestrictToLayers == enabled )
3153 return;
3154
3155 mRestrictToLayers = enabled;
3156 emit changed();
3157}
3158
3160{
3161 return _qgis_listRefToRaw( mLayersToClip );
3162}
3163
3164void QgsLayoutItemMapAtlasClippingSettings::setLayersToClip( const QList< QgsMapLayer * > &layersToClip )
3165{
3166 mLayersToClip = _qgis_listRawToRef( layersToClip );
3167 emit changed();
3168}
3169
3170bool QgsLayoutItemMapAtlasClippingSettings::writeXml( QDomElement &element, QDomDocument &document, const QgsReadWriteContext & ) const
3171{
3172 QDomElement settingsElem = document.createElement( QStringLiteral( "atlasClippingSettings" ) );
3173 settingsElem.setAttribute( QStringLiteral( "enabled" ), mClipToAtlasFeature ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
3174 settingsElem.setAttribute( QStringLiteral( "forceLabelsInside" ), mForceLabelsInsideFeature ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
3175 settingsElem.setAttribute( QStringLiteral( "clippingType" ), QString::number( static_cast<int>( mFeatureClippingType ) ) );
3176 settingsElem.setAttribute( QStringLiteral( "restrictLayers" ), mRestrictToLayers ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
3177
3178 //layer set
3179 QDomElement layerSetElem = document.createElement( QStringLiteral( "layersToClip" ) );
3180 for ( const QgsMapLayerRef &layerRef : mLayersToClip )
3181 {
3182 if ( !layerRef )
3183 continue;
3184 QDomElement layerElem = document.createElement( QStringLiteral( "Layer" ) );
3185 QDomText layerIdText = document.createTextNode( layerRef.layerId );
3186 layerElem.appendChild( layerIdText );
3187
3188 layerElem.setAttribute( QStringLiteral( "name" ), layerRef.name );
3189 layerElem.setAttribute( QStringLiteral( "source" ), layerRef.source );
3190 layerElem.setAttribute( QStringLiteral( "provider" ), layerRef.provider );
3191
3192 layerSetElem.appendChild( layerElem );
3193 }
3194 settingsElem.appendChild( layerSetElem );
3195
3196 element.appendChild( settingsElem );
3197 return true;
3198}
3199
3200bool QgsLayoutItemMapAtlasClippingSettings::readXml( const QDomElement &element, const QDomDocument &, const QgsReadWriteContext & )
3201{
3202 const QDomElement settingsElem = element.firstChildElement( QStringLiteral( "atlasClippingSettings" ) );
3203
3204 mClipToAtlasFeature = settingsElem.attribute( QStringLiteral( "enabled" ), QStringLiteral( "0" ) ).toInt();
3205 mForceLabelsInsideFeature = settingsElem.attribute( QStringLiteral( "forceLabelsInside" ), QStringLiteral( "0" ) ).toInt();
3206 mFeatureClippingType = static_cast< QgsMapClippingRegion::FeatureClippingType >( settingsElem.attribute( QStringLiteral( "clippingType" ), QStringLiteral( "0" ) ).toInt() );
3207 mRestrictToLayers = settingsElem.attribute( QStringLiteral( "restrictLayers" ), QStringLiteral( "0" ) ).toInt();
3208
3209 mLayersToClip.clear();
3210 QDomNodeList layerSetNodeList = settingsElem.elementsByTagName( QStringLiteral( "layersToClip" ) );
3211 if ( !layerSetNodeList.isEmpty() )
3212 {
3213 QDomElement layerSetElem = layerSetNodeList.at( 0 ).toElement();
3214 QDomNodeList layerIdNodeList = layerSetElem.elementsByTagName( QStringLiteral( "Layer" ) );
3215 mLayersToClip.reserve( layerIdNodeList.size() );
3216 for ( int i = 0; i < layerIdNodeList.size(); ++i )
3217 {
3218 QDomElement layerElem = layerIdNodeList.at( i ).toElement();
3219 QString layerId = layerElem.text();
3220 QString layerName = layerElem.attribute( QStringLiteral( "name" ) );
3221 QString layerSource = layerElem.attribute( QStringLiteral( "source" ) );
3222 QString layerProvider = layerElem.attribute( QStringLiteral( "provider" ) );
3223
3224 QgsMapLayerRef ref( layerId, layerName, layerSource, layerProvider );
3225 if ( mMap->layout() && mMap->layout()->project() )
3226 ref.resolveWeakly( mMap->layout()->project() );
3227 mLayersToClip << ref;
3228 }
3229 }
3230
3231 return true;
3232}
3233
3234void QgsLayoutItemMapAtlasClippingSettings::layersAboutToBeRemoved( const QList<QgsMapLayer *> &layers )
3235{
3236 if ( !mLayersToClip.isEmpty() )
3237 {
3238 _qgis_removeLayers( mLayersToClip, layers );
3239 }
3240}
3241
3242//
3243// QgsLayoutItemMapItemClipPathSettings
3244//
3250
3252{
3253 return mEnabled && mClipPathSource;
3254}
3255
3257{
3258 return mEnabled;
3259}
3260
3262{
3263 if ( enabled == mEnabled )
3264 return;
3265
3266 mEnabled = enabled;
3267
3268 if ( mClipPathSource )
3269 {
3270 // may need to refresh the clip source in order to get it to render/not render depending on enabled state
3271 mClipPathSource->refresh();
3272 }
3273 emit changed();
3274}
3275
3277{
3278 if ( isActive() )
3279 {
3280 QgsGeometry clipGeom( mClipPathSource->clipPath() );
3281 clipGeom.transform( mMap->layoutToMapCoordsTransform() );
3282 return clipGeom;
3283 }
3284 return QgsGeometry();
3285}
3286
3288{
3289 if ( isActive() )
3290 {
3291 QgsGeometry clipGeom( mClipPathSource->clipPath() );
3292 clipGeom.transform( mMap->sceneTransform().inverted() );
3293 return clipGeom;
3294 }
3295 return QgsGeometry();
3296}
3297
3299{
3301 region.setFeatureClip( mFeatureClippingType );
3302 return region;
3303}
3304
3306{
3307 if ( mClipPathSource == item )
3308 return;
3309
3310 if ( mClipPathSource )
3311 {
3312 disconnect( mClipPathSource, &QgsLayoutItem::clipPathChanged, mMap, &QgsLayoutItemMap::refresh );
3313 disconnect( mClipPathSource, &QgsLayoutItem::rotationChanged, mMap, &QgsLayoutItemMap::refresh );
3314 disconnect( mClipPathSource, &QgsLayoutItem::clipPathChanged, mMap, &QgsLayoutItemMap::extentChanged );
3315 disconnect( mClipPathSource, &QgsLayoutItem::rotationChanged, mMap, &QgsLayoutItemMap::extentChanged );
3316 }
3317
3318 QgsLayoutItem *oldItem = mClipPathSource;
3319 mClipPathSource = item;
3320
3321 if ( mClipPathSource )
3322 {
3323 // if item size or rotation changes, we need to redraw this map
3324 connect( mClipPathSource, &QgsLayoutItem::clipPathChanged, mMap, &QgsLayoutItemMap::refresh );
3325 connect( mClipPathSource, &QgsLayoutItem::rotationChanged, mMap, &QgsLayoutItemMap::refresh );
3326 // and if clip item size or rotation changes, then effectively we've changed the visible extent of the map
3327 connect( mClipPathSource, &QgsLayoutItem::clipPathChanged, mMap, &QgsLayoutItemMap::extentChanged );
3328 connect( mClipPathSource, &QgsLayoutItem::rotationChanged, mMap, &QgsLayoutItemMap::extentChanged );
3329 // trigger a redraw of the clip source, so that it becomes invisible
3330 mClipPathSource->refresh();
3331 }
3332
3333 if ( oldItem )
3334 {
3335 // may need to refresh the previous item in order to get it to render
3336 oldItem->refresh();
3337 }
3338
3339 emit changed();
3340}
3341
3343{
3344 return mClipPathSource;
3345}
3346
3351
3353{
3354 if ( mFeatureClippingType == type )
3355 return;
3356
3357 mFeatureClippingType = type;
3358 emit changed();
3359}
3360
3362{
3363 return mForceLabelsInsideClipPath;
3364}
3365
3367{
3368 if ( forceInside == mForceLabelsInsideClipPath )
3369 return;
3370
3371 mForceLabelsInsideClipPath = forceInside;
3372 emit changed();
3373}
3374
3375bool QgsLayoutItemMapItemClipPathSettings::writeXml( QDomElement &element, QDomDocument &document, const QgsReadWriteContext & ) const
3376{
3377 QDomElement settingsElem = document.createElement( QStringLiteral( "itemClippingSettings" ) );
3378 settingsElem.setAttribute( QStringLiteral( "enabled" ), mEnabled ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
3379 settingsElem.setAttribute( QStringLiteral( "forceLabelsInside" ), mForceLabelsInsideClipPath ? QStringLiteral( "1" ) : QStringLiteral( "0" ) );
3380 settingsElem.setAttribute( QStringLiteral( "clippingType" ), QString::number( static_cast<int>( mFeatureClippingType ) ) );
3381 if ( mClipPathSource )
3382 settingsElem.setAttribute( QStringLiteral( "clipSource" ), mClipPathSource->uuid() );
3383 else
3384 settingsElem.setAttribute( QStringLiteral( "clipSource" ), QString() );
3385
3386 element.appendChild( settingsElem );
3387 return true;
3388}
3389
3390bool QgsLayoutItemMapItemClipPathSettings::readXml( const QDomElement &element, const QDomDocument &, const QgsReadWriteContext & )
3391{
3392 const QDomElement settingsElem = element.firstChildElement( QStringLiteral( "itemClippingSettings" ) );
3393
3394 mEnabled = settingsElem.attribute( QStringLiteral( "enabled" ), QStringLiteral( "0" ) ).toInt();
3395 mForceLabelsInsideClipPath = settingsElem.attribute( QStringLiteral( "forceLabelsInside" ), QStringLiteral( "0" ) ).toInt();
3396 mFeatureClippingType = static_cast< QgsMapClippingRegion::FeatureClippingType >( settingsElem.attribute( QStringLiteral( "clippingType" ), QStringLiteral( "0" ) ).toInt() );
3397 mClipPathUuid = settingsElem.attribute( QStringLiteral( "clipSource" ) );
3398
3399 return true;
3400}
3401
3403{
3404 if ( !mClipPathUuid.isEmpty() )
3405 {
3406 if ( QgsLayoutItem *item = mMap->layout()->itemByUuid( mClipPathUuid, true ) )
3407 {
3408 setSourceItem( item );
3409 }
3410 }
3411}
QFlags< VectorRenderingSimplificationFlag > VectorRenderingSimplificationFlags
Simplification flags for vector feature rendering.
Definition qgis.h:2938
@ Millimeters
Millimeters.
@ NoSimplification
No simplification can be applied.
@ CollectUnplacedLabels
Whether unplaced labels should be collected in the labeling results (regardless of whether they are b...
@ DrawUnplacedLabels
Whether to render unplaced labels as an indicator/warning for users.
@ UsePartialCandidates
Whether to use also label candidates that are partially outside of the map view.
@ Export
Renderer used for printing or exporting to a file.
@ View
Renderer used for displaying on screen.
@ Preferred
Preferred format, matching the most recent WKT ISO standard. Currently an alias to WKT2_2019,...
@ DrawEditingInfo
Enable drawing of vertex markers for layers in editing mode.
@ UseRenderingOptimization
Enable vector simplification and other rendering optimizations.
@ ForceVectorOutput
Vector graphics should not be cached and drawn as raster images.
@ RenderPartialOutput
Whether to make extra effort to update map image with partially rendered layers (better for interacti...
@ ForceRasterMasks
Force symbol masking to be applied using a raster method. This is considerably faster when compared t...
@ LosslessImageRendering
Render images losslessly whenever possible, instead of the default lossy jpeg rendering used for some...
@ AlwaysUseGlobalMasks
When applying clipping paths for selective masking, always use global ("entire map") paths,...
@ DrawSelection
Whether vector selections should be shown in the rendered map.
@ Antialiasing
Enable anti-aliasing for map rendering.
@ UseAdvancedEffects
Enable layer opacity and blending effects.
@ HighQualityImageTransforms
Enable high quality image transformations, which results in better appearance of scaled or rotated ra...
virtual QPainterPath asQPainterPath() const =0
Returns the geometry represented as a QPainterPath.
QDateTime valueAsDateTime(int key, const QgsExpressionContext &context, const QDateTime &defaultDateTime=QDateTime(), bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a datetime.
double valueAsDouble(int key, const QgsExpressionContext &context, double defaultValue=0.0, bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a double.
QString valueAsString(int key, const QgsExpressionContext &context, const QString &defaultString=QString(), bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a string.
Abstract base class for annotation items which are drawn over a map.
bool hasFixedMapPosition
QgsCoordinateReferenceSystem mapPositionCrs() const
Returns the CRS of the map position, or an invalid CRS if the annotation does not have a fixed map po...
QgsPointXY mapPosition
void render(QgsRenderContext &context) const
Renders the annotation to a target render context.
bool isVisible() const
Returns true if the annotation is visible and should be rendered.
QPointF relativePosition() const
Returns the relative position of the annotation, if it is not attached to a fixed map position.
static QIcon getThemeIcon(const QString &name, const QColor &fillColor=QColor(), const QColor &strokeColor=QColor())
Helper to get a theme icon.
static QgsCoordinateReferenceSystemRegistry * coordinateReferenceSystemRegistry()
Returns the application's coordinate reference system (CRS) registry, which handles known CRS definit...
void userCrsChanged(const QString &id)
Emitted whenever an existing user CRS definition is changed.
Represents a coordinate reference system (CRS).
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
QString toProj() const
Returns a Proj string representation of this CRS.
bool readXml(const QDomNode &node)
Restores state from the given DOM node.
QString ellipsoidAcronym() const
Returns the ellipsoid acronym for the ellipsoid used by the CRS.
QString projectionAcronym() const
Returns the projection acronym for the projection used by the CRS.
void updateDefinition()
Updates the definition and parameters of the coordinate reference system to their latest values.
QString toWkt(Qgis::CrsWktVariant variant=Qgis::CrsWktVariant::Wkt1Gdal, bool multiline=false, int indentationWidth=4) const
Returns a WKT representation of this CRS.
bool writeXml(QDomNode &node, QDomDocument &doc) const
Stores state to the given Dom node in the given document.
QgsProjOperation operation() const
Returns information about the PROJ operation associated with the coordinate reference system,...
Handles coordinate transforms between two coordinate systems.
Custom exception class for Coordinate Reference System related exceptions.
QgsRange which stores a range of double values.
Definition qgsrange.h:233
Single scope for storing variables and functions for use within a QgsExpressionContext.
void addFunction(const QString &name, QgsScopedExpressionFunction *function)
Adds a function to the scope.
void addVariable(const QgsExpressionContextScope::StaticVariable &variable)
Adds a variable into the context scope.
void setVariable(const QString &name, const QVariant &value, bool isStatic=false)
Convenience method for setting a variable in the context scope by name name and value.
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.
A geometry is the spatial representation of a feature.
QPolygonF asQPolygonF() const
Returns contents of the geometry as a QPolygonF.
static QgsGeometry fromRect(const QgsRectangle &rect)
Creates a new geometry from a QgsRectangle.
static QgsGeometry fromQPolygonF(const QPolygonF &polygon)
Construct geometry from a QPolygonF.
Qgis::GeometryOperationResult transform(const QgsCoordinateTransform &ct, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward, bool transformZ=false)
Transforms this geometry as described by the coordinate transform ct.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
static QgsGeometry fromPointXY(const QgsPointXY &point)
Creates a new geometry from a QgsPointXY object.
QgsGeometry intersection(const QgsGeometry &geometry, const QgsGeometryParameters &parameters=QgsGeometryParameters()) const
Returns a geometry representing the points shared by this geometry and other.
QgsGeometry buffer(double distance, int segments) const
Returns a buffer region around this geometry having the given width and with a specified number of se...
bool isEmpty() const
Returns true if the geometry is empty (eg a linestring with no vertices, or a collection with no geom...
QgsRectangle boundingBox() const
Returns the bounding box of the geometry.
Qgis::GeometryOperationResult rotate(double rotation, const QgsPointXY &center)
Rotate this geometry around the Z axis.
A map layer which consists of a set of child layers, where all component layers are rendered as a sin...
QList< QgsMapLayer * > childLayers() const
Returns the child layers contained by the group.
void setChildLayers(const QList< QgsMapLayer * > &layers)
Sets the child layers contained by the group.
A representation of the interval between two datetime values.
Definition qgsinterval.h:46
Label blocking region (in map coordinates and CRS).
Stores global configuration for labeling engine.
void setFlag(Qgis::LabelingFlag f, bool enabled=true)
Sets whether a particual flag is enabled.
Stores computed placement from labeling engine.
void layerOrderChanged()
Emitted when the layer order has changed.
Contains settings relating to clipping a layout map by the current atlas feature.
void setFeatureClippingType(QgsMapClippingRegion::FeatureClippingType type)
Sets the feature clipping type to apply when clipping to the current atlas feature.
bool restrictToLayers() const
Returns true if clipping should be restricted to a subset of layers.
QgsLayoutItemMapAtlasClippingSettings(QgsLayoutItemMap *map=nullptr)
Constructor for QgsLayoutItemMapAtlasClippingSettings, with the specified map parent.
bool readXml(const QDomElement &element, const QDomDocument &doc, const QgsReadWriteContext &context)
Sets the setting's state from a DOM document, where element is the DOM node corresponding to a 'Layou...
bool writeXml(QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context) const
Stores settings in a DOM element, where element is the DOM element corresponding to a 'LayoutMap' tag...
void setRestrictToLayers(bool enabled)
Sets whether clipping should be restricted to a subset of layers.
void setLayersToClip(const QList< QgsMapLayer * > &layers)
Sets the list of map layers to clip to the atlas feature.
QList< QgsMapLayer * > layersToClip() const
Returns the list of map layers to clip to the atlas feature.
void setEnabled(bool enabled)
Sets whether the map content should be clipped to the current atlas feature.
void changed()
Emitted when the atlas clipping settings are changed.
bool forceLabelsInsideFeature() const
Returns true if labels should only be placed inside the atlas feature geometry.
bool enabled() const
Returns true if the map content should be clipped to the current atlas feature.
void setForceLabelsInsideFeature(bool forceInside)
Sets whether labels should only be placed inside the atlas feature geometry.
QgsMapClippingRegion::FeatureClippingType featureClippingType() const
Returns the feature clipping type to apply when clipping to the current atlas feature.
An individual grid which is drawn above the map content in a QgsLayoutItemMap.
Contains settings relating to clipping a layout map by another layout item.
bool writeXml(QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context) const
Stores settings in a DOM element, where element is the DOM element corresponding to a 'LayoutMap' tag...
void setForceLabelsInsideClipPath(bool forceInside)
Sets whether labels should only be placed inside the clip path geometry.
void setSourceItem(QgsLayoutItem *item)
Sets the source item which will provide the clipping path for the map.
QgsLayoutItemMapItemClipPathSettings(QgsLayoutItemMap *map=nullptr)
Constructor for QgsLayoutItemMapItemClipPathSettings, with the specified map parent.
bool readXml(const QDomElement &element, const QDomDocument &doc, const QgsReadWriteContext &context)
Sets the setting's state from a DOM document, where element is the DOM node corresponding to a 'Layou...
QgsGeometry clipPathInMapItemCoordinates() const
Returns the clipping path geometry, in the map item's coordinate space.
QgsGeometry clippedMapExtent() const
Returns the geometry to use for clipping the parent map, in the map item's CRS.
QgsLayoutItem * sourceItem()
Returns the source item which will provide the clipping path for the map, or nullptr if no item is se...
void setEnabled(bool enabled)
Sets whether the map content should be clipped to the associated item.
bool forceLabelsInsideClipPath() const
Returns true if labels should only be placed inside the clip path geometry.
void finalizeRestoreFromXml()
To be called after all pending items have been restored from XML.
QgsMapClippingRegion::FeatureClippingType featureClippingType() const
Returns the feature clipping type to apply when clipping to the associated item.
bool enabled() const
Returns true if the map content should be clipped to the associated item.
QgsMapClippingRegion toMapClippingRegion() const
Returns the clip path as a map clipping region.
void changed()
Emitted when the item clipping settings are changed.
void setFeatureClippingType(QgsMapClippingRegion::FeatureClippingType type)
Sets the feature clipping type to apply when clipping to the associated item.
bool isActive() const
Returns true if the item clipping is enabled and set to a valid source item.
An item which is drawn inside a QgsLayoutItemMap, e.g., a grid or map overview.
@ StackAboveMapLabels
Render above all map layers and labels.
StackingPosition stackingPosition() const
Returns the item's stacking position, which specifies where the in the map's stack the item should be...
bool enabled() const
Returns whether the item will be drawn.
An individual overview which is drawn above the map content in a QgsLayoutItemMap,...
Layout graphical items for displaying a map.
void setFollowVisibilityPreset(bool follow)
Sets whether the map should follow a map theme.
bool nextExportPart() override
Moves to the next export part for a multi-layered export item, during a multi-layered export.
void removeRenderedFeatureHandler(QgsRenderedFeatureHandlerInterface *handler)
Removes a previously added rendered feature handler.
void extentChanged()
Emitted when the map's extent changes.
QPointF mapToItemCoords(QPointF mapCoords) const
Transforms map coordinates to item coordinates (considering rotation and move offset)
bool accept(QgsStyleEntityVisitorInterface *visitor) const override
Accepts the specified style entity visitor, causing it to visit all style entities associated with th...
~QgsLayoutItemMap() override
QIcon icon() const override
Returns the item's icon.
void refreshDataDefinedProperty(QgsLayoutObject::DataDefinedProperty property=QgsLayoutObject::DataDefinedProperty::AllProperties) override
void preparedForAtlas()
Emitted when the map has been prepared for atlas rendering, just before actual rendering.
void setFollowVisibilityPresetName(const QString &name)
Sets preset name for map rendering.
QTransform layoutToMapCoordsTransform() const
Creates a transform from layout coordinates to map coordinates.
bool writePropertiesToElement(QDomElement &element, QDomDocument &document, const QgsReadWriteContext &context) const override
Stores item state within an XML DOM element.
QgsMapSettings mapSettings(const QgsRectangle &extent, QSizeF size, double dpi, bool includeLayerSettings) const
Returns map settings that will be used for drawing of the map.
bool isLabelBlockingItem(QgsLayoutItem *item) const
Returns true if the specified item is a "label blocking item".
void storeCurrentLayerStyles()
Stores the current project layer styles into style overrides.
void setAtlasDriven(bool enabled)
Sets whether the map extent will follow the current atlas feature.
QgsLayoutMeasurement labelMargin() const
Returns the margin from the map edges in which no labels may be placed.
AtlasScalingMode
Scaling modes used for the serial rendering (atlas)
@ Predefined
A scale is chosen from the predefined scales.
@ Auto
The extent is adjusted so that each feature is fully visible.
@ Fixed
The current scale of the map is used for each feature of the atlas.
bool requiresRasterization() const override
Returns true if the item is drawn in such a way that forces the whole layout to be rasterized when ex...
Q_DECL_DEPRECATED int numberExportLayers() const override
void layerStyleOverridesChanged()
Emitted when layer style overrides are changed... a means to let associated legend items know they sh...
void updateBoundingRect()
Updates the bounding rect of this item. Call this function before doing any changes related to annota...
void moveContent(double dx, double dy) override
Moves the content of the item, by a specified dx and dy in layout units.
void setZRangeEnabled(bool enabled)
Sets whether the z range is enabled (i.e.
QgsLayoutItemMapGrid * grid()
Returns the map item's first grid.
void mapRotationChanged(double newRotation)
Emitted when the map's rotation changes.
int type() const override
void previewRefreshed()
Emitted whenever the item's map preview has been refreshed.
friend class QgsLayoutItemMapOverview
void setExtent(const QgsRectangle &extent)
Sets a new extent for the map.
void paint(QPainter *painter, const QStyleOptionGraphicsItem *itemStyle, QWidget *pWidget) override
QFlags< MapItemFlag > MapItemFlags
void draw(QgsLayoutItemRenderContext &context) override
Draws the item's contents using the specified item render context.
QPolygonF visibleExtentPolygon() const
Returns a polygon representing the current visible map extent, considering map extents and rotation.
QgsRectangle requestedExtent() const
Calculates the extent to request and the yShift of the top-left point in case of rotation.
void setMapFlags(QgsLayoutItemMap::MapItemFlags flags)
Sets the map item's flags, which control how the map content is drawn.
void zoomContent(double factor, QPointF point) override
Zooms content of item.
QList< QgsMapLayer * > layersToRender(const QgsExpressionContext *context=nullptr) const
Returns a list of the layers which will be rendered within this map item, considering any locked laye...
void crsChanged()
Emitted when the map's coordinate reference system is changed.
void setLayers(const QList< QgsMapLayer * > &layers)
Sets the stored layers set.
QPolygonF transformedMapPolygon() const
Returns extent that considers rotation and shift with mOffsetX / mOffsetY.
static QgsLayoutItemMap * create(QgsLayout *layout)
Returns a new map item for the specified layout.
QRectF boundingRect() const override
QString displayName() const override
Gets item display name.
bool atlasDriven() const
Returns whether the map extent is set to follow the current atlas feature.
void setZRange(const QgsDoubleRange &range)
Sets the map's z range, which is used to filter the map's content to only display features within the...
void setLayerStyleOverrides(const QMap< QString, QString > &overrides)
Sets the stored overrides of styles for layers.
void stopLayeredExport() override
Stops a multi-layer export operation.
void addRenderedFeatureHandler(QgsRenderedFeatureHandlerInterface *handler)
Adds a rendered feature handler to use while rendering the map.
double estimatedFrameBleed() const override
Returns the estimated amount the item's frame bleeds outside the item's actual rectangle.
QPainterPath framePath() const override
Returns the path to use when drawing the item's frame or background.
QgsExpressionContext createExpressionContext() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
void startLayeredExport() override
Starts a multi-layer export operation.
bool containsWmsLayer() const
Returns true if the map contains a WMS layer.
void setScale(double scale, bool forceUpdate=true)
Sets new map scale and changes only the map extent.
void refresh() override
double mapRotation(QgsLayoutObject::PropertyValueType valueType=QgsLayoutObject::EvaluatedValue) const
Returns the rotation used for drawing the map within the layout item, in degrees clockwise.
double mapUnitsToLayoutUnits() const
Returns the conversion factor from map units to layout units.
QgsLayoutItemMap::MapItemFlags mapFlags() const
Returns the map item's flags, which control how the map content is drawn.
bool zRangeEnabled() const
Returns whether the z range is enabled (i.e.
void setLabelMargin(const QgsLayoutMeasurement &margin)
Sets the margin from the map edges in which no labels may be placed.
void themeChanged(const QString &theme)
Emitted when the map's associated theme is changed.
QgsLayoutItem::ExportLayerDetail exportLayerDetails() const override
Returns the details for the specified current export layer.
void zoomToExtent(const QgsRectangle &extent)
Zooms the map so that the specified extent is fully visible within the map item.
double scale() const
Returns the map scale.
@ ShowPartialLabels
Whether to draw labels which are partially outside of the map view.
@ ShowUnplacedLabels
Whether to render unplaced labels in the map view.
bool drawAnnotations() const
Returns whether annotations are drawn within the map.
QgsDoubleRange zRange() const
Returns the map's z range, which is used to filter the map's content to only display features within ...
void removeLabelBlockingItem(QgsLayoutItem *item)
Removes the specified layout item from the map's "label blocking items".
void setCrs(const QgsCoordinateReferenceSystem &crs)
Sets the map's preset crs (coordinate reference system).
void invalidateCache() override
QgsRectangle extent() const
Returns the current map extent.
QgsLayoutItemMapOverview * overview()
Returns the map item's first overview.
void finalizeRestoreFromXml() override
Called after all pending items have been restored from XML.
bool readPropertiesFromElement(const QDomElement &element, const QDomDocument &document, const QgsReadWriteContext &context) override
Sets item state from a DOM element.
void setFrameStrokeWidth(QgsLayoutMeasurement width) override
Sets the frame stroke width.
bool containsAdvancedEffects() const override
Returns true if the item contains contents with blend modes or transparency effects which can only be...
friend class QgsLayoutItemMapGrid
QgsLayoutItem::Flags itemFlags() const override
Returns the item's flags, which indicate how the item behaves.
QList< QgsMapLayer * > layers() const
Returns the stored layer set.
void setMoveContentPreviewOffset(double dx, double dy) override
Sets temporary offset for the item, by a specified dx and dy in layout units.
void setMapRotation(double rotation)
Sets the rotation for the map - this does not affect the layout item shape, only the way the map is d...
QgsCoordinateReferenceSystem crs() const
Returns coordinate reference system used for rendering the map.
double atlasMargin(QgsLayoutObject::PropertyValueType valueType=QgsLayoutObject::EvaluatedValue)
Returns the margin size (percentage) used when the map is in atlas mode.
void addLabelBlockingItem(QgsLayoutItem *item)
Sets the specified layout item as a "label blocking item" for this map.
void assignFreeId()
Sets the map id() to a number not yet used in the layout.
ExportLayerBehavior exportLayerBehavior() const override
Returns the behavior of this item during exporting to layered exports (e.g.
QgsLabelingResults * previewLabelingResults() const
Returns the labeling results of the most recent preview map render.
Contains settings and helpers relating to a render of a QgsLayoutItem.
Base class for graphical items within a QgsLayout.
virtual void drawFrame(QgsRenderContext &context)
Draws the frame around the item.
virtual QPainterPath framePath() const
Returns the path to use when drawing the item's frame or background.
QColor backgroundColor(bool useDataDefined=true) const
Returns the background color for this item.
void drawRefreshingOverlay(QPainter *painter, const QStyleOptionGraphicsItem *itemStyle)
Draws a "refreshing" overlay icon on the item.
virtual void refreshDataDefinedProperty(QgsLayoutObject::DataDefinedProperty property=QgsLayoutObject::DataDefinedProperty::AllProperties)
Refreshes a data defined property for the item by reevaluating the property's value and redrawing the...
virtual void setFrameStrokeWidth(QgsLayoutMeasurement width)
Sets the frame stroke width.
void rotationChanged(double newRotation)
Emitted on item rotation change.
friend class QgsLayoutItemMap
QgsExpressionContext createExpressionContext() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
virtual void drawBackground(QgsRenderContext &context)
Draws the background for the item.
bool shouldDrawItem() const
Returns whether the item should be drawn in the current context.
@ FlagOverridesPaint
Item overrides the default layout item painting method.
@ FlagDisableSceneCaching
Item should not have QGraphicsItem caching enabled.
virtual bool containsAdvancedEffects() const
Returns true if the item contains contents with blend modes or transparency effects which can only be...
void sizePositionChanged()
Emitted when the item's size or position changes.
virtual QString uuid() const
Returns the item identification string.
QString id() const
Returns the item's ID name.
bool frameEnabled() const
Returns true if the item includes a frame.
ExportLayerBehavior
Behavior of item when exporting to layered outputs.
@ ItemContainsSubLayers
Item contains multiple sublayers which must be individually exported.
void clipPathChanged()
Emitted when the item's clipping path has changed.
bool hasBackground() const
Returns true if the item has a background.
QFlags< Flag > Flags
void refresh() override
Refreshes the item, causing a recalculation of any property overrides and recalculation of its positi...
void attemptSetSceneRect(const QRectF &rect, bool includesFrame=false)
Attempts to update the item's position and size to match the passed rect in layout coordinates.
virtual double estimatedFrameBleed() const
Returns the estimated amount the item's frame bleeds outside the item's actual rectangle.
QPainter::CompositionMode blendMode() const
Returns the item's composition blending mode.
void backgroundTaskCountChanged(int count)
Emitted whenever the number of background tasks an item is executing changes.
Provides a method of storing measurements for use in QGIS layouts using a variety of different measur...
void setLength(const double length)
Sets the length of the measurement.
static QgsLayoutMeasurement decodeMeasurement(const QString &string)
Decodes a measurement from a string.
QString encodeMeasurement() const
Encodes the layout measurement to a string.
Qgis::LayoutUnit units() const
Returns the units for the measurement.
void setUnits(const Qgis::LayoutUnit units)
Sets the units for the measurement.
double length() const
Returns the length of the measurement.
QgsPropertyCollection mDataDefinedProperties
const QgsLayout * layout() const
Returns the layout the object is attached to.
void changed()
Emitted when the object's properties change.
QPointer< QgsLayout > mLayout
DataDefinedProperty
Data defined properties for different item types.
@ MapYMin
Map extent y minimum.
@ MapZRangeUpper
Map frame Z-range lower value.
@ StartDateTime
Temporal range's start DateTime.
@ MapZRangeLower
Map frame Z-range lower value.
@ MapXMax
Map extent x maximum.
@ MapStylePreset
Layer and style map theme.
@ MapYMax
Map extent y maximum.
@ EndDateTime
Temporal range's end DateTime.
@ MapXMin
Map extent x minimum.
@ AllProperties
All properties for item.
PropertyValueType
Specifies whether the value returned by a function should be the original, user set value,...
@ EvaluatedValue
Return the current evaluated value for the property.
void predefinedScalesChanged()
Emitted when the list of predefined scales changes.
@ FlagRenderLabelsByMapLayer
When rendering map items to multi-layered exports, render labels belonging to different layers into s...
@ FlagAlwaysUseGlobalMasks
When applying clipping paths for selective masking, always use global ("entire map") paths,...
@ FlagUseAdvancedEffects
Enable advanced effects such as blend modes.
@ FlagLosslessImageRendering
Render images losslessly whenever possible, instead of the default lossy jpeg rendering used for some...
@ FlagAntialiasing
Use antialiasing when drawing items.
@ FlagForceVectorOutput
Force output in vector format where possible, even if items require rasterization to keep their corre...
@ FlagHideCoverageLayer
Hide coverage layer in outputs.
@ FlagDisableTiledRasterLayerRenders
If set, then raster layers will not be drawn as separate tiles. This may improve the appearance in ex...
static QgsRenderContext createRenderContextForMap(QgsLayoutItemMap *map, QPainter *painter, double dpi=-1)
Creates a render context suitable for the specified layout map and painter destination.
static void rotate(double angle, double &x, double &y)
Rotates a point / vector around the origin.
static Q_DECL_DEPRECATED double scaleFactorFromItemStyle(const QStyleOptionGraphicsItem *style)
Extracts the scale factor from an item style.
Base class for layouts, which can contain items such as maps, labels, scalebars, etc.
Definition qgslayout.h:49
QgsLayoutItem * itemByUuid(const QString &uuid, bool includeTemplateUuids=false) const
Returns the layout item with matching uuid unique identifier, or nullptr if a matching item could not...
void refreshed()
Emitted when the layout has been refreshed and items should also be refreshed and updated.
QgsProject * project() const
The project associated with the layout.
A map clipping region (in map coordinates and CRS).
void setRestrictToLayers(bool enabled)
Sets whether clipping should be restricted to a subset of layers.
FeatureClippingType
Feature clipping behavior, which controls how features from vector layers will be clipped.
void setFeatureClip(FeatureClippingType type)
Sets the feature clipping type.
void setRestrictedLayers(const QList< QgsMapLayer * > &layers)
Sets a list of layers to restrict the clipping region effects to.
Stores style information (renderer, opacity, labeling, diagrams etc.) applicable to a map layer.
void readXml(const QDomElement &styleElement)
Read style configuration (for project file reading)
void readFromLayer(QgsMapLayer *layer)
Store layer's active style information in the instance.
void writeXml(QDomElement &styleElement) const
Write style configuration (for project file writing)
QString xmlData() const
Returns XML content of the style.
Base class for all map layer types.
Definition qgsmaplayer.h:77
QString name
Definition qgsmaplayer.h:81
QString id
Definition qgsmaplayer.h:80
QgsProject * project() const
Returns the parent project if this map layer is added to a project.
Job implementation that renders everything sequentially using a custom painter.
void cancelWithoutBlocking() override
Triggers cancellation of the rendering job without blocking.
void finished()
emitted when asynchronous rendering is finished (or canceled).
Render job implementation that renders maps in stages, allowing different stages (e....
@ RenderLabelsByMapLayer
Labels should be rendered in individual stages by map layer. This allows separation of labels belongi...
static QStringList containsAdvancedEffects(const QgsMapSettings &mapSettings, EffectsCheckFlags flags=QgsMapSettingsUtils::EffectsCheckFlags())
Checks whether any of the layers attached to a map settings object contain advanced effects.
Contains configuration for rendering maps.
void setElevationShadingRenderer(const QgsElevationShadingRenderer &renderer)
Sets the shading renderer used to render shading on the entire map.
void addClippingRegion(const QgsMapClippingRegion &region)
Adds a new clipping region to the map settings.
QList< QgsMapLayer * > layers(bool expandGroupLayers=false) const
Returns the list of layers which will be rendered in the map.
void setSelectionColor(const QColor &color)
Sets the color that is used for drawing of selected vector features.
void setSimplifyMethod(const QgsVectorSimplifyMethod &method)
Sets the simplification setting to use when rendering vector layers.
QPolygonF visiblePolygon() const
Returns the visible area as a polygon (may be rotated)
void addRenderedFeatureHandler(QgsRenderedFeatureHandlerInterface *handler)
Adds a rendered feature handler to use while rendering the map settings.
void setTextRenderFormat(Qgis::TextRenderFormat format)
Sets the text render format, which dictates how text is rendered (e.g.
void setLayers(const QList< QgsMapLayer * > &layers)
Sets the list of layers to render in the map.
bool setEllipsoid(const QString &ellipsoid)
Sets the ellipsoid by its acronym.
void setScaleMethod(Qgis::ScaleCalculationMethod method)
Sets the method to use for scale calculations for the map.
void setDpiTarget(double dpi)
Sets the target dpi (dots per inch) to be taken into consideration when rendering.
void setDevicePixelRatio(float dpr)
Sets the device pixel ratio.
void setZRange(const QgsDoubleRange &range)
Sets the range of z-values which will be visible in the map.
long long currentFrame() const
Returns the current frame number of the map, for maps which are part of an animation.
void setOutputDpi(double dpi)
Sets the dpi (dots per inch) used for conversion between real world units (e.g.
void setRendererUsage(Qgis::RendererUsage rendererUsage)
Sets the rendering usage.
QgsRectangle extent() const
Returns geographical coordinates of the rectangle that should be rendered.
void setMaskSettings(const QgsMaskRenderSettings &settings)
Sets the mask render settings, which control how masks are drawn and behave during the map render.
void setLayerStyleOverrides(const QMap< QString, QString > &overrides)
Sets the map of map layer style overrides (key: layer ID, value: style name) where a different style ...
void setExtent(const QgsRectangle &rect, bool magnified=true)
Sets the coordinates of the rectangle which should be rendered.
void setExpressionContext(const QgsExpressionContext &context)
Sets the expression context.
double frameRate() const
Returns the frame rate of the map (in frames per second), for maps which are part of an animation.
void setLabelingEngineSettings(const QgsLabelingEngineSettings &settings)
Sets the global configuration of the labeling engine.
void setTransformContext(const QgsCoordinateTransformContext &context)
Sets the coordinate transform context, which stores various information regarding which datum transfo...
void setRotation(double rotation)
Sets the rotation of the resulting map image, in degrees clockwise.
void setPathResolver(const QgsPathResolver &resolver)
Sets the path resolver for conversion between relative and absolute paths during rendering operations...
void setLabelBoundaryGeometry(const QgsGeometry &boundary)
Sets the label boundary geometry, which restricts where in the rendered map labels are permitted to b...
void setLabelBlockingRegions(const QList< QgsLabelBlockingRegion > &regions)
Sets a list of regions to avoid placing labels within.
void setOutputSize(QSize size)
Sets the size of the resulting map image, in pixels.
void setBackgroundColor(const QColor &color)
Sets the background color of the map.
QgsCoordinateReferenceSystem destinationCrs() const
Returns the destination coordinate reference system for the map render.
void setFlag(Qgis::MapSettingsFlag flag, bool on=true)
Enable or disable a particular flag (other flags are not affected)
void setDestinationCrs(const QgsCoordinateReferenceSystem &crs)
Sets the destination crs (coordinate reference system) for the map render.
void mapThemeRenamed(const QString &name, const QString &newName)
Emitted when a map theme within the collection is renamed.
void mapThemeChanged(const QString &theme)
Emitted when a map theme changes definition.
Represents a 2D point.
Definition qgspointxy.h:60
double y
Definition qgspointxy.h:64
double x
Definition qgspointxy.h:63
QString description() const
Description.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
Definition qgsproject.h:107
Q_INVOKABLE QgsMapLayer * mapLayer(const QString &layerId) const
Retrieve a pointer to a registered layer by layer ID.
void layersWillBeRemoved(const QStringList &layerIds)
Emitted when one or more layers are about to be removed from the registry.
void crsChanged()
Emitted when the crs() of the project has changed.
QgsMapThemeCollection * mapThemeCollection
Definition qgsproject.h:115
void projectColorsChanged()
Emitted whenever the project's color scheme has been changed.
QgsLayerTree * layerTreeRoot() const
Returns pointer to the root (invisible) node of the project's layer tree.
Qgis::ScaleCalculationMethod scaleMethod
Definition qgsproject.h:128
T lower() const
Returns the lower bound of the range.
Definition qgsrange.h:78
T upper() const
Returns the upper bound of the range.
Definition qgsrange.h:85
A container for the context for various read/write operations on objects.
A rectangle specified with double values.
void scale(double scaleFactor, const QgsPointXY *c=nullptr)
Scale the rectangle around its center point.
double xMinimum
double yMinimum
double xMaximum
void setYMinimum(double y)
Set the minimum y value.
void setXMinimum(double x)
Set the minimum x value.
void setYMaximum(double y)
Set the maximum y value.
void setXMaximum(double x)
Set the maximum x value.
double yMaximum
static QgsRectangle fromCenterAndSize(const QgsPointXY &center, double width, double height)
Creates a new rectangle, given the specified center point and width and height.
QgsPointXY center
bool isFinite() const
Returns true if the rectangle has finite boundaries.
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...
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 setExpressionContext(const QgsExpressionContext &context)
Sets the expression context.
An interface for classes which provide custom handlers for features rendered as part of a map render ...
Calculates scale for a given combination of canvas size, map extent, and monitor dpi.
double calculate(const QgsRectangle &mapExtent, double canvasWidth) const
Calculate the scale denominator.
void setDpi(double dpi)
Sets the dpi (dots per inch) for the output resolution, to be used in scale calculations.
void setMapUnits(Qgis::DistanceUnit mapUnits)
Set the map units.
void setMethod(Qgis::ScaleCalculationMethod method)
Sets the method to use for map scale calculations.
Scoped object for saving and restoring a QPainter object's state.
An interface for classes which can visit style entity (e.g.
@ LayoutItem
Individual item in a print layout.
virtual bool visitExit(const QgsStyleEntityVisitorInterface::Node &node)
Called when the visitor stops visiting a node.
virtual bool visitEnter(const QgsStyleEntityVisitorInterface::Node &node)
Called when the visitor starts visiting a node.
const QgsDateTimeRange & temporalRange() const
Returns the datetime range for the object.
bool isTemporal() const
Returns true if the object's temporal range is enabled, and the object will be filtered when renderin...
void setIsTemporal(bool enabled)
Sets whether the temporal range is enabled (i.e.
void setTemporalRange(const QgsDateTimeRange &range)
Sets the temporal range for the object.
T begin() const
Returns the beginning of the range.
Definition qgsrange.h:446
T end() const
Returns the upper bound of the range.
Definition qgsrange.h:453
static Q_INVOKABLE QString toString(Qgis::DistanceUnit unit)
Returns a translated string representing a distance unit.
static double scaleToZoom(double mapScale, double z0Scale=559082264.0287178)
Finds zoom level given map scale denominator.
static int scaleToZoomLevel(double mapScale, int sourceMinZoom, int sourceMaxZoom, double z0Scale=559082264.0287178)
Finds the best fitting zoom level given a map scale denominator and allowed zoom level range.
static Qgis::GeometryType geometryType(Qgis::WkbType type)
Returns the geometry type for a WKB type, e.g., both MultiPolygon and CurvePolygon would have a Polyg...
#define Q_NOWARN_DEPRECATED_POP
Definition qgis.h:6819
QString qgsDoubleToString(double a, int precision=17)
Returns a string representation of a double.
Definition qgis.h:6203
#define Q_NOWARN_DEPRECATED_PUSH
Definition qgis.h:6818
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference)
Definition qgis.h:6286
QPointer< QgsMapLayer > QgsWeakMapLayerPointer
Weak pointer for QgsMapLayer.
QgsTemporalRange< QDateTime > QgsDateTimeRange
QgsRange which stores a range of date times.
Definition qgsrange.h:760
const QgsCoordinateReferenceSystem & crs
Single variable definition for use within a QgsExpressionContextScope.
Contains details of a particular export layer relating to a layout item.
QPainter::CompositionMode compositionMode
Associated composition mode if this layer is associated with a map layer.
QString mapLayerId
Associated map layer ID, or an empty string if this export layer is not associated with a map layer.
double opacity
Associated opacity, if this layer is associated with a map layer.
QString name
User-friendly name for the export layer.
QString mapTheme
Associated map theme, or an empty string if this export layer does not need to be associated with a m...
Contains information relating to a node (i.e.
TYPE * resolveWeakly(const QgsProject *project, MatchType matchType=MatchType::All)
Resolves the map layer by attempting to find a matching layer in a project using a weak match.
QString source
Weak reference to layer public source.
QString name
Weak reference to layer name.
QString provider
Weak reference to layer provider.
TYPE * resolve(const QgsProject *project)
Resolves the map layer by attempting to find a layer with matching ID within a project.
QString layerId
Original layer ID.