QGIS API Documentation 3.99.0-Master (8e76e220402)
Loading...
Searching...
No Matches
qgspointcloudrgbrenderer.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgspointcloudrgbrenderer.h
3 --------------------
4 begin : October 2020
5 copyright : (C) 2020 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 <memory>
21
23#include "qgspointcloudblock.h"
24
25#include <QString>
26
27using namespace Qt::StringLiterals;
28
33
35{
36 return u"rgb"_s;
37}
38
40{
41 auto res = std::make_unique< QgsPointCloudRgbRenderer >();
42 res->mRedAttribute = mRedAttribute;
43 res->mGreenAttribute = mGreenAttribute;
44 res->mBlueAttribute = mBlueAttribute;
45
46 if ( mRedContrastEnhancement )
47 {
48 res->setRedContrastEnhancement( new QgsContrastEnhancement( *mRedContrastEnhancement ) );
49 }
50 if ( mGreenContrastEnhancement )
51 {
52 res->setGreenContrastEnhancement( new QgsContrastEnhancement( *mGreenContrastEnhancement ) );
53 }
54 if ( mBlueContrastEnhancement )
55 {
56 res->setBlueContrastEnhancement( new QgsContrastEnhancement( *mBlueContrastEnhancement ) );
57 }
58
59 copyCommonProperties( res.get() );
60
61 return res.release();
62}
63
65{
66 QgsRectangle visibleExtent = context.renderContext().extent();
67 if ( renderAsTriangles() )
68 {
69 // we need to include also points slightly outside of the visible extent,
70 // otherwise the triangulation may be missing triangles near the edges and corners
71 visibleExtent.grow( std::max( visibleExtent.width(), visibleExtent.height() ) * 0.05 );
72 }
73
74 const char *ptr = block->data();
75 const int count = block->pointCount();
76 const QgsPointCloudAttributeCollection request = block->attributes();
77
78 const std::size_t recordSize = request.pointRecordSize();
79 int redOffset = 0;
80 const QgsPointCloudAttribute *attribute = request.find( mRedAttribute, redOffset );
81 if ( !attribute )
82 return;
83 const QgsPointCloudAttribute::DataType redType = attribute->type();
84
85 int greenOffset = 0;
86 attribute = request.find( mGreenAttribute, greenOffset );
87 if ( !attribute )
88 return;
89 const QgsPointCloudAttribute::DataType greenType = attribute->type();
90
91 int blueOffset = 0;
92 attribute = request.find( mBlueAttribute, blueOffset );
93 if ( !attribute )
94 return;
95 const QgsPointCloudAttribute::DataType blueType = attribute->type();
96
97 const bool useRedContrastEnhancement = mRedContrastEnhancement && mRedContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
98 const bool useBlueContrastEnhancement = mBlueContrastEnhancement && mBlueContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
99 const bool useGreenContrastEnhancement = mGreenContrastEnhancement && mGreenContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
100
101 const bool renderElevation = context.renderContext().elevationMap();
102 const QgsDoubleRange zRange = context.renderContext().zRange();
103 const bool considerZ = !zRange.isInfinite() || renderElevation;
104
105 int rendered = 0;
106 double x = 0;
107 double y = 0;
108 double z = 0;
110 const bool reproject = ct.isValid();
111 for ( int i = 0; i < count; ++i )
112 {
113 if ( context.renderContext().renderingStopped() )
114 {
115 break;
116 }
117
118 if ( considerZ )
119 {
120 // z value filtering is cheapest, if we're doing it...
121 z = pointZ( context, ptr, i );
122 if ( !zRange.contains( z ) )
123 continue;
124 }
125
126 pointXY( context, ptr, i, x, y );
127 if ( visibleExtent.contains( x, y ) )
128 {
129 if ( reproject )
130 {
131 try
132 {
133 ct.transformInPlace( x, y, z );
134 }
135 catch ( QgsCsException & )
136 {
137 continue;
138 }
139 }
140
141 int red = 0;
142 context.getAttribute( ptr, i * recordSize + redOffset, redType, red );
143 int green = 0;
144 context.getAttribute( ptr, i * recordSize + greenOffset, greenType, green );
145 int blue = 0;
146 context.getAttribute( ptr, i * recordSize + blueOffset, blueType, blue );
147
148 //skip if red, green or blue not in displayable range
149 if ( ( useRedContrastEnhancement && !mRedContrastEnhancement->isValueInDisplayableRange( red ) )
150 || ( useGreenContrastEnhancement && !mGreenContrastEnhancement->isValueInDisplayableRange( green ) )
151 || ( useBlueContrastEnhancement && !mBlueContrastEnhancement->isValueInDisplayableRange( blue ) ) )
152 {
153 continue;
154 }
155
156 //stretch color values
157 if ( useRedContrastEnhancement )
158 {
159 red = mRedContrastEnhancement->enhanceContrast( red );
160 }
161 if ( useGreenContrastEnhancement )
162 {
163 green = mGreenContrastEnhancement->enhanceContrast( green );
164 }
165 if ( useBlueContrastEnhancement )
166 {
167 blue = mBlueContrastEnhancement->enhanceContrast( blue );
168 }
169
170 red = std::max( 0, std::min( 255, red ) );
171 green = std::max( 0, std::min( 255, green ) );
172 blue = std::max( 0, std::min( 255, blue ) );
173
174 if ( renderAsTriangles() )
175 {
176 addPointToTriangulation( x, y, z, QColor( red, green, blue ), context );
177
178 // We don't want to render any points if we're rendering triangles and there is no preview painter
179 if ( !context.renderContext().previewRenderPainter() )
180 continue;
181 }
182
183 drawPoint( x, y, QColor( red, green, blue ), context );
184 if ( renderElevation )
185 drawPointToElevationMap( x, y, z, context );
186
187 rendered++;
188 }
189 }
190 context.incrementPointsRendered( rendered );
191}
192
193
195{
196 auto r = std::make_unique< QgsPointCloudRgbRenderer >();
197
198 r->setRedAttribute( element.attribute( u"red"_s, u"Red"_s ) );
199 r->setGreenAttribute( element.attribute( u"green"_s, u"Green"_s ) );
200 r->setBlueAttribute( element.attribute( u"blue"_s, u"Blue"_s ) );
201
202 r->restoreCommonProperties( element, context );
203
204 //contrast enhancements
206 const QDomElement redContrastElem = element.firstChildElement( u"redContrastEnhancement"_s );
207 if ( !redContrastElem.isNull() )
208 {
210 redContrastEnhancement->readXml( redContrastElem );
211 r->setRedContrastEnhancement( redContrastEnhancement );
212 }
213
215 const QDomElement greenContrastElem = element.firstChildElement( u"greenContrastEnhancement"_s );
216 if ( !greenContrastElem.isNull() )
217 {
219 greenContrastEnhancement->readXml( greenContrastElem );
220 r->setGreenContrastEnhancement( greenContrastEnhancement );
221 }
222
224 const QDomElement blueContrastElem = element.firstChildElement( u"blueContrastEnhancement"_s );
225 if ( !blueContrastElem.isNull() )
226 {
228 blueContrastEnhancement->readXml( blueContrastElem );
229 r->setBlueContrastEnhancement( blueContrastEnhancement );
230 }
231
232 return r.release();
233}
234
235QDomElement QgsPointCloudRgbRenderer::save( QDomDocument &doc, const QgsReadWriteContext &context ) const
236{
237 QDomElement rendererElem = doc.createElement( u"renderer"_s );
238
239 rendererElem.setAttribute( u"type"_s, u"rgb"_s );
240
241 rendererElem.setAttribute( u"red"_s, mRedAttribute );
242 rendererElem.setAttribute( u"green"_s, mGreenAttribute );
243 rendererElem.setAttribute( u"blue"_s, mBlueAttribute );
244
245 saveCommonProperties( rendererElem, context );
246
247 //contrast enhancement
248 if ( mRedContrastEnhancement )
249 {
250 QDomElement redContrastElem = doc.createElement( u"redContrastEnhancement"_s );
251 mRedContrastEnhancement->writeXml( doc, redContrastElem );
252 rendererElem.appendChild( redContrastElem );
253 }
254 if ( mGreenContrastEnhancement )
255 {
256 QDomElement greenContrastElem = doc.createElement( u"greenContrastEnhancement"_s );
257 mGreenContrastEnhancement->writeXml( doc, greenContrastElem );
258 rendererElem.appendChild( greenContrastElem );
259 }
260 if ( mBlueContrastEnhancement )
261 {
262 QDomElement blueContrastElem = doc.createElement( u"blueContrastEnhancement"_s );
263 mBlueContrastEnhancement->writeXml( doc, blueContrastElem );
264 rendererElem.appendChild( blueContrastElem );
265 }
266
267 return rendererElem;
268}
269
271{
272 QSet<QString> res;
273 res << mRedAttribute << mGreenAttribute << mBlueAttribute;
274 return res;
275}
276
277std::unique_ptr<QgsPreparedPointCloudRendererData> QgsPointCloudRgbRenderer::prepare()
278{
279 auto data = std::make_unique< QgsPointCloudRgbRendererPreparedData >();
280 data->redAttribute = mRedAttribute;
281 if ( mRedContrastEnhancement )
282 data->redContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mRedContrastEnhancement );
283 data->greenAttribute = mGreenAttribute;
284 if ( mGreenContrastEnhancement )
285 data->greenContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mGreenContrastEnhancement );
286 data->blueAttribute = mBlueAttribute;
287 if ( mBlueContrastEnhancement )
288 data->blueContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mBlueContrastEnhancement );
289
290 data->useRedContrastEnhancement = mRedContrastEnhancement && mRedContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
291 data->useBlueContrastEnhancement = mBlueContrastEnhancement && mBlueContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
292 data->useGreenContrastEnhancement = mGreenContrastEnhancement && mGreenContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
293
294 return data;
295}
296
301
303{
304 const QgsPointCloudAttributeCollection request = block->attributes();
305 redOffset = 0;
306 const QgsPointCloudAttribute *attribute = request.find( redAttribute, redOffset );
307 if ( !attribute )
308 return false;
309 redType = attribute->type();
310
311 greenOffset = 0;
312 attribute = request.find( greenAttribute, greenOffset );
313 if ( !attribute )
314 return false;
315 greenType = attribute->type();
316
317 blueOffset = 0;
318 attribute = request.find( blueAttribute, blueOffset );
319 if ( !attribute )
320 return false;
321 blueType = attribute->type();
322 return true;
323}
324
326{
327 const char *ptr = block->data();
328 const int pointRecordSize = block->pointRecordSize();
329
330 int red = 0;
331 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + redOffset, redType, red );
332 int green = 0;
333 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + greenOffset, greenType, green );
334 int blue = 0;
335 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + blueOffset, blueType, blue );
336
337 //skip if red, green or blue not in displayable range
338 if ( ( useRedContrastEnhancement && !redContrastEnhancement->isValueInDisplayableRange( red ) )
339 || ( useGreenContrastEnhancement && !greenContrastEnhancement->isValueInDisplayableRange( green ) )
340 || ( useBlueContrastEnhancement && !blueContrastEnhancement->isValueInDisplayableRange( blue ) ) )
341 {
342 return QColor();
343 }
344
345 //stretch color values
347 {
348 red = redContrastEnhancement->enhanceContrast( red );
349 }
351 {
352 green = greenContrastEnhancement->enhanceContrast( green );
353 }
355 {
356 blue = blueContrastEnhancement->enhanceContrast( blue );
357 }
358
359 red = std::max( 0, std::min( 255, red ) );
360 green = std::max( 0, std::min( 255, green ) );
361 blue = std::max( 0, std::min( 255, blue ) );
362
363 return QColor( red, green, blue );
364}
365
367{
368 return mRedAttribute;
369}
370
372{
373 mRedAttribute = redAttribute;
374}
375
377{
378 return mGreenAttribute;
379}
380
382{
383 mGreenAttribute = greenAttribute;
384}
385
387{
388 return mBlueAttribute;
389}
390
392{
393 mBlueAttribute = blueAttribute;
394}
395
397{
398 return mRedContrastEnhancement.get();
399}
400
402{
403 mRedContrastEnhancement.reset( enhancement );
404}
405
407{
408 return mGreenContrastEnhancement.get();
409}
410
412{
413 mGreenContrastEnhancement.reset( enhancement );
414}
415
417{
418 return mBlueContrastEnhancement.get();
419}
420
422{
423 mBlueContrastEnhancement.reset( enhancement );
424}
@ UnknownDataType
Unknown or unspecified type.
Definition qgis.h:380
Handles contrast enhancement and clipping.
@ NoEnhancement
Default color scaling algorithm, no scaling is applied.
Handles coordinate transforms between two coordinate systems.
void transformInPlace(double &x, double &y, double &z, Qgis::TransformDirection direction=Qgis::TransformDirection::Forward) const
Transforms an array of x, y and z double coordinates in place, from the source CRS to the destination...
bool isValid() const
Returns true if the coordinate transform is valid, ie both the source and destination CRS have been s...
Custom exception class for Coordinate Reference System related exceptions.
QgsRange which stores a range of double values.
Definition qgsrange.h:236
bool isInfinite() const
Returns true if the range consists of all possible values.
Definition qgsrange.h:290
A collection of point cloud attributes.
int pointRecordSize() const
Returns total size of record.
const QgsPointCloudAttribute * find(const QString &attributeName, int &offset) const
Finds the attribute with the name.
Attribute for point cloud data pair of name and size in bytes.
DataType
Systems of unit measurement.
DataType type() const
Returns the data type.
Base class for storing raw data from point cloud nodes.
const char * data() const
Returns raw pointer to data.
QgsPointCloudAttributeCollection attributes() const
Returns the attributes that are stored in the data block, along with their size.
int pointCount() const
Returns number of points that are stored in the block.
int pointRecordSize() const
Returns the total size of each individual point record.
Encapsulates the render context for a 2D point cloud rendering operation.
QgsRenderContext & renderContext()
Returns a reference to the context's render context.
void incrementPointsRendered(long count)
Increments the count of points rendered by the specified amount.
static void getAttribute(const char *data, std::size_t offset, QgsPointCloudAttribute::DataType type, T &value)
Retrieves the attribute value from data at the specified offset, where type indicates the original da...
bool renderAsTriangles() const
Returns whether points are triangulated to render solid surface.
void drawPointToElevationMap(double x, double y, double z, QgsPointCloudRenderContext &context) const
Draws a point at the elevation z using at the specified x and y (in map coordinates) on the elevation...
void saveCommonProperties(QDomElement &element, const QgsReadWriteContext &context) const
Saves common renderer properties (such as point size and screen error) to the specified DOM element.
void addPointToTriangulation(double x, double y, double z, const QColor &color, QgsPointCloudRenderContext &context)
Adds a point to the list of points to be triangulated (only used when renderAsTriangles() is enabled)...
void copyCommonProperties(QgsPointCloudRenderer *destination) const
Copies common point cloud properties (such as point size and screen error) to the destination rendere...
void drawPoint(double x, double y, const QColor &color, QgsPointCloudRenderContext &context) const
Draws a point using a color at the specified x and y (in map coordinates).
static double pointZ(QgsPointCloudRenderContext &context, const char *ptr, int i)
Retrieves the z value for the point at index i.
static void pointXY(QgsPointCloudRenderContext &context, const char *ptr, int i, double &x, double &y)
Retrieves the x and y coordinate for the point at index i.
std::unique_ptr< QgsContrastEnhancement > redContrastEnhancement
QSet< QString > usedAttributes() const override
Returns the set of attributes used by the prepared point cloud renderer.
std::unique_ptr< QgsContrastEnhancement > greenContrastEnhancement
bool prepareBlock(const QgsPointCloudBlock *block) override
Prepares the renderer for using the specified block.
QColor pointColor(const QgsPointCloudBlock *block, int i, double z) override
An optimised method of retrieving the color of a point from a point cloud block.
std::unique_ptr< QgsContrastEnhancement > blueContrastEnhancement
QgsPointCloudAttribute::DataType blueType
QgsPointCloudAttribute::DataType redType
QgsPointCloudAttribute::DataType greenType
std::unique_ptr< QgsPreparedPointCloudRendererData > prepare() override
Returns prepared data container for bulk point color retrieval.
void setRedContrastEnhancement(QgsContrastEnhancement *enhancement)
Sets the contrast enhancement to use for the red channel.
QString redAttribute() const
Returns the attribute to use for the red channel.
static QgsPointCloudRenderer * create(QDomElement &element, const QgsReadWriteContext &context)
Creates an RGB renderer from an XML element.
QString greenAttribute() const
Returns the attribute to use for the green channel.
QString type() const override
Returns the identifier of the renderer type.
QDomElement save(QDomDocument &doc, const QgsReadWriteContext &context) const override
Saves the renderer configuration to an XML element.
void setBlueContrastEnhancement(QgsContrastEnhancement *enhancement)
Sets the contrast enhancement to use for the blue channel.
void renderBlock(const QgsPointCloudBlock *block, QgsPointCloudRenderContext &context) override
Renders a block of point cloud data using the specified render context.
const QgsContrastEnhancement * greenContrastEnhancement() const
Returns the contrast enhancement to use for the green channel.
QString blueAttribute() const
Returns the attribute to use for the blue channel.
QgsPointCloudRgbRenderer()
Constructor for QgsPointCloudRgbRenderer.
void setGreenContrastEnhancement(QgsContrastEnhancement *enhancement)
Sets the contrast enhancement to use for the green channel.
void setBlueAttribute(const QString &attribute)
Sets the attribute to use for the blue channel.
const QgsContrastEnhancement * blueContrastEnhancement() const
Returns the contrast enhancement to use for the blue channel.
void setGreenAttribute(const QString &attribute)
Sets the attribute to use for the green channel.
void setRedAttribute(const QString &attribute)
Sets the attribute to use for the red channel.
QgsPointCloudRenderer * clone() const override
Create a deep copy of this renderer.
const QgsContrastEnhancement * redContrastEnhancement() const
Returns the contrast enhancement to use for the red channel.
QSet< QString > usedAttributes(const QgsPointCloudRenderContext &context) const override
Returns a list of attributes required by this renderer.
bool contains(const QgsRange< T > &other) const
Returns true if this range contains another range.
Definition qgsrange.h:140
A container for the context for various read/write operations on objects.
A rectangle specified with double values.
bool contains(const QgsRectangle &rect) const
Returns true when rectangle contains other rectangle.
void grow(double delta)
Grows the rectangle in place by the specified amount.
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...
QgsDoubleRange zRange() const
Returns the range of z-values which should be rendered.
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.