QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
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
31
33{
34 return u"rgb"_s;
35}
36
38{
39 auto res = std::make_unique< QgsPointCloudRgbRenderer >();
40 res->mRedAttribute = mRedAttribute;
41 res->mGreenAttribute = mGreenAttribute;
42 res->mBlueAttribute = mBlueAttribute;
43
44 if ( mRedContrastEnhancement )
45 {
46 res->setRedContrastEnhancement( new QgsContrastEnhancement( *mRedContrastEnhancement ) );
47 }
48 if ( mGreenContrastEnhancement )
49 {
50 res->setGreenContrastEnhancement( new QgsContrastEnhancement( *mGreenContrastEnhancement ) );
51 }
52 if ( mBlueContrastEnhancement )
53 {
54 res->setBlueContrastEnhancement( new QgsContrastEnhancement( *mBlueContrastEnhancement ) );
55 }
56
57 copyCommonProperties( res.get() );
58
59 return res.release();
60}
61
63{
64 QgsRectangle visibleExtent = context.renderContext().extent();
65 if ( renderAsTriangles() )
66 {
67 // we need to include also points slightly outside of the visible extent,
68 // otherwise the triangulation may be missing triangles near the edges and corners
69 visibleExtent.grow( std::max( visibleExtent.width(), visibleExtent.height() ) * 0.05 );
70 }
71
72 const char *ptr = block->data();
73 const int count = block->pointCount();
74 const QgsPointCloudAttributeCollection request = block->attributes();
75
76 const std::size_t recordSize = request.pointRecordSize();
77 int redOffset = 0;
78 const QgsPointCloudAttribute *attribute = request.find( mRedAttribute, redOffset );
79 if ( !attribute )
80 return;
81 const QgsPointCloudAttribute::DataType redType = attribute->type();
82
83 int greenOffset = 0;
84 attribute = request.find( mGreenAttribute, greenOffset );
85 if ( !attribute )
86 return;
87 const QgsPointCloudAttribute::DataType greenType = attribute->type();
88
89 int blueOffset = 0;
90 attribute = request.find( mBlueAttribute, blueOffset );
91 if ( !attribute )
92 return;
93 const QgsPointCloudAttribute::DataType blueType = attribute->type();
94
95 const bool useRedContrastEnhancement = mRedContrastEnhancement && mRedContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
96 const bool useBlueContrastEnhancement = mBlueContrastEnhancement && mBlueContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
97 const bool useGreenContrastEnhancement = mGreenContrastEnhancement && mGreenContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
98
99 const bool renderElevation = context.renderContext().elevationMap();
100 const QgsDoubleRange zRange = context.renderContext().zRange();
101 const bool considerZ = !zRange.isInfinite() || renderElevation;
102
103 int rendered = 0;
104 double x = 0;
105 double y = 0;
106 double z = 0;
108 const bool reproject = ct.isValid();
109 for ( int i = 0; i < count; ++i )
110 {
111 if ( context.renderContext().renderingStopped() )
112 {
113 break;
114 }
115
116 if ( considerZ )
117 {
118 // z value filtering is cheapest, if we're doing it...
119 z = pointZ( context, ptr, i );
120 if ( !zRange.contains( z ) )
121 continue;
122 }
123
124 pointXY( context, ptr, i, x, y );
125 if ( visibleExtent.contains( x, y ) )
126 {
127 if ( reproject )
128 {
129 try
130 {
131 ct.transformInPlace( x, y, z );
132 }
133 catch ( QgsCsException & )
134 {
135 continue;
136 }
137 }
138
139 int red = 0;
140 context.getAttribute( ptr, i * recordSize + redOffset, redType, red );
141 int green = 0;
142 context.getAttribute( ptr, i * recordSize + greenOffset, greenType, green );
143 int blue = 0;
144 context.getAttribute( ptr, i * recordSize + blueOffset, blueType, blue );
145
146 //skip if red, green or blue not in displayable range
147 if ( ( useRedContrastEnhancement && !mRedContrastEnhancement->isValueInDisplayableRange( red ) )
148 || ( useGreenContrastEnhancement && !mGreenContrastEnhancement->isValueInDisplayableRange( green ) )
149 || ( useBlueContrastEnhancement && !mBlueContrastEnhancement->isValueInDisplayableRange( blue ) ) )
150 {
151 continue;
152 }
153
154 //stretch color values
155 if ( useRedContrastEnhancement )
156 {
157 red = mRedContrastEnhancement->enhanceContrast( red );
158 }
159 if ( useGreenContrastEnhancement )
160 {
161 green = mGreenContrastEnhancement->enhanceContrast( green );
162 }
163 if ( useBlueContrastEnhancement )
164 {
165 blue = mBlueContrastEnhancement->enhanceContrast( blue );
166 }
167
168 red = std::max( 0, std::min( 255, red ) );
169 green = std::max( 0, std::min( 255, green ) );
170 blue = std::max( 0, std::min( 255, blue ) );
171
172 if ( renderAsTriangles() )
173 {
174 addPointToTriangulation( x, y, z, QColor( red, green, blue ), context );
175
176 // We don't want to render any points if we're rendering triangles and there is no preview painter
177 if ( !context.renderContext().previewRenderPainter() )
178 continue;
179 }
180
181 drawPoint( x, y, QColor( red, green, blue ), context );
182 if ( renderElevation )
183 drawPointToElevationMap( x, y, z, context );
184
185 rendered++;
186 }
187 }
188 context.incrementPointsRendered( rendered );
189}
190
191
193{
194 auto r = std::make_unique< QgsPointCloudRgbRenderer >();
195
196 r->setRedAttribute( element.attribute( u"red"_s, u"Red"_s ) );
197 r->setGreenAttribute( element.attribute( u"green"_s, u"Green"_s ) );
198 r->setBlueAttribute( element.attribute( u"blue"_s, u"Blue"_s ) );
199
200 r->restoreCommonProperties( element, context );
201
202 //contrast enhancements
204 const QDomElement redContrastElem = element.firstChildElement( u"redContrastEnhancement"_s );
205 if ( !redContrastElem.isNull() )
206 {
208 redContrastEnhancement->readXml( redContrastElem );
209 r->setRedContrastEnhancement( redContrastEnhancement );
210 }
211
213 const QDomElement greenContrastElem = element.firstChildElement( u"greenContrastEnhancement"_s );
214 if ( !greenContrastElem.isNull() )
215 {
217 greenContrastEnhancement->readXml( greenContrastElem );
218 r->setGreenContrastEnhancement( greenContrastEnhancement );
219 }
220
222 const QDomElement blueContrastElem = element.firstChildElement( u"blueContrastEnhancement"_s );
223 if ( !blueContrastElem.isNull() )
224 {
226 blueContrastEnhancement->readXml( blueContrastElem );
227 r->setBlueContrastEnhancement( blueContrastEnhancement );
228 }
229
230 return r.release();
231}
232
233QDomElement QgsPointCloudRgbRenderer::save( QDomDocument &doc, const QgsReadWriteContext &context ) const
234{
235 QDomElement rendererElem = doc.createElement( u"renderer"_s );
236
237 rendererElem.setAttribute( u"type"_s, u"rgb"_s );
238
239 rendererElem.setAttribute( u"red"_s, mRedAttribute );
240 rendererElem.setAttribute( u"green"_s, mGreenAttribute );
241 rendererElem.setAttribute( u"blue"_s, mBlueAttribute );
242
243 saveCommonProperties( rendererElem, context );
244
245 //contrast enhancement
246 if ( mRedContrastEnhancement )
247 {
248 QDomElement redContrastElem = doc.createElement( u"redContrastEnhancement"_s );
249 mRedContrastEnhancement->writeXml( doc, redContrastElem );
250 rendererElem.appendChild( redContrastElem );
251 }
252 if ( mGreenContrastEnhancement )
253 {
254 QDomElement greenContrastElem = doc.createElement( u"greenContrastEnhancement"_s );
255 mGreenContrastEnhancement->writeXml( doc, greenContrastElem );
256 rendererElem.appendChild( greenContrastElem );
257 }
258 if ( mBlueContrastEnhancement )
259 {
260 QDomElement blueContrastElem = doc.createElement( u"blueContrastEnhancement"_s );
261 mBlueContrastEnhancement->writeXml( doc, blueContrastElem );
262 rendererElem.appendChild( blueContrastElem );
263 }
264
265 return rendererElem;
266}
267
269{
270 QSet<QString> res;
271 res << mRedAttribute << mGreenAttribute << mBlueAttribute;
272 return res;
273}
274
275std::unique_ptr<QgsPreparedPointCloudRendererData> QgsPointCloudRgbRenderer::prepare()
276{
277 auto data = std::make_unique< QgsPointCloudRgbRendererPreparedData >();
278 data->redAttribute = mRedAttribute;
279 if ( mRedContrastEnhancement )
280 data->redContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mRedContrastEnhancement );
281 data->greenAttribute = mGreenAttribute;
282 if ( mGreenContrastEnhancement )
283 data->greenContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mGreenContrastEnhancement );
284 data->blueAttribute = mBlueAttribute;
285 if ( mBlueContrastEnhancement )
286 data->blueContrastEnhancement = std::make_unique<QgsContrastEnhancement>( *mBlueContrastEnhancement );
287
288 data->useRedContrastEnhancement = mRedContrastEnhancement && mRedContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
289 data->useBlueContrastEnhancement = mBlueContrastEnhancement && mBlueContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
290 data->useGreenContrastEnhancement = mGreenContrastEnhancement && mGreenContrastEnhancement->contrastEnhancementAlgorithm() != QgsContrastEnhancement::NoEnhancement;
291
292 return data;
293}
294
299
301{
302 const QgsPointCloudAttributeCollection request = block->attributes();
303 redOffset = 0;
304 const QgsPointCloudAttribute *attribute = request.find( redAttribute, redOffset );
305 if ( !attribute )
306 return false;
307 redType = attribute->type();
308
309 greenOffset = 0;
310 attribute = request.find( greenAttribute, greenOffset );
311 if ( !attribute )
312 return false;
313 greenType = attribute->type();
314
315 blueOffset = 0;
316 attribute = request.find( blueAttribute, blueOffset );
317 if ( !attribute )
318 return false;
319 blueType = attribute->type();
320 return true;
321}
322
324{
325 const char *ptr = block->data();
326 const int pointRecordSize = block->pointRecordSize();
327
328 int red = 0;
329 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + redOffset, redType, red );
330 int green = 0;
331 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + greenOffset, greenType, green );
332 int blue = 0;
333 QgsPointCloudRenderContext::getAttribute( ptr, i * pointRecordSize + blueOffset, blueType, blue );
334
335 //skip if red, green or blue not in displayable range
336 if ( ( useRedContrastEnhancement && !redContrastEnhancement->isValueInDisplayableRange( red ) )
337 || ( useGreenContrastEnhancement && !greenContrastEnhancement->isValueInDisplayableRange( green ) )
338 || ( useBlueContrastEnhancement && !blueContrastEnhancement->isValueInDisplayableRange( blue ) ) )
339 {
340 return QColor();
341 }
342
343 //stretch color values
345 {
346 red = redContrastEnhancement->enhanceContrast( red );
347 }
349 {
350 green = greenContrastEnhancement->enhanceContrast( green );
351 }
353 {
354 blue = blueContrastEnhancement->enhanceContrast( blue );
355 }
356
357 red = std::max( 0, std::min( 255, red ) );
358 green = std::max( 0, std::min( 255, green ) );
359 blue = std::max( 0, std::min( 255, blue ) );
360
361 return QColor( red, green, blue );
362}
363
365{
366 return mRedAttribute;
367}
368
370{
371 mRedAttribute = redAttribute;
372}
373
375{
376 return mGreenAttribute;
377}
378
380{
381 mGreenAttribute = greenAttribute;
382}
383
385{
386 return mBlueAttribute;
387}
388
390{
391 mBlueAttribute = blueAttribute;
392}
393
395{
396 return mRedContrastEnhancement.get();
397}
398
400{
401 mRedContrastEnhancement.reset( enhancement );
402}
403
405{
406 return mGreenContrastEnhancement.get();
407}
408
410{
411 mGreenContrastEnhancement.reset( enhancement );
412}
413
415{
416 return mBlueContrastEnhancement.get();
417}
418
420{
421 mBlueContrastEnhancement.reset( enhancement );
422}
@ UnknownDataType
Unknown or unspecified type.
Definition qgis.h:394
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:217
bool isInfinite() const
Returns true if the range consists of all possible values.
Definition qgsrange.h:266
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:138
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.