QGIS API Documentation 3.99.0-Master (26c88405ac0)
Loading...
Searching...
No Matches
qgsgrouplayerrenderer.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsgrouplayerrenderer.cpp
3 ----------------
4 Date : September 2021
5 Copyright : (C) 2021 by Nyall Dawson
6 Email : nyall dot dawson at gmail dot com
7 ***************************************************************************/
8
9/***************************************************************************
10 * *
11 * This program is free software; you can redistribute it and/or modify *
12 * it under the terms of the GNU General Public License as published by *
13 * the Free Software Foundation; either version 2 of the License, or *
14 * (at your option) any later version. *
15 * *
16 ***************************************************************************/
17
19
20#include <optional>
21
22#include "qgsfeedback.h"
23#include "qgsgrouplayer.h"
24#include "qgslogger.h"
25#include "qgspainteffect.h"
26#include "qgsrendercontext.h"
27
29 : QgsMapLayerRenderer( layer->id(), &context )
30 , mFeedback( std::make_unique< QgsFeedback >() )
31 , mLayerOpacity( layer->opacity() )
32{
33 const QList< QgsMapLayer * > layers = layer->childLayers();
34 const QgsCoordinateReferenceSystem destinationCrs = context.coordinateTransform().destinationCrs();
35 for ( QgsMapLayer *childLayer : layers )
36 {
37 // we have to temporarily set the context's crs and extent to the correct one for the child layer, BEFORE creating the
38 // child layer's renderer
39 QgsCoordinateTransform layerToDestTransform( childLayer->crs(), destinationCrs, context.transformContext() );
40 layerToDestTransform.setBallparkTransformsAreAppropriate( true );
41 context.setCoordinateTransform( layerToDestTransform );
42 try
43 {
44 const QgsRectangle extentInChildLayerCrs = layerToDestTransform.transformBoundingBox( context.mapExtent(), Qgis::TransformDirection::Reverse );
45 context.setExtent( extentInChildLayerCrs );
46 }
47 catch ( QgsCsException & )
48 {
49 QgsDebugError( QStringLiteral( "Error transforming extent of %1 to destination CRS" ).arg( childLayer->id() ) );
50 continue;
51 }
52
53 mChildRenderers.emplace_back( childLayer->createMapRenderer( context ) );
54 mRendererCompositionModes.emplace_back( childLayer->blendMode() );
55 mRendererOpacity.emplace_back( childLayer->type() != Qgis::LayerType::Raster ? childLayer->opacity() : 1.0 );
56 mTransforms.emplace_back( layerToDestTransform );
57 }
58
59 mPaintEffect.reset( layer->paintEffect() && layer->paintEffect()->enabled() ? layer->paintEffect()->clone() : nullptr );
60
61 mForceRasterRender = layer->blendMode() != QPainter::CompositionMode_SourceOver;
62}
63
65
67{
68 return mFeedback.get();
69}
70
72{
73 QgsRenderContext &context = *renderContext();
74
75 context.painter()->save();
76 if ( mPaintEffect )
77 {
78 mPaintEffect->begin( context );
79 }
80
81 const QgsCoordinateReferenceSystem destinationCrs = context.coordinateTransform().destinationCrs();
82 bool canceled = false;
83 int i = 0;
84 for ( const std::unique_ptr< QgsMapLayerRenderer > &renderer : std::as_const( mChildRenderers ) )
85 {
86 if ( mFeedback->isCanceled() )
87 {
88 canceled = true;
89 break;
90 }
91
92 context.setCoordinateTransform( mTransforms[i] );
93
94 // don't need to catch exceptions here -- it would have already been caught in the QgsGroupLayerRenderer constructor!
95 const QgsRectangle extentInChildLayerCrs = mTransforms[i].transformBoundingBox( context.mapExtent(), Qgis::TransformDirection::Reverse );
96 context.setExtent( extentInChildLayerCrs );
97
98 QImage image;
100 {
101 context.painter()->setCompositionMode( mRendererCompositionModes[i] );
102 }
103
104 QPainter *prevPainter = context.painter();
105 std::unique_ptr< QPainter > imagePainter;
106 if ( renderer->forceRasterRender() )
107 {
108 image = QImage( context.deviceOutputSize(), context.imageFormat() );
109 image.setDevicePixelRatio( static_cast<qreal>( context.devicePixelRatio() ) );
110 image.fill( 0 );
111 imagePainter = std::make_unique< QPainter >( &image );
112
113 context.setPainterFlagsUsingContext( imagePainter.get() );
114 context.setPainter( imagePainter.get() );
115 }
116 renderer->render();
117
118 if ( imagePainter )
119 {
120 imagePainter->end();
121 context.setPainter( prevPainter );
122
123 context.painter()->setOpacity( mRendererOpacity[i] );
124 context.painter()->drawImage( 0, 0, image );
125 context.painter()->setOpacity( 1.0 );
126 }
127 context.painter()->setCompositionMode( QPainter::CompositionMode_SourceOver );
128 i++;
129 }
130
131 if ( mPaintEffect )
132 {
133 mPaintEffect->end( context );
134 }
135
136 context.painter()->restore();
137
138 return !canceled;
139}
140
142{
143 switch ( renderContext()->rasterizedRenderingPolicy() )
144 {
147 break;
148
150 return false;
151 }
152
153 if ( mForceRasterRender || !qgsDoubleNear( mLayerOpacity, 1.0 ) )
154 return true;
155
156 for ( QPainter::CompositionMode mode : mRendererCompositionModes )
157 {
158 if ( mode != QPainter::CompositionMode_SourceOver )
159 return true;
160 }
161
162 return false;
163}
@ Default
Allow raster-based rendering in situations where it is required for correct rendering or where it wil...
Definition qgis.h:2704
@ PreferVector
Prefer vector-based rendering, when the result will still be visually near-identical to a raster-base...
Definition qgis.h:2705
@ ForceVector
Always force vector-based rendering, even when the result will be visually different to a raster-base...
Definition qgis.h:2706
@ Raster
Raster layer.
Definition qgis.h:192
@ Reverse
Reverse/inverse transform (from destination to source).
Definition qgis.h:2673
Represents a coordinate reference system (CRS).
Handles coordinate transforms between two coordinate systems.
void setBallparkTransformsAreAppropriate(bool appropriate)
Sets whether approximate "ballpark" results are appropriate for this coordinate transform.
QgsRectangle transformBoundingBox(const QgsRectangle &rectangle, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward, bool handle180Crossover=false) const
Transforms a rectangle from the source CRS to the destination CRS.
QgsCoordinateReferenceSystem destinationCrs() const
Returns the destination coordinate reference system, which the transform will transform coordinates t...
Custom exception class for Coordinate Reference System related exceptions.
Base class for feedback objects to be used for cancellation of something running in a worker thread.
Definition qgsfeedback.h:44
~QgsGroupLayerRenderer() override
QgsGroupLayerRenderer(QgsGroupLayer *layer, QgsRenderContext &context)
Constructor for a QgsGroupLayerRenderer, for the specified layer.
bool forceRasterRender() const override
Returns true if the renderer must be rendered to a raster paint device (e.g.
QgsFeedback * feedback() const override
Access to feedback object of the layer renderer (may be nullptr).
bool render() override
Do the rendering (based on data stored in the class).
A map layer which consists of a set of child layers, where all component layers are rendered as a sin...
QgsPaintEffect * paintEffect() const
Returns the current paint effect for the group layer.
QList< QgsMapLayer * > childLayers() const
Returns the child layers contained by the group.
QgsRenderContext * renderContext()
Returns the render context associated with the renderer.
QgsMapLayerRenderer(const QString &layerID, QgsRenderContext *context=nullptr)
Constructor for QgsMapLayerRenderer, with the associated layerID and render context.
Base class for all map layer types.
Definition qgsmaplayer.h:80
QPainter::CompositionMode blendMode() const
Returns the current blending mode for a layer.
bool enabled() const
Returns whether the effect is enabled.
virtual QgsPaintEffect * clone() const =0
Duplicates an effect by creating a deep copy of the effect.
A rectangle specified with double values.
Contains information about the context of a rendering operation.
void setCoordinateTransform(const QgsCoordinateTransform &t)
Sets the current coordinate transform for the context.
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 ...
QgsCoordinateTransformContext transformContext() const
Returns the context's coordinate transform context, which stores various information regarding which ...
float devicePixelRatio() const
Returns the device pixel ratio.
QgsRectangle mapExtent() const
Returns the original extent of the map being rendered.
void setExtent(const QgsRectangle &extent)
When rendering a map layer, calling this method sets the "clipping" extent for the layer (in the laye...
Qgis::RasterizedRenderingPolicy rasterizedRenderingPolicy() const
Returns the policy controlling when rasterisation of content during renders is permitted.
void setPainter(QPainter *p)
Sets the destination QPainter for the render operation.
QSize deviceOutputSize() const
Returns the device output size of the render.
QgsCoordinateTransform coordinateTransform() const
Returns the current coordinate transform for the context.
QImage::Format imageFormat() const
Returns the QImage format which should be used for QImages created during rendering.
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference).
Definition qgis.h:6607
#define QgsDebugError(str)
Definition qgslogger.h:57