22#include "delaunator.hpp"
40#include "qgsvirtualpointcloudprovider.h"
42#include <QElapsedTimer>
46using namespace Qt::StringLiterals;
50 , mLayerName( layer->name() )
51 , mLayerAttributes( layer->attributes() )
52 , mSubIndexes( layer->subIndexes() )
54 , mEnableProfile( context.
flags() &
Qgis::RenderContextFlag::RecordProfile )
59 mIndex = layer->
index();
65 if ( !mSubIndexes.isEmpty() )
67 mSubIndexExtentRenderer = std::make_unique<QgsPointCloudExtentRenderer>();
68 mSubIndexExtentRenderer->setShowLabels( mRenderer->showLabels() );
69 mSubIndexExtentRenderer->setLabelTextFormat( mRenderer->labelTextFormat() );
74 mScale = mIndex.scale();
75 mOffset = mIndex.offset();
80 mZOffset = elevationProps->zOffset();
81 mZScale = elevationProps->zScale();
84 if (
const QgsVirtualPointCloudProvider *vpcProvider =
dynamic_cast<QgsVirtualPointCloudProvider *
>( layer->
dataProvider() ) )
87 mAverageSubIndexWidth = vpcProvider->averageSubIndexWidth();
88 mAverageSubIndexHeight = vpcProvider->averageSubIndexHeight();
89 mOverviewIndexes = vpcProvider->overviews();
98 mPreparationTime = timer.elapsed();
105 std::unique_ptr< QgsScopedRuntimeProfile > profile;
106 if ( mEnableProfile )
108 profile = std::make_unique< QgsScopedRuntimeProfile >( mLayerName, u
"rendering"_s,
layerId() );
109 if ( mPreparationTime > 0 )
113 std::unique_ptr< QgsScopedRuntimeProfile > preparingProfile;
114 if ( mEnableProfile )
116 preparingProfile = std::make_unique< QgsScopedRuntimeProfile >( QObject::tr(
"Preparing render" ), u
"rendering"_s );
127 if ( !mClippingRegions.empty() )
129 bool needsPainterClipPath =
false;
131 if ( needsPainterClipPath )
135 if ( mRenderer->type() ==
"extent"_L1 )
138 mRenderer->startRender( context );
140 mRenderer->stopRender( context );
145 if ( mSubIndexes.isEmpty() && ( !mIndex || !mIndex.isValid() ) )
155 mBlockRenderUpdates =
true;
156 mElapsedTimer.start();
160 if ( elevationShadingRenderer.
isActive() )
162 auto elevationMap = std::make_unique<QgsElevationMap>(
renderContext()->deviceOutputSize(),
renderContext()->devicePixelRatio() );
166 mRenderer->startRender( context );
178 QSet< QString > rendererAttributes = mRenderer->usedAttributes( context );
181 for (
const QString &attribute : std::as_const( rendererAttributes ) )
183 if ( mAttributes.indexOf( attribute ) >= 0 )
186 const int layerIndex = mLayerAttributes.indexOf( attribute );
187 if ( layerIndex < 0 )
189 QgsMessageLog::logMessage( QObject::tr(
"Required attribute %1 not found in layer" ).arg( attribute ), QObject::tr(
"Point Cloud" ) );
193 mAttributes.push_back( mLayerAttributes.at( layerIndex ) );
206 preparingProfile.reset();
207 std::unique_ptr< QgsScopedRuntimeProfile > renderingProfile;
208 if ( mEnableProfile )
210 renderingProfile = std::make_unique< QgsScopedRuntimeProfile >( QObject::tr(
"Rendering" ), u
"rendering"_s );
213 bool canceled =
false;
214 if ( mSubIndexes.isEmpty() )
216 canceled = !renderIndex( mIndex );
220 QVector< QgsPointCloudSubIndex > visibleIndexes;
221 for (
const QgsPointCloudSubIndex &si : mSubIndexes )
225 visibleIndexes.append( si );
229 const double overviewSwitchingScale = mRenderer->overviewSwitchingScale();
230 const bool zoomedOut = renderExtent.
width() > mAverageSubIndexWidth * overviewSwitchingScale || renderExtent.
height() > mAverageSubIndexHeight * overviewSwitchingScale;
232 bool shouldRenderOverviews, shouldRenderExtents;
233 switch ( mRenderer->zoomOutBehavior() )
236 shouldRenderOverviews =
true;
237 shouldRenderExtents =
false;
240 shouldRenderOverviews =
true;
241 shouldRenderExtents =
true;
244 shouldRenderOverviews =
false;
245 shouldRenderExtents =
true;
249 if ( zoomedOut && shouldRenderOverviews )
253 if ( ovIdx.isValid() && renderExtent.
intersects( ovIdx.extent() ) )
254 renderIndex( ovIdx );
258 mSubIndexExtentRenderer->startRender( context );
259 for (
const QgsPointCloudSubIndex &si : visibleIndexes )
266 if ( ( zoomedOut && shouldRenderExtents ) || ( !zoomedOut && ( !pc || !pc.
isValid() ) ) )
268 mSubIndexExtentRenderer->renderExtent( si.polygonBounds(), context );
269 if ( mSubIndexExtentRenderer->showLabels() )
276 if ( pc && pc.
isValid() && !zoomedOut )
278 canceled = !renderIndex( pc );
281 mSubIndexExtentRenderer->stopRender( context );
284 if ( elevationShadingRenderer.
isActive() )
286 QImage *img =
dynamic_cast< QImage *
>( painter->device() );
291 mRenderer->elevationShadingRenderer().renderShading( *elevationMap, *img, *
renderContext() );
295 mRenderer->stopRender( context );
312 const double maximumError = context.renderContext().convertToPainterUnits( mRenderer->maximumScreenError(), mRenderer->maximumScreenErrorUnit() );
317 if ( !context.renderContext().coordinateTransform().isShortCircuited() )
325 catch ( QgsCsException & )
327 QgsDebugError( u
"Could not transform node extent to map CRS"_s );
328 rootNodeExtentMapCoords = rootNodeExtentLayerCoords;
333 rootNodeExtentMapCoords = rootNodeExtentLayerCoords;
336 const double rootErrorInMapCoordinates = rootNodeExtentMapCoords.
width() / pc.
span();
338 double mapUnitsPerPixel = context.renderContext().mapToPixel().mapUnitsPerPixel();
339 if ( ( rootErrorInMapCoordinates < 0.0 ) || ( mapUnitsPerPixel < 0.0 ) || ( maximumError < 0.0 ) )
344 double rootErrorPixels = rootErrorInMapCoordinates / mapUnitsPerPixel;
345 const QVector<QgsPointCloudNodeId> nodes = traverseTree( pc, context.renderContext(), pc.
root(), maximumError, rootErrorPixels );
347 QgsPointCloudRequest request;
352 bool canceled =
false;
355 if ( mRenderer->renderAsTriangles() )
367 nodesDrawn += renderNodesSorted( nodes, pc, context, request, canceled, mRenderer->drawOrder2d() );
376 nodesDrawn += renderNodesSync( nodes, pc, context, request, canceled );
381 nodesDrawn += renderNodesAsync( nodes, pc, context, request, canceled );
389 QgsDebugMsgLevel( u
"totals: %1 nodes | %2 points | %3ms"_s.arg( nodesDrawn ).arg( context.pointsRendered() ).arg( t.elapsed() ), 2 );
408 for ( QgsPointCloudNodeId n : nodes )
416 std::unique_ptr<QgsPointCloudBlock> block( pc.
nodeData( n, request ) );
421 QgsVector3D contextScale = context.
scale();
422 QgsVector3D contextOffset = context.
offset();
429 mRenderer->renderBlock( block.get(), context );
445 if ( mRenderer->renderAsTriangles() )
449 renderTriangulatedSurface( context );
457 if ( nodes.isEmpty() )
474 QVector<QgsPointCloudBlockRequest *> blockRequests;
479 for (
int i = 0; i < nodes.size(); ++i )
481 QgsPointCloudNodeId n = nodes[i];
483 QgsPointCloudBlockRequest *blockRequest = pc.
asyncNodeData( n, request );
484 blockRequests.append( blockRequest );
485 QObject::connect( blockRequest, &
QgsPointCloudBlockRequest::finished, &loop, [
this, &canceled, &nodesDrawn, &loop, &blockRequests, &context, nStr, blockRequest]() {
486 blockRequests.removeOne( blockRequest );
489 if ( blockRequests.isEmpty() )
492 std::unique_ptr<QgsPointCloudBlock> block( blockRequest->
takeBlock() );
494 blockRequest->deleteLater();
508 QgsVector3D contextScale = context.
scale();
509 QgsVector3D contextOffset = context.
offset();
515 mRenderer->renderBlock( block.get(), context );
533 if ( !blockRequests.isEmpty() )
538 for ( QgsPointCloudBlockRequest *blockRequest : std::as_const( blockRequests ) )
540 std::unique_ptr<QgsPointCloudBlock> block = blockRequest->
takeBlock();
543 blockRequest->deleteLater();
546 if ( mRenderer->renderAsTriangles() )
550 renderTriangulatedSurface( context );
556int QgsPointCloudLayerRenderer::renderNodesSorted(
563 QgsVector3D blockScale;
564 QgsVector3D blockOffset;
565 QgsPointCloudAttributeCollection blockAttributes;
569 QByteArray allByteArrays;
571 QVector<QPair<int, double>> allPairs;
573 for ( QgsPointCloudNodeId n : nodes )
581 std::unique_ptr<QgsPointCloudBlock> block( pc.
nodeData( n, request ) );
588 QgsVector3D offsetDifference = QgsVector3D( 0, 0, 0 );
589 if ( blockCount == 0 )
591 blockScale = block->scale();
592 blockOffset = block->offset();
593 blockAttributes = block->attributes();
597 offsetDifference = blockOffset - block->offset();
600 const char *ptr = block->data();
608 for (
int i = 0; i < block->pointCount(); ++i )
610 allByteArrays.append( ptr + i * recordSize, recordSize );
613 if ( offsetDifference.
x() != 0 )
615 qint32 ix = *
reinterpret_cast< const qint32 *
>( ptr + i * recordSize + context.
xOffset() );
616 ix -= std::lround( offsetDifference.
x() / context.
scale().
x() );
617 const char *xPtr =
reinterpret_cast< const char *
>( &ix );
618 allByteArrays.replace( pointCount * recordSize + context.
xOffset(), 4, QByteArray( xPtr, 4 ) );
620 if ( offsetDifference.
y() != 0 )
622 qint32 iy = *
reinterpret_cast< const qint32 *
>( ptr + i * recordSize + context.
yOffset() );
623 iy -= std::lround( offsetDifference.
y() / context.
scale().
y() );
624 const char *yPtr =
reinterpret_cast< const char *
>( &iy );
625 allByteArrays.replace( pointCount * recordSize + context.
yOffset(), 4, QByteArray( yPtr, 4 ) );
628 qint32 iz = *
reinterpret_cast< const qint32 *
>( ptr + i * recordSize + context.
zOffset() );
629 if ( offsetDifference.
z() != 0 )
631 iz -= std::lround( offsetDifference.
z() / context.
scale().
z() );
632 const char *zPtr =
reinterpret_cast< const char *
>( &iz );
633 allByteArrays.replace( pointCount * recordSize + context.
zOffset(), 4, QByteArray( zPtr, 4 ) );
635 allPairs.append( qMakePair( pointCount,
double( iz ) + block->offset().z() ) );
642 if ( pointCount == 0 )
648 std::sort( allPairs.begin(), allPairs.end(), []( QPair<int, double> a, QPair<int, double> b ) { return a.second < b.second; } );
651 std::sort( allPairs.begin(), allPairs.end(), []( QPair<int, double> a, QPair<int, double> b ) { return a.second > b.second; } );
658 QByteArray sortedByteArray;
659 sortedByteArray.reserve( allPairs.size() );
660 for ( QPair<int, double> pair : allPairs )
661 sortedByteArray.append( allByteArrays.mid( pair.first * recordSize, recordSize ) );
663 std::unique_ptr<QgsPointCloudBlock> bigBlock {
new QgsPointCloudBlock( pointCount, blockAttributes, sortedByteArray, blockScale, blockOffset ) };
665 QgsVector3D contextScale = context.
scale();
666 QgsVector3D contextOffset = context.
offset();
668 context.
setScale( bigBlock->scale() );
672 mRenderer->renderBlock( bigBlock.get(), context );
680inline bool isEdgeTooLong(
const QPointF &p1,
const QPointF &p2,
float length )
683 return p.x() * p.x() + p.y() * p.y() > length;
686static void renderTriangle( QImage &img, QPointF *pts, QRgb c0, QRgb c1, QRgb c2,
float horizontalFilter,
float *elev,
QgsElevationMap *elevationMap )
688 if ( horizontalFilter > 0 )
690 float filterThreshold2 = horizontalFilter * horizontalFilter;
695 QgsRectangle screenBBox = QgsMeshLayerUtils::triangleBoundingBox( pts[0], pts[1], pts[2] );
697 QSize outputSize = img.size();
699 int topLim = std::max(
int( screenBBox.
yMinimum() ), 0 );
700 int bottomLim = std::min(
int( screenBBox.
yMaximum() ), outputSize.height() - 1 );
701 int leftLim = std::max(
int( screenBBox.
xMinimum() ), 0 );
702 int rightLim = std::min(
int( screenBBox.
xMaximum() ), outputSize.width() - 1 );
704 int red0 = qRed( c0 ), green0 = qGreen( c0 ), blue0 = qBlue( c0 );
705 int red1 = qRed( c1 ), green1 = qGreen( c1 ), blue1 = qBlue( c1 );
706 int red2 = qRed( c2 ), green2 = qGreen( c2 ), blue2 = qBlue( c2 );
710 for (
int j = topLim; j <= bottomLim; j++ )
712 QRgb *scanLine = ( QRgb * ) img.scanLine( j );
713 QRgb *elevScanLine = elevData ? elevData +
static_cast<size_t>( outputSize.width() * j ) : nullptr;
714 for (
int k = leftLim; k <= rightLim; k++ )
717 double lam1, lam2, lam3;
718 if ( !QgsMeshLayerUtils::calculateBarycentricCoordinates( pts[0], pts[1], pts[2], pt, lam3, lam2, lam1 ) )
722 int r =
static_cast<int>( red0 * lam1 + red1 * lam2 + red2 * lam3 );
723 int g =
static_cast<int>( green0 * lam1 + green1 * lam2 + green2 * lam3 );
724 int b =
static_cast<int>( blue0 * lam1 + blue1 * lam2 + blue2 * lam3 );
725 scanLine[k] = qRgb( r, g, b );
730 float z =
static_cast<float>( elev[0] * lam1 + elev[1] * lam2 + elev[2] * lam3 );
739 const QgsPointCloudRenderContext::TriangulationData &triangulation = context.
triangulationData();
740 const std::vector<double> &points = triangulation.
points;
743 if ( points.size() < 3 )
749 std::unique_ptr<delaunator::Delaunator> delaunator;
752 delaunator = std::make_unique<delaunator::Delaunator>( points );
754 catch ( std::exception & )
761 float horizontalFilter = 0;
762 if ( mRenderer->horizontalTriangleFilter() )
764 horizontalFilter =
static_cast<float>(
renderContext()->
convertToPainterUnits( mRenderer->horizontalTriangleFilterThreshold(), mRenderer->horizontalTriangleFilterUnit() ) );
771 const std::vector<size_t> &triangleIndexes = delaunator->triangles;
775 float elev[3] { 0, 0, 0 };
776 for (
size_t i = 0; i < triangleIndexes.size(); i += 3 )
778 size_t v0 = triangleIndexes[i], v1 = triangleIndexes[i + 1], v2 = triangleIndexes[i + 2];
779 triangle[0].rx() = points[v0 * 2];
780 triangle[0].ry() = points[v0 * 2 + 1];
781 triangle[1].rx() = points[v1 * 2];
782 triangle[1].ry() = points[v1 * 2 + 1];
783 triangle[2].rx() = points[v2 * 2];
784 triangle[2].ry() = points[v2 * 2 + 1];
793 QRgb c0 = triangulation.
colors[v0], c1 = triangulation.
colors[v1], c2 = triangulation.
colors[v2];
794 renderTriangle( img, triangle, c0, c1, c2, horizontalFilter, elev, elevationMap );
797 painter->drawImage( 0, 0, img );
805 if ( mRenderer->renderAsTriangles() )
815 return mRenderer ? mRenderer->type() !=
"extent"_L1 :
false;
820 mRenderTimeHint = time;
825 QVector<QgsPointCloudNodeId> nodes;
833 QgsPointCloudNode node = pc.
getNode( n );
834 QgsBox3D nodeExtent = node.
bounds();
839 const QgsDoubleRange nodeZRange( nodeExtent.
zMinimum(), nodeExtent.
zMaximum() );
840 const QgsDoubleRange adjustedNodeZRange = QgsDoubleRange( nodeZRange.lower() + mZOffset, nodeZRange.upper() + mZOffset );
847 double childrenErrorPixels = nodeErrorPixels / 2.0;
848 if ( childrenErrorPixels < maxErrorPixels )
851 for ( QgsPointCloudNodeId nn : node.
children() )
853 nodes += traverseTree( pc, context, nn, maxErrorPixels, childrenErrorPixels );
Provides global constants and enumerations for use throughout the application.
QFlags< MapLayerRendererFlag > MapLayerRendererFlags
Flags which control how map layer renderers behave.
PointCloudDrawOrder
Pointcloud rendering order for 2d views.
@ BottomToTop
Draw points with larger Z values last.
@ Default
Draw points in the order they are stored.
@ TopToBottom
Draw points with larger Z values first.
@ RenderOverviewAndExtents
Render point cloud extents over overview point cloud.
@ RenderExtents
Render only point cloud extents when zoomed out.
@ RenderOverview
Render overview point cloud when zoomed out.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
@ 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...
@ Local
Local means the source is a local file on the machine.
@ Remote
Remote means it's loaded through a protocol like HTTP.
@ Reverse
Reverse/inverse transform (from destination to source).
static QgsRuntimeProfiler * profiler()
Returns the application runtime profiler.
double zMaximum() const
Returns the maximum z value.
QgsRectangle toRectangle() const
Converts the box to a 2D rectangle.
double zMinimum() const
Returns the minimum z value.
Custom exception class for Coordinate Reference System related exceptions.
bool isInfinite() const
Returns true if the range consists of all possible values.
Stores a digital elevation model in a raster image which may get updated as a part of the map layer r...
static QRgb encodeElevation(float z)
Converts elevation value to an actual color.
QRgb * rawElevationImageData()
Returns pointer to the actual elevation image data.
Renders elevation shading on an image with different methods (eye dome lighting, hillshading,...
bool isActive() const
Returns whether this shading renderer is active.
Base class for feedback objects to be used for cancellation of something running in a worker thread.
bool isCanceled() const
Tells whether the operation has been canceled already.
void canceled()
Internal routines can connect to this signal if they use event loop.
static QPainterPath calculatePainterClipRegion(const QList< QgsMapClippingRegion > ®ions, const QgsRenderContext &context, Qgis::LayerType layerType, bool &shouldClip)
Returns a QPainterPath representing the intersection of clipping regions from context which should be...
static QList< QgsMapClippingRegion > collectClippingRegionsForLayer(const QgsRenderContext &context, const QgsMapLayer *layer)
Collects the list of map clipping regions from a context which apply to a map layer.
bool mReadyToCompose
The flag must be set to false in renderer's constructor if wants to use the smarter map redraws funct...
static constexpr int MAX_TIME_TO_USE_CACHED_PREVIEW_IMAGE
Maximum time (in ms) to allow display of a previously cached preview image while rendering layers,...
QString layerId() const
Gets access to the ID of the layer rendered by this class.
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.
QRectF transformBounds(const QRectF &bounds) const
Transforms a bounding box from map coordinates to device coordinates.
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true, const char *file=__builtin_FILE(), const char *function=__builtin_FUNCTION(), int line=__builtin_LINE(), Qgis::StringFormat format=Qgis::StringFormat::PlainText)
Adds a message to the log instance (and creates it if necessary).
Attribute for point cloud data pair of name and size in bytes.
QString errorStr() const
Returns the error message string of the request.
void finished()
Emitted when the request processing has finished.
std::unique_ptr< QgsPointCloudBlock > takeBlock()
Returns the requested block.
virtual QgsGeometry polygonBounds() const
Returns the polygon bounds of the layer.
A renderer for 2d visualisation of point clouds which shows the dataset's extents using a fill symbol...
void renderExtent(const QgsGeometry &extent, QgsPointCloudRenderContext &context)
Renders a polygon extent geometry to the specified render context.
Smart pointer for QgsAbstractPointCloudIndex.
int span() const
Returns the number of points in one direction in a single node.
std::unique_ptr< QgsPointCloudBlock > nodeData(QgsPointCloudNodeId n, const QgsPointCloudRequest &request)
Returns node data block.
QgsVector3D offset() const
Returns offset of data from CRS.
QgsVector3D scale() const
Returns scale of data relative to CRS.
QgsPointCloudNode getNode(QgsPointCloudNodeId id) const
Returns object for a given node.
bool isValid() const
Returns whether index is loaded and valid.
QgsRectangle extent() const
Returns extent of the data.
QgsPointCloudNodeId root() const
Returns root node of the index.
QgsPointCloudBlockRequest * asyncNodeData(QgsPointCloudNodeId n, const QgsPointCloudRequest &request)
Returns a handle responsible for loading a node data block.
Qgis::PointCloudAccessType accessType() const
Returns the access type of the data If the access type is Remote, data will be fetched from an HTTP s...
Point cloud layer specific subclass of QgsMapLayerElevationProperties.
~QgsPointCloudLayerRenderer() override
bool forceRasterRender() const override
Returns true if the renderer must be rendered to a raster paint device (e.g.
QgsPointCloudLayerRenderer(QgsPointCloudLayer *layer, QgsRenderContext &context)
Ctor.
void setLayerRenderingTimeHint(int time) override
Sets approximate render time (in ms) for the layer to render.
bool render() override
Do the rendering (based on data stored in the class).
Qgis::MapLayerRendererFlags flags() const override
Returns flags which control how the map layer rendering behaves.
Represents a map layer supporting display of point clouds.
QgsMapLayerElevationProperties * elevationProperties() override
Returns the layer's elevation properties.
QgsPointCloudRenderer * renderer()
Returns the 2D renderer for the point cloud.
QgsPointCloudIndex index() const
Returns the point cloud index associated with the layer.
QgsPointCloudDataProvider * dataProvider() override
Returns the layer's data provider, it may be nullptr.
Represents an indexed point cloud node's position in octree.
QString toString() const
Encode node to string.
Keeps metadata for an indexed point cloud node.
QList< QgsPointCloudNodeId > children() const
Returns IDs of child nodes.
qint64 pointCount() const
Returns number of points contained in node data.
QgsBox3D bounds() const
Returns node's bounding cube in CRS coords.
Encapsulates the render context for a 2D point cloud rendering operation.
int yOffset() const
Returns the offset for the y value in a point record.
QgsVector3D offset() const
Returns the offset of the layer's int32 coordinates compared to CRS coords.
QgsRenderContext & renderContext()
Returns a reference to the context's render context.
void setOffset(const QgsVector3D &offset)
Sets the offset of the layer's int32 coordinates compared to CRS coords.
void setScale(const QgsVector3D &scale)
Sets the scale of the layer's int32 coordinates compared to CRS coords.
int pointRecordSize() const
Returns the size of a single point record.
int xOffset() const
Returns the offset for the x value in a point record.
QgsVector3D scale() const
Returns the scale of the layer's int32 coordinates compared to CRS coords.
TriangulationData & triangulationData()
Returns reference to the triangulation data structure (only used when rendering as triangles is enabl...
int zOffset() const
Returns the offset for the y value in a point record.
QgsFeedback * feedback() const
Returns the feedback object used to cancel rendering.
void setAttributes(const QgsPointCloudAttributeCollection &attributes)
Sets the attributes associated with the rendered block.
virtual QgsPointCloudRenderer * clone() const =0
Create a deep copy of this renderer.
Point cloud data request.
void setAttributes(const QgsPointCloudAttributeCollection &attributes)
Set attributes filter in the request.
bool overlaps(const QgsRange< T > &other) const
Returns true if this range overlaps another range.
A rectangle specified with double values.
bool intersects(const QgsRectangle &rect) const
Returns true when rectangle intersects with other rectangle.
Contains information about the context of a rendering operation.
double convertToPainterUnits(double size, Qgis::RenderUnit unit, const QgsMapUnitScale &scale=QgsMapUnitScale(), Qgis::RenderSubcomponentProperty property=Qgis::RenderSubcomponentProperty::Generic) const
Converts a size from the specified units to painter units (pixels).
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 ...
QgsElevationMap * elevationMap() const
Returns the destination elevation map for the render operation.
const QgsRectangle & extent() const
When rendering a map layer, calling this method returns the "clipping" extent for the layer (in the l...
float devicePixelRatio() const
Returns the device pixel ratio.
const QgsMapToPixel & mapToPixel() const
Returns the context's map to pixel transform, which transforms between map coordinates and device coo...
void setPainter(QPainter *p)
Sets the destination QPainter for the render operation.
QgsDoubleRange zRange() const
Returns the range of z-values which should be rendered.
QSize deviceOutputSize() const
Returns the device output size of the render.
bool renderingStopped() const
Returns true if the rendering operation has been stopped and any ongoing rendering should be canceled...
QPainter * previewRenderPainter()
Returns the const destination QPainter for temporary in-progress preview renders.
QgsCoordinateTransform coordinateTransform() const
Returns the current coordinate transform for the context.
void setElevationMap(QgsElevationMap *map)
Sets the destination elevation map for the render operation.
void record(const QString &name, double time, const QString &group="startup", const QString &id=QString())
Manually adds a profile event with the given name and total time (in seconds).
Scoped object for saving and restoring a QPainter object's state.
Scoped object for setting the current thread name.
double y() const
Returns Y coordinate.
double z() const
Returns Z coordinate.
double x() const
Returns X coordinate.
#define QgsDebugMsgLevel(str, level)
#define QgsDebugError(str)
bool isEdgeTooLong(const QPointF &p1, const QPointF &p2, float length)
std::vector< QRgb > colors
RGB color for each point.
std::vector< float > elevations
Z value for each point (only used when global map shading is enabled).
std::vector< double > points
X,Y for each point - kept in this structure so that we can use it without further conversions in Dela...