17#include "moc_qgsmaprendererjob.cpp"
20#include <QElapsedTimer>
22#include <QtConcurrentMap>
64LayerRenderJob &LayerRenderJob::operator=( LayerRenderJob &&other )
66 mContext = std::move( other.mContext );
71 renderer = other.renderer;
72 other.renderer =
nullptr;
74 previewRenderImage = other.previewRenderImage;
75 other.previewRenderImage =
nullptr;
77 imageInitialized = other.imageInitialized;
78 previewRenderImageInitialized = other.previewRenderImageInitialized;
80 blendMode = other.blendMode;
81 opacity = other.opacity;
82 cached = other.cached;
84 renderAboveLabels = other.renderAboveLabels;
85 completed = other.completed;
86 renderingTime = other.renderingTime;
87 estimatedRenderingTime = other.estimatedRenderingTime ;
88 errors = other.errors;
89 layerId = other.layerId;
91 maskPaintDevice = std::move( other.maskPaintDevice );
93 firstPassJob = other.firstPassJob;
94 other.firstPassJob =
nullptr;
96 picture = std::move( other.picture );
98 maskJobs = other.maskJobs;
100 maskRequiresLayerRasterization = other.maskRequiresLayerRasterization;
102 elevationMap = other.elevationMap;
103 maskPainter = std::move( other.maskPainter );
108LayerRenderJob::LayerRenderJob( LayerRenderJob &&other )
109 : imageInitialized( other.imageInitialized )
110 , previewRenderImageInitialized( other.previewRenderImageInitialized )
111 , blendMode( other.blendMode )
112 , opacity( other.opacity )
113 , cached( other.cached )
114 , renderAboveLabels( other.renderAboveLabels )
115 , layer( other.layer )
116 , completed( other.completed )
117 , renderingTime( other.renderingTime )
118 , estimatedRenderingTime( other.estimatedRenderingTime )
119 , errors( other.errors )
120 , layerId( other.layerId )
121 , maskPainter( nullptr )
122 , maskRequiresLayerRasterization( other.maskRequiresLayerRasterization )
123 , maskJobs( other.maskJobs )
125 mContext = std::move( other.mContext );
130 previewRenderImage = other.previewRenderImage;
131 other.previewRenderImage =
nullptr;
133 renderer = other.renderer;
134 other.renderer =
nullptr;
136 elevationMap = other.elevationMap;
137 other.elevationMap =
nullptr;
139 maskPaintDevice = std::move( other.maskPaintDevice );
141 firstPassJob = other.firstPassJob;
142 other.firstPassJob =
nullptr;
144 picture = std::move( other.picture );
147bool LayerRenderJob::imageCanBeComposed()
const
149 if ( imageInitialized )
153 return renderer->isReadyToCompose();
167 : mSettings( settings )
213 return mLabelingEngineFeedback;
218 QHash<QgsMapLayer *, int> result;
221 if (
auto &&lKey = it.key() )
222 result.insert( lKey, it.value() );
242 QSet< QgsMapLayer * > labeledLayers;
249 switch ( ml->type() )
297 bool canUseCache = canCache && QSet< QgsMapLayer * >( labelDependentLayers.begin(), labelDependentLayers.end() ) == labeledLayers;
326 static const double SPLIT_COORD = 180.0;
338 QgsDebugMsgLevel( QStringLiteral(
"\n0:%1 %2x%3\n1:%4\n2:%5 %6x%7 (w:%8 h:%9)" )
341 .arg( std::fabs( 1.0 - extent2.
width() / extent.
width() ) )
342 .arg( std::fabs( 1.0 - extent2.
height() / extent.
height() ) )
376 if ( ll.
x() > ur.
x() )
405 extent =
QgsRectangle( std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest(), std::numeric_limits<double>::max(), std::numeric_limits<double>::max() );
415 extent =
QgsRectangle( std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest(), std::numeric_limits<double>::max(), std::numeric_limits<double>::max() );
416 r2 =
QgsRectangle( std::numeric_limits<double>::lowest(), std::numeric_limits<double>::lowest(), std::numeric_limits<double>::max(), std::numeric_limits<double>::max() );
423QImage *QgsMapRendererJob::allocateImage( QString layerId )
430 if ( image->isNull() )
439QgsElevationMap *QgsMapRendererJob::allocateElevationMap( QString layerId )
442 if ( !elevationMap->isValid() )
447 return elevationMap.release();
450QPainter *QgsMapRendererJob::allocateImageAndPainter( QString layerId, QImage *&image,
const QgsRenderContext *context )
452 QPainter *painter =
nullptr;
453 image = allocateImage( layerId );
456 painter =
new QPainter( image );
462QgsMapRendererJob::PictureAndPainter QgsMapRendererJob::allocatePictureAndPainter(
const QgsRenderContext *context )
464 std::unique_ptr<QPicture> picture = std::make_unique<QPicture>();
465 QPainter *painter =
new QPainter( picture.get() );
467 return { std::move( picture ), painter };
472 std::vector< LayerRenderJob > layerJobs;
481 Q_UNUSED( cacheValid )
482 QgsDebugMsgLevel( QStringLiteral(
"CACHE VALID: %1" ).arg( cacheValid ), 4 );
487 while ( li.hasPrevious() )
491 QgsDebugMsgLevel( QStringLiteral(
"layer %1: minscale:%2 maxscale:%3 scaledepvis:%4 blendmode:%5 isValid:%6" )
508 QgsDebugMsgLevel( QStringLiteral(
"Layer not rendered because it is not within the defined visibility scale range" ), 3 );
514 QgsDebugMsgLevel( QStringLiteral(
"Layer not rendered because it is not visible within the map's time range" ), 3 );
520 QgsDebugMsgLevel( QStringLiteral(
"Layer not rendered because it is not visible within the map's z range" ), 3 );
529 bool haveExtentInLayerCrs =
true;
532 haveExtentInLayerCrs = reprojectToLayerExtent( ml, ct, r1, r2 );
537 mErrors.append( Error( ml->
id(), tr(
"There was a problem transforming the layer's extent. Layer skipped." ) ) );
548 if ( ( vl && vl->
isEditable() ) || requiresLabeling )
554 layerJobs.emplace_back( LayerRenderJob() );
555 LayerRenderJob &job = layerJobs.back();
557 job.layerId = ml->
id();
558 job.renderAboveLabels = ml->
customProperty( QStringLiteral(
"rendering/renderAboveLabels" ) ).toBool();
562 if ( !ml->
customProperty( QStringLiteral(
"_noset_layer_expression_context" ) ).toBool() )
564 job.context()->setPainter( painter );
565 job.context()->setLabelingEngine( labelingEngine2 );
566 job.context()->setLabelSink(
labelSink() );
567 job.context()->setCoordinateTransform( ct );
568 job.context()->setExtent( r1 );
574 if ( mFeatureFilterProvider )
575 job.context()->setFeatureFilterProvider( mFeatureFilterProvider );
608 job.imageInitialized =
true;
616 job.renderer =
nullptr;
617 job.context()->setPainter(
nullptr );
622 QElapsedTimer layerTime;
628 job.context()->setFeedback( job.renderer->feedback() );
639 if ( canUseCache || ( !painter && !deferredPainterSet ) || ( job.renderer && job.renderer->forceRasterRender() ) )
642 job.context()->setPainter( allocateImageAndPainter( ml->
id(), job.img, job.context() ) );
646 job.renderer =
nullptr;
647 layerJobs.pop_back();
656 job.elevationMap = allocateElevationMap( ml->
id() );
657 job.context()->setElevationMap( job.elevationMap );
665 if ( !cachedImage.isNull() )
667 job.previewRenderImage =
new QImage( cachedImage );
668 job.previewRenderImageInitialized =
true;
669 job.context()->setPreviewRenderPainter(
new QPainter( job.previewRenderImage ) );
670 job.context()->setPainterFlagsUsingContext( painter );
673 if ( !job.previewRenderImage )
675 job.context()->setPreviewRenderPainter( allocateImageAndPainter( ml->
id(), job.previewRenderImage, job.context() ) );
676 job.previewRenderImageInitialized =
false;
679 if ( !job.previewRenderImage )
681 delete job.context()->previewRenderPainter();
682 job.context()->setPreviewRenderPainter(
nullptr );
686 job.renderingTime = layerTime.elapsed();
694 std::vector< LayerRenderJob > secondPassJobs;
697 QHash<QString, LayerRenderJob *> layerJobMapping;
700 QMap<QString, bool> maskLayerHasEffects;
701 QMap<int, bool> labelHasEffects;
709 MaskSource(
const QString &layerId_,
const QString &labelRuleId_,
int labelMaskId_,
bool hasEffects_ ):
710 layerId( layerId_ ), labelRuleId( labelRuleId_ ), labelMaskId( labelMaskId_ ), hasEffects( hasEffects_ ) {}
715 QHash<QString, QPair<QSet<QString>, QList<MaskSource>>> maskedSymbolLayers;
722 for ( LayerRenderJob &job : firstPassJobs )
724 layerJobMapping[job.layerId] = &job;
729 for ( LayerRenderJob &job : firstPassJobs )
736 auto collectMasks = [&](
QgsMaskedLayers * masks, QString sourceLayerId, QString ruleId = QString(),
int labelMaskId = -1 )
738 bool hasEffects =
false;
739 for (
auto it = masks->begin(); it != masks->end(); ++it )
741 auto lit = maskedSymbolLayers.find( it.key() );
742 if ( lit == maskedSymbolLayers.end() )
744 maskedSymbolLayers[it.key()] = qMakePair( it.value().symbolLayerIds, QList<MaskSource>() << MaskSource( sourceLayerId, ruleId, labelMaskId, it.value().hasEffects ) );
748 if ( lit->first != it.value().symbolLayerIds )
750 QgsLogger::warning( QStringLiteral(
"Layer %1 : Different sets of symbol layers are masked by different sources ! Only one (arbitrary) set will be retained !" ).arg( it.key() ) );
753 lit->second.push_back( MaskSource( sourceLayerId, ruleId, labelMaskId, hasEffects ) );
755 hasEffects |= it.value().hasEffects;
757 if ( ! masks->isEmpty() && labelMaskId == -1 )
758 maskLayerHasEffects[ sourceLayerId ] = hasEffects;
763 for (
auto it = labelMasks.begin(); it != labelMasks.end(); it++ )
765 QString labelRule = it.key();
771 for (
auto mit = masks.begin(); mit != masks.end(); mit++ )
773 const QString sourceLayerId = mit.key();
775 if ( !layerJobMapping.contains( sourceLayerId ) )
778 usableMasks.insert( sourceLayerId, mit.value() );
781 if ( usableMasks.empty() )
785 QSet<QgsSymbolLayerReference> slRefs;
786 bool hasEffects =
false;
787 for (
auto mit = usableMasks.begin(); mit != usableMasks.end(); mit++ )
789 const QString sourceLayerId = mit.key();
791 if ( !layerJobMapping.contains( sourceLayerId ) )
794 for (
const QString &symbolLayerId : mit.value().symbolLayerIds )
797 hasEffects |= mit.value().hasEffects;
800 int labelMaskId = labelJob.maskIdProvider.insertLabelLayer( vl->
id(), it.key(), slRefs );
801 labelHasEffects[ labelMaskId ] = hasEffects;
804 collectMasks( &usableMasks, vl->
id(), labelRule, labelMaskId );
809 collectMasks( &symbolLayerMasks, vl->
id() );
812 if ( maskedSymbolLayers.isEmpty() )
813 return secondPassJobs;
816 for (
int maskId = 0; maskId < labelJob.maskIdProvider.size(); maskId++ )
818 QPaintDevice *maskPaintDevice =
nullptr;
819 QPainter *maskPainter =
nullptr;
820 if ( forceVector && !labelHasEffects[ maskId ] )
823 std::unique_ptr< QgsGeometryPaintDevice > geomPaintDevice = std::make_unique< QgsGeometryPaintDevice >(
true );
824 geomPaintDevice->setStrokedPathSegments( 4 );
825 geomPaintDevice->setSimplificationTolerance( labelJob.context.maskSettings().simplifyTolerance() );
826 maskPaintDevice = geomPaintDevice.release();
827 maskPainter =
new QPainter( maskPaintDevice );
832 QImage *maskImage =
nullptr;
833 maskPainter = allocateImageAndPainter( QStringLiteral(
"label mask" ), maskImage, &labelJob.context );
834 maskImage->fill( 0 );
835 maskPaintDevice = maskImage;
838 labelJob.context.setMaskPainter( maskPainter, maskId );
839 labelJob.maskPainters.push_back( std::unique_ptr<QPainter>( maskPainter ) );
840 labelJob.maskPaintDevices.push_back( std::unique_ptr<QPaintDevice>( maskPaintDevice ) );
842 labelJob.context.setMaskIdProvider( &labelJob.maskIdProvider );
853 if ( !labelJob.img && !forceVector )
855 labelJob.img = allocateImage( QStringLiteral(
"labels" ) );
857 else if ( !labelJob.picture && forceVector )
859 labelJob.picture.reset(
new QPicture() );
863 for ( LayerRenderJob &job : firstPassJobs )
865 job.maskRequiresLayerRasterization =
false;
867 auto it = maskedSymbolLayers.find( job.layerId );
868 if ( it != maskedSymbolLayers.end() )
870 const QList<MaskSource> &sourceList = it->second;
871 for (
const MaskSource &source : sourceList )
873 job.maskRequiresLayerRasterization |= source.hasEffects;
878 const bool isRasterRendering = !forceVector || job.maskRequiresLayerRasterization || ( job.renderer && job.renderer->forceRasterRender() );
879 if ( isRasterRendering && !job.img )
881 job.context()->setPainter( allocateImageAndPainter( job.layerId, job.img, job.context() ) );
883 else if ( !isRasterRendering && !job.picture )
885 PictureAndPainter pictureAndPainter = allocatePictureAndPainter( job.context() );
886 job.picture = std::move( pictureAndPainter.first );
887 if ( job.context()->painter()->hasClipping() )
890 pictureAndPainter.second->setClipping(
true );
891 pictureAndPainter.second->setClipPath( job.context()->painter()->clipPath() );
893 job.context()->setPainter( pictureAndPainter.second );
896 job.renderer = job.layer->createMapRenderer( *( job.context() ) );
900 if ( maskLayerHasEffects.contains( job.layerId ) )
902 QPaintDevice *maskPaintDevice =
nullptr;
903 QPainter *maskPainter =
nullptr;
904 if ( forceVector && !maskLayerHasEffects[ job.layerId ] )
907 std::unique_ptr< QgsGeometryPaintDevice > geomPaintDevice = std::make_unique< QgsGeometryPaintDevice >( );
908 geomPaintDevice->setStrokedPathSegments( 4 );
909 geomPaintDevice->setSimplificationTolerance( job.context()->maskSettings().simplifyTolerance() );
910 maskPaintDevice = geomPaintDevice.release();
911 maskPainter =
new QPainter( maskPaintDevice );
916 QImage *maskImage =
nullptr;
917 maskPainter = allocateImageAndPainter( job.layerId, maskImage, job.context() );
918 maskImage->fill( 0 );
919 maskPaintDevice = maskImage;
922 job.context()->setMaskPainter( maskPainter );
923 job.maskPainter.reset( maskPainter );
924 job.maskPaintDevice.reset( maskPaintDevice );
928 for ( LayerRenderJob &job : firstPassJobs )
932 auto it = maskedSymbolLayers.find( job.layerId );
933 if ( it == maskedSymbolLayers.end() )
936 QList<MaskSource> &sourceList = it->second;
937 const QSet<QString> symbolList = it->first;
939 secondPassJobs.emplace_back( LayerRenderJob() );
940 LayerRenderJob &job2 = secondPassJobs.back();
942 job2.maskRequiresLayerRasterization = job.maskRequiresLayerRasterization;
945 for ( MaskSource &source : sourceList )
947 if ( source.labelMaskId != -1 )
948 job2.maskJobs.push_back( qMakePair(
nullptr, source.labelMaskId ) );
950 job2.maskJobs.push_back( qMakePair( layerJobMapping[source.layerId], -1 ) );
954 job2.setContext( std::make_unique< QgsRenderContext >( *job.context() ) );
956 job2.layer = job.layer;
957 job2.renderAboveLabels = job.renderAboveLabels;
958 job2.layerId = job.layerId;
961 job2.firstPassJob = &job;
963 if ( !forceVector || job2.maskRequiresLayerRasterization )
965 job2.context()->setPainter( allocateImageAndPainter( job.layerId, job2.img, job2.context() ) );
969 PictureAndPainter pictureAndPainter = allocatePictureAndPainter( job2.context() );
970 if ( job.context()->painter()->hasClipping() )
973 pictureAndPainter.second->setClipping(
true );
974 pictureAndPainter.second->setClipPath( job.context()->painter()->clipPath() );
976 job2.picture = std::move( pictureAndPainter.first );
977 job2.context()->setPainter( pictureAndPainter.second );
980 if ( ! job2.img && ! job2.picture )
982 secondPassJobs.pop_back();
989 job2.renderer = mapRenderer;
992 job2.context()->setFeedback( job2.renderer->feedback() );
997 job2.context()->setDisabledSymbolLayersV2( symbolList );
1000 return secondPassJobs;
1005 QList<QPointer<QgsMapLayer> > res = _qgis_listRawToQPointer( engine->
participatingLayers() );
1009 if ( !res.contains( it ) )
1021 for ( LayerRenderJob &job : secondPassJobs )
1023 if ( job.maskRequiresLayerRasterization )
1029 for (
const QPair<LayerRenderJob *, int> &p : std::as_const( job.maskJobs ) )
1031 QPainter *maskPainter = p.first ? p.first->maskPainter.get() : labelJob.maskPainters[p.second].get();
1033 const QSet<QString> layers = job.context()->disabledSymbolLayersV2();
1036 QgsGeometry geometry( geometryDevice->geometry().clone() );
1038#if GEOS_VERSION_MAJOR==3 && GEOS_VERSION_MINOR<10
1045 for (
const QString &symbolLayerId : layers )
1047 job.context()->addSymbolLayerClipGeometry( symbolLayerId, geometry );
1052 job.context()->setDisabledSymbolLayersV2( QSet<QString>() );
1060 job.context.setPainter( painter );
1061 job.context.setLabelingEngine( labelingEngine2 );
1062 job.context.setFeedback( mLabelingEngineFeedback );
1063 if ( labelingEngine2 )
1064 job.context.labelingEngine()->prepare( job.context );
1068 job.context.setExtent( r1 );
1070 job.context.setFeatureFilterProvider( mFeatureFilterProvider );
1073 job.context.setCoordinateTransform( ct );
1084 job.complete =
true;
1087 job.context.setPainter(
nullptr );
1091 if ( canUseLabelCache && (
mCache || !painter ) )
1093 job.img = allocateImage( QStringLiteral(
"labels" ) );
1103 for ( LayerRenderJob &job : jobs )
1107 delete job.context()->painter();
1108 job.context()->setPainter(
nullptr );
1110 if (
mCache && !job.cached && job.completed && job.layer )
1112 QgsDebugMsgLevel( QStringLiteral(
"caching image for %1" ).arg( job.layerId ), 2 );
1121 if ( job.previewRenderImage )
1123 delete job.context()->previewRenderPainter();
1124 job.context()->setPreviewRenderPainter(
nullptr );
1125 delete job.previewRenderImage;
1126 job.previewRenderImage =
nullptr;
1129 if ( job.elevationMap )
1131 job.context()->setElevationMap(
nullptr );
1132 if (
mCache && !job.cached && job.completed && job.layer )
1134 QgsDebugMsgLevel( QStringLiteral(
"caching elevation map for %1" ).arg( job.layerId ), 2 );
1137 job.elevationMap->rawElevationImage(),
1140 QList< QgsMapLayer * >() << job.layer );
1143 job.elevationMap->rawElevationImage(),
1146 QList< QgsMapLayer * >() << job.layer );
1149 delete job.elevationMap;
1150 job.elevationMap =
nullptr;
1155 delete job.context()->painter();
1156 job.context()->setPainter(
nullptr );
1157 job.picture.reset(
nullptr );
1162 const QStringList
errors = job.renderer->errors();
1163 for (
const QString &message :
errors )
1164 mErrors.append( Error( job.renderer->layerId(), message ) );
1166 mRenderedItemResults->appendResults( job.renderer->takeRenderedItemDetails(), *job.context() );
1168 delete job.renderer;
1169 job.renderer =
nullptr;
1175 job.maskPainter.reset(
nullptr );
1176 job.maskPaintDevice.reset(
nullptr );
1184 for ( LayerRenderJob &job : jobs )
1188 delete job.context()->painter();
1189 job.context()->setPainter(
nullptr );
1195 if ( job.previewRenderImage )
1197 delete job.context()->previewRenderPainter();
1198 job.context()->setPreviewRenderPainter(
nullptr );
1199 delete job.previewRenderImage;
1200 job.previewRenderImage =
nullptr;
1205 delete job.context()->painter();
1206 job.context()->setPainter(
nullptr );
1211 delete job.renderer;
1212 job.renderer =
nullptr;
1226 if (
mCache && !job.cached && !job.context.renderingStopped() )
1237 job.picture.reset(
nullptr );
1238 job.maskPainters.clear();
1239 job.maskPaintDevices.clear();
1243#define DEBUG_RENDERING 0
1246 const std::vector<LayerRenderJob> &jobs,
1247 const LabelRenderJob &labelJob,
1253 image.setDotsPerMeterX(
static_cast<int>( settings.
outputDpi() * 39.37 ) );
1254 image.setDotsPerMeterY(
static_cast<int>( settings.
outputDpi() * 39.37 ) );
1258 std::unique_ptr<QgsElevationMap> mainElevationMap;
1259 if ( mapShadingRenderer.
isActive() )
1262 QPainter painter( &image );
1267 for (
const LayerRenderJob &job : jobs )
1269 if ( job.renderAboveLabels )
1276 painter.setCompositionMode( job.blendMode );
1277 painter.setOpacity( job.opacity );
1279 if ( mainElevationMap )
1282 if ( layerElevationMap.
isValid() )
1288 img.save( QString(
"/tmp/final_%1.png" ).arg( i ) );
1292 painter.drawImage( 0, 0, img );
1295 if ( mapShadingRenderer.
isActive() && mainElevationMap )
1303 if ( labelJob.img && labelJob.complete )
1305 painter.setCompositionMode( QPainter::CompositionMode_SourceOver );
1306 painter.setOpacity( 1.0 );
1307 painter.drawImage( 0, 0, *labelJob.img );
1315 painter.setCompositionMode( QPainter::CompositionMode_SourceOver );
1316 painter.setOpacity( 1.0 );
1317 painter.drawImage( 0, 0, labelCacheImage );
1321 for (
const LayerRenderJob &job : jobs )
1323 if ( !job.renderAboveLabels )
1330 painter.setCompositionMode( job.blendMode );
1331 painter.setOpacity( job.opacity );
1333 painter.drawImage( 0, 0, img );
1338 image.save(
"/tmp/final.png" );
1345 const LayerRenderJob &job,
1349 if ( job.imageCanBeComposed() )
1351 if ( job.previewRenderImage && !job.completed )
1352 return *job.previewRenderImage;
1354 Q_ASSERT( job.img );
1359 if ( cache && cache->
hasAnyCacheImage( job.layerId + QStringLiteral(
"_preview" ) ) )
1370 if ( job.imageCanBeComposed() && job.elevationMap )
1372 return *job.elevationMap;
1386 for ( LayerRenderJob &job : secondPassJobs )
1388 const bool isRasterRendering = !forceVector || job.maskRequiresLayerRasterization;
1391 if ( isRasterRendering && job.maskJobs.size() > 1 )
1393 QPainter *maskPainter =
nullptr;
1394 for ( QPair<LayerRenderJob *, int> p : job.maskJobs )
1396 QImage *maskImage =
static_cast<QImage *
>( p.first ? p.first->maskPaintDevice.get() : labelJob.maskPaintDevices[p.second].get() );
1399 maskPainter = p.first ? p.first->maskPainter.get() : labelJob.maskPainters[ p.second ].get();
1403 maskPainter->drawImage( 0, 0, *maskImage );
1408 if ( ! job.maskJobs.isEmpty() )
1411 QPair<LayerRenderJob *, int> p = *job.maskJobs.begin();
1412 if ( isRasterRendering )
1414 QImage *maskImage =
static_cast<QImage *
>( p.first ? p.first->maskPaintDevice.get() : labelJob.maskPaintDevices[p.second].get() );
1417 QPainter *painter = job.context()->painter();
1419 painter->setCompositionMode( QPainter::CompositionMode_DestinationIn );
1424 QImage maskBinAlpha = maskImage->createMaskFromColor( 0 );
1425 QVector<QRgb> mswTable;
1426 mswTable.push_back( qRgba( 0, 0, 0, 255 ) );
1427 mswTable.push_back( qRgba( 0, 0, 0, 0 ) );
1428 maskBinAlpha.setColorTable( mswTable );
1429 painter->drawImage( 0, 0, maskBinAlpha );
1433 QPainter tempPainter;
1436 QPainter *painter1 = job.firstPassJob->context()->painter();
1439 tempPainter.begin( job.firstPassJob->img );
1440 painter1 = &tempPainter;
1444 painter1->setCompositionMode( QPainter::CompositionMode_DestinationOut );
1445 painter1->drawImage( 0, 0, *maskImage );
1448 painter1->setCompositionMode( QPainter::CompositionMode_DestinationOver );
1449 painter1->drawImage( 0, 0, *job.img );
1454 job.firstPassJob->picture = std::move( job.picture );
1455 job.picture =
nullptr;
1466 QMultiMap<int, QString> elapsed;
1467 for (
const LayerRenderJob &job : jobs )
1469 for (
const LayerRenderJob &job : secondPassJobs )
1470 elapsed.insert( job.
renderingTime, job.layerId + QString(
" (second pass)" ) );
1472 elapsed.insert( labelJob.renderingTime, tr(
"Labeling" ) );
1474 QList<int> tt( elapsed.uniqueKeys() );
1475 std::sort( tt.begin(), tt.end(), std::greater<int>() );
1476 for (
int t : std::as_const( tt ) )
1478 QgsMessageLog::logMessage( tr(
"%1 ms: %2" ).arg( t ).arg( QStringList( elapsed.values( t ) ).join( QLatin1String(
", " ) ) ), tr(
"Rendering" ) );
1487 std::unique_ptr< QgsScopedRuntimeProfile > labelingProfile;
1490 labelingProfile = std::make_unique< QgsScopedRuntimeProfile >( QObject::tr(
"(labeling)" ), QStringLiteral(
"rendering" ) );
1497 painter->setCompositionMode( QPainter::CompositionMode_SourceOver );
1501 if ( labelingEngine2 )
1503 labelingEngine2->
run( renderContext );
1506 QgsDebugMsgLevel( QStringLiteral(
"Draw labeling took (seconds): %1" ).arg( t.elapsed() / 1000. ), 2 );
1511 Q_UNUSED( settings )
1513 drawLabeling( renderContext, labelingEngine2, painter );
@ InternalLayerOpacityHandling
The renderer internally handles the raster layer's opacity, so the default layer level opacity handli...
@ Group
Composite group layer. Added in QGIS 3.24.
@ Plugin
Plugin based layer.
@ TiledScene
Tiled scene layer. Added in QGIS 3.34.
@ Annotation
Contains freeform, georeferenced annotations. Added in QGIS 3.16.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
@ Mesh
Mesh layer. Added in QGIS 3.2.
@ PointCloud
Point cloud layer. Added in QGIS 3.18.
@ AffectsLabeling
The layer rendering will interact with the map labeling.
@ RenderPartialOutputOverPreviousCachedImage
When rendering temporary in-progress preview renders, these preview renders can be drawn over any pre...
@ RenderPartialOutputs
The renderer benefits from rendering temporary in-progress preview renders. These are temporary resul...
@ ApplyClipAfterReprojection
Feature geometry clipping to mapExtent() must be performed after the geometries are transformed using...
@ RecordProfile
Enable run-time profiling while rendering.
@ Linework
Combines all rings into a set of noded lines and then extracts valid polygons from that linework.
@ Structure
Structured method, first makes all rings valid and then merges shells and subtracts holes from shells...
@ Forward
Forward transform (from source to destination)
@ Reverse
Reverse/inverse transform (from destination to source)
@ 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...
virtual bool requiresAdvancedEffects() const =0
Returns true if drawing labels requires advanced effects like composition modes, which could prevent ...
virtual bool requiresAdvancedEffects() const =0
Returns true if drawing labels requires advanced effects like composition modes, which could prevent ...
Custom exception class for Coordinate Reference System related exceptions.
bool isInfinite() const
Returns true if the range consists of all possible values.
Stores digital elevation model in a raster image which may get updated as a part of map layer renderi...
bool isValid() const
Returns whether the elevation map is valid.
This class can render elevation shading on an image with different methods (eye dome lighting,...
Qgis::ElevationMapCombineMethod combinedElevationMethod() const
Returns the method used when conbining different elevation sources.
bool isActive() const
Returns whether this shading renderer is active.
void renderShading(const QgsElevationMap &elevation, QImage &image, const QgsRenderContext &context) const
Render shading on image condidering the elevation map elevation and the renderer context context If e...
static QgsExpressionContextScope * layerScope(const QgsMapLayer *layer)
Creates a new scope which contains variables and functions relating to a QgsMapLayer.
A paint device which converts everything renderer to a QgsGeometry representation of the rendered sha...
A geometry is the spatial representation of a feature.
QgsFeedback subclass for granular reporting of labeling engine progress.
The QgsLabelingEngine class provides map labeling functionality.
virtual void run(QgsRenderContext &context)=0
Runs the labeling job.
QList< QgsMapLayer * > participatingLayers() const
Returns a list of layers with providers in the engine.
static void warning(const QString &msg)
Goes to qWarning.
virtual bool isVisibleInZRange(const QgsDoubleRange &range, QgsMapLayer *layer=nullptr) const
Returns true if the layer should be visible and rendered for the specified z range.
virtual bool hasElevation() const
Returns true if the layer has an elevation or z component.
virtual void setLayerRenderingTimeHint(int time)
Sets approximate render time (in ms) for the layer to render.
Restore overridden layer style on destruction.
virtual bool isVisibleInTemporalRange(const QgsDateTimeRange &range) const
Returns true if the layer should be visible and rendered for the specified time range.
Base class for all map layer types.
bool isInScaleRange(double scale) const
Tests whether the layer should be visible at the specified scale.
Q_INVOKABLE QVariant customProperty(const QString &value, const QVariant &defaultValue=QVariant()) const
Read a custom property from layer.
QgsCoordinateReferenceSystem crs
QPainter::CompositionMode blendMode() const
Returns the current blending mode for a layer.
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
virtual QgsMapLayerTemporalProperties * temporalProperties()
Returns the layer's temporal properties.
virtual QgsMapLayerRenderer * createMapRenderer(QgsRenderContext &rendererContext)=0
Returns new instance of QgsMapLayerRenderer that will be used for rendering of given context.
double minimumScale() const
Returns the minimum map scale (i.e.
virtual QgsMapLayerElevationProperties * elevationProperties()
Returns the layer's elevation properties.
double maximumScale() const
Returns the maximum map scale (i.e.
This class is responsible for keeping cache of rendered images resulting from a map rendering job.
bool updateParameters(const QgsRectangle &extent, const QgsMapToPixel &mtp)
Sets extent and scale parameters.
QList< QgsMapLayer * > dependentLayers(const QString &cacheKey) const
Returns a list of map layers on which an image in the cache depends.
bool hasCacheImage(const QString &cacheKey) const
Returns true if the cache contains an image with the specified cacheKey that has the same extent and ...
QImage cacheImage(const QString &cacheKey) const
Returns the cached image for the specified cacheKey.
bool hasAnyCacheImage(const QString &cacheKey, double minimumScaleThreshold=0, double maximumScaleThreshold=0) const
Returns true if the cache contains an image with the specified cacheKey with any cache's parameters (...
void setCacheImageWithParameters(const QString &cacheKey, const QImage &image, const QgsRectangle &extent, const QgsMapToPixel &mapToPixel, const QList< QgsMapLayer * > &dependentLayers=QList< QgsMapLayer * >())
Set the cached image for a particular cacheKey, using a specific extent and mapToPixel (which may dif...
void clearCacheImage(const QString &cacheKey)
Removes an image from the cache with matching cacheKey.
QImage transformedCacheImage(const QString &cacheKey, const QgsMapToPixel &mtp) const
Returns the cached image for the specified cacheKey transformed to the particular extent and scale.
Abstract base class for map rendering implementations.
void logRenderingTime(const std::vector< LayerRenderJob > &jobs, const std::vector< LayerRenderJob > &secondPassJobs, const LabelRenderJob &labelJob)
QList< QPointer< QgsMapLayer > > participatingLabelLayers(QgsLabelingEngine *engine)
Returns a list of the layers participating in the map labeling.
static QgsElevationMap layerElevationToBeComposed(const QgsMapSettings &settings, const LayerRenderJob &job, const QgsMapRendererCache *cache)
static QImage composeImage(const QgsMapSettings &settings, const std::vector< LayerRenderJob > &jobs, const LabelRenderJob &labelJob, const QgsMapRendererCache *cache=nullptr)
void cleanupSecondPassJobs(std::vector< LayerRenderJob > &jobs)
void setCache(QgsMapRendererCache *cache)
Assign a cache to be used for reading and storing rendered images of individual layers.
QHash< QgsMapLayer *, int > perLayerRenderingTime() const
Returns the render time (in ms) per layer.
void initSecondPassJobs(std::vector< LayerRenderJob > &secondPassJobs, LabelRenderJob &labelJob) const
Initialize secondPassJobs according to what have been rendered (mask clipping path e....
static QImage layerImageToBeComposed(const QgsMapSettings &settings, const LayerRenderJob &job, const QgsMapRendererCache *cache)
QHash< QString, int > mLayerRenderingTimeHints
Approximate expected layer rendering time per layer, by layer ID.
std::unique_ptr< QgsRenderedItemResults > mRenderedItemResults
Errors errors() const
List of errors that happened during the rendering job - available when the rendering has been finishe...
static Q_DECL_DEPRECATED void drawLabeling(const QgsMapSettings &settings, QgsRenderContext &renderContext, QgsLabelingEngine *labelingEngine2, QPainter *painter)
static const QString LABEL_PREVIEW_CACHE_ID
QgsMapRendererCache ID string for cached label image during preview compositions only.
QList< QPointer< QgsMapLayer > > mAdditionalLabelLayers
Additional layers participating in labeling problem.
std::vector< LayerRenderJob > prepareJobs(QPainter *painter, QgsLabelingEngine *labelingEngine2, bool deferredPainterSet=false)
Creates a list of layer rendering jobs and prepares them for later render.
void cleanupJobs(std::vector< LayerRenderJob > &jobs)
const QgsMapSettings & mapSettings() const
Returns map settings with which this job was started.
QgsMapRendererCache * mCache
void finished()
emitted when asynchronous rendering is finished (or canceled).
static const QgsSettingsEntryBool * settingsLogCanvasRefreshEvent
Settings entry log canvas refresh event.
QgsMapRendererJob(const QgsMapSettings &settings)
~QgsMapRendererJob() override
void start()
Start the rendering job and immediately return.
int renderingTime() const
Returns the total time it took to finish the job (in milliseconds).
QStringList mLayersRedrawnFromCache
QStringList layersRedrawnFromCache() const
Returns a list of the layer IDs for all layers which were redrawn from cached images.
QList< QgsMapRendererJob::Error > Errors
static const QString LABEL_CACHE_ID
QgsMapRendererCache ID string for cached label image.
static const QString ELEVATION_MAP_CACHE_PREFIX
QgsMapRendererCache prefix string for cached elevation map image.
QHash< QgsWeakMapLayerPointer, int > mPerLayerRenderingTime
Render time (in ms) per layer, by layer ID.
QgsRenderedItemResults * takeRenderedItemResults()
Takes the rendered item results from the map render job and returns them.
QgsLabelingEngineFeedback * labelingEngineFeedback()
Returns the associated labeling engine feedback object.
static void composeSecondPass(std::vector< LayerRenderJob > &secondPassJobs, LabelRenderJob &labelJob, bool forceVector=false)
Compose second pass images into first pass images.
std::vector< LayerRenderJob > prepareSecondPassJobs(std::vector< LayerRenderJob > &firstPassJobs, LabelRenderJob &labelJob)
Prepares jobs for a second pass, if selective masks exist (from labels or symbol layers).
static const QgsSettingsEntryString * settingsMaskBackend
Settings entry for mask painting backend engine.
LabelRenderJob prepareLabelingJob(QPainter *painter, QgsLabelingEngine *labelingEngine2, bool canUseLabelCache=true)
Prepares a labeling job.
void setLayerRenderingTimeHints(const QHash< QString, int > &hints)
Sets approximate render times (in ms) for map layers.
void cleanupLabelJob(LabelRenderJob &job)
Handles clean up tasks for a label job, including deletion of images and storing cached label results...
QgsLabelSink * labelSink() const
Returns the label sink associated to this rendering job.
bool prepareLabelCache() const
Prepares the cache for storing the result of labeling.
QgsMapRendererQImageJob(const QgsMapSettings &settings)
The QgsMapSettings class contains configuration for rendering of the map.
QSize deviceOutputSize() const
Returns the device output size of the map render.
QList< QgsMapLayer * > layers(bool expandGroupLayers=false) const
Returns the list of layers which will be rendered in the map.
double scale() const
Returns the calculated map scale.
QgsCoordinateTransform layerTransform(const QgsMapLayer *layer) const
Returns the coordinate transform from layer's CRS to destination CRS.
QgsDoubleRange zRange() const
Returns the range of z-values which will be visible in the map.
QColor backgroundColor() const
Returns the background color of the map.
const QgsMapToPixel & mapToPixel() const
float devicePixelRatio() const
Returns the device pixel ratio.
QSize outputSize() const
Returns the size of the resulting map image, in pixels.
QImage::Format outputImageFormat() const
format of internal QImage, default QImage::Format_ARGB32_Premultiplied
double extentBuffer() const
Returns the buffer in map units to use around the visible extent for rendering symbols whose correspo...
Qgis::MapSettingsFlags flags() const
Returns combination of flags used for rendering.
QgsRectangle visibleExtent() const
Returns the actual extent derived from requested extent that takes output image size into account.
const QgsElevationShadingRenderer & elevationShadingRenderer() const
Returns the shading renderer used to render shading on the entire map.
double outputDpi() const
Returns the DPI (dots per inch) used for conversion between real world units (e.g.
bool testFlag(Qgis::MapSettingsFlag flag) const
Check whether a particular flag is enabled.
QMap< QString, QString > layerStyleOverrides() const
Returns the map of map layer style overrides (key: layer ID, value: style name) where a different sty...
bool hasValidSettings() const
Check whether the map settings are valid and can be used for rendering.
QgsCoordinateReferenceSystem destinationCrs() const
Returns the destination coordinate reference system for the map render.
Represents a mesh layer supporting display of data on structured or unstructured meshes.
const QgsAbstractMeshLayerLabeling * labeling() const
Access to const labeling configuration.
bool labelsEnabled() const
Returns whether the layer contains labels which are enabled and should be drawn.
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true)
Adds a message to the log instance (and creates it if necessary).
static bool staticWillUseLayer(const QgsMapLayer *layer)
Called to find out whether a specified layer is used for labeling.
A class to represent a 2D point.
QString toString(int precision=-1) const
Returns a string representation of the point (x, y) with a preset precision.
Represents a raster layer.
QgsRasterRenderer * renderer() const
Returns the raster's renderer.
virtual Qgis::RasterRendererFlags flags() const
Returns flags which dictate renderer behavior.
A rectangle specified with double values.
Q_INVOKABLE QString toString(int precision=16) const
Returns a string representation of form xmin,ymin : xmax,ymax Coordinates will be truncated to the sp...
void setXMinimum(double x)
Set the minimum x value.
void setXMaximum(double x)
Set the maximum x value.
void grow(double delta)
Grows the rectangle in place by the specified amount.
bool isFinite() const
Returns true if the rectangle has finite boundaries.
Contains information about the context of a rendering operation.
void setPainterFlagsUsingContext(QPainter *painter=nullptr) const
Sets relevant flags on a destination painter, using the flags and settings currently defined for the ...
void setPainter(QPainter *p)
Sets the destination QPainter for the render operation.
static QgsRenderContext fromMapSettings(const QgsMapSettings &mapSettings)
create initialized QgsRenderContext instance from given QgsMapSettings
Qgis::RenderContextFlags flags() const
Returns combination of flags used for rendering.
Stores collated details of rendered items during a map rendering operation.
T value(const QString &dynamicKeyPart=QString()) const
Returns settings value.
A boolean settings entry.
static QgsSettingsTreeNode * sTreeMap
Type used to refer to a specific symbol layer in a symbol of a layer.
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...
Implementation of threaded rendering for vector layers.
static QgsMaskedLayers symbolLayerMasks(const QgsVectorLayer *)
Returns all masks that may be defined on symbol layers for a given vector layer.
static QHash< QString, QgsMaskedLayers > labelMasks(const QgsVectorLayer *)
Returns masks defined in labeling options of a layer.
Represents a vector layer which manages a vector based data sets.
bool labelsEnabled() const
Returns whether the layer contains labels which are enabled and should be drawn.
const QgsAbstractVectorLayerLabeling * labeling() const
Access to const labeling configuration.
bool isEditable() const FINAL
Returns true if the provider is in editing mode.
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference)
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)
QHash< QString, QgsMaskedLayer > QgsMaskedLayers
masked layers where key is the layer id