QGIS API Documentation 3.43.0-Master (ac54a16a525)
qgs3dmapscene.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgs3dmapscene.cpp
3 --------------------------------------
4 Date : July 2017
5 Copyright : (C) 2017 by Martin Dobias
6 Email : wonder dot sk at gmail dot com
7 ***************************************************************************
8 * *
9 * This program is free software; you can redistribute it and/or modify *
10 * it under the terms of the GNU General Public License as published by *
11 * the Free Software Foundation; either version 2 of the License, or *
12 * (at your option) any later version. *
13 * *
14 ***************************************************************************/
15
16#include "qgs3dmapscene.h"
17#include "moc_qgs3dmapscene.cpp"
18
19#include <Qt3DRender/QCamera>
20#include <Qt3DRender/QMesh>
21#include <Qt3DRender/QRenderSettings>
22#include <Qt3DRender/QSceneLoader>
23#include <Qt3DExtras/QForwardRenderer>
24#include <Qt3DExtras/QPhongMaterial>
25#include <Qt3DExtras/QPhongAlphaMaterial>
26#include <Qt3DExtras/QDiffuseSpecularMaterial>
27#include <Qt3DExtras/QSphereMesh>
28#include <Qt3DLogic/QFrameAction>
29#include <Qt3DRender/QEffect>
30#include <Qt3DRender/QTechnique>
31#include <Qt3DRender/QRenderPass>
32#include <Qt3DRender/QRenderState>
33#include <Qt3DRender/QCullFace>
34#include <Qt3DRender/QDepthTest>
35#include <QSurface>
36#include <QUrl>
37#include <QtMath>
38
39#include <QOpenGLContext>
40#include <QOpenGLFunctions>
41#include <QTimer>
42
43#include "qgs3daxis.h"
44#include "qgslogger.h"
45#include "qgsapplication.h"
46#include "qgsaabb.h"
47#include "qgsabstract3dengine.h"
48#include "qgs3dmapsettings.h"
49#include "qgs3dutils.h"
51#include "qgscameracontroller.h"
52#include "qgschunkedentity.h"
53#include "qgschunknode.h"
54#include "qgseventtracing.h"
55#include "qgsgeotransform.h"
57#include "qgsmaterial.h"
58#include "qgsmeshlayer.h"
60#include "qgspoint3dsymbol.h"
62#include "qgspointcloudlayer.h"
64#include "qgssourcecache.h"
65#include "qgsterrainentity.h"
66#include "qgsterraingenerator.h"
67#include "qgstiledscenelayer.h"
70#include "qgsvectorlayer.h"
75#include "qgslinematerial_p.h"
76#include "qgs3dsceneexporter.h"
78#include "qgsmessageoutput.h"
79#include "qgsframegraph.h"
81
82#include "qgsskyboxentity.h"
83#include "qgsskyboxsettings.h"
84
85#include "qgswindow3dengine.h"
86#include "qgspointcloudlayer.h"
87#include "qgsshadowrenderview.h"
89
90std::function<QMap<QString, Qgs3DMapScene *>()> Qgs3DMapScene::sOpenScenesFunction = [] { return QMap<QString, Qgs3DMapScene *>(); };
91
93 : mMap( map )
94 , mEngine( engine )
95{
96 connect( &map, &Qgs3DMapSettings::backgroundColorChanged, this, &Qgs3DMapScene::onBackgroundColorChanged );
97 onBackgroundColorChanged();
98
99 // The default render policy in Qt3D is "Always" - i.e. the 3D map scene gets refreshed up to 60 fps
100 // even if there's no change. Switching to "on demand" should only re-render when something has changed
101 // and we save quite a lot of resources
102 mEngine->renderSettings()->setRenderPolicy( Qt3DRender::QRenderSettings::OnDemand );
103
104 QRect viewportRect( QPoint( 0, 0 ), mEngine->size() );
105
106 // Get the maximum of clip planes available
107 mMaxClipPlanes = Qgs3DUtils::openGlMaxClipPlanes( mEngine->surface() );
108
109 // Camera
110 float aspectRatio = ( float ) viewportRect.width() / viewportRect.height();
111 mEngine->camera()->lens()->setPerspectiveProjection( mMap.fieldOfView(), aspectRatio, 10.f, 10000.0f );
112
113 mFrameAction = new Qt3DLogic::QFrameAction();
114 connect( mFrameAction, &Qt3DLogic::QFrameAction::triggered, this, &Qgs3DMapScene::onFrameTriggered );
115 addComponent( mFrameAction ); // takes ownership
116
117 // Camera controlling
118 mCameraController = new QgsCameraController( this ); // attaches to the scene
119
120 if ( mMap.sceneMode() == Qgis::SceneMode::Globe )
121 mCameraController->resetGlobe( 10'000'000 );
122 else
123 mCameraController->resetView( 1000 );
124
125 addCameraViewCenterEntity( mEngine->camera() );
126 addCameraRotationCenterEntity( mCameraController );
127 updateLights();
128
129 // create terrain entity
130
131 createTerrainDeferred();
132 connect( &map, &Qgs3DMapSettings::extentChanged, this, &Qgs3DMapScene::createTerrain );
133 connect( &map, &Qgs3DMapSettings::terrainGeneratorChanged, this, &Qgs3DMapScene::createTerrain );
134
135 connect( &map, &Qgs3DMapSettings::terrainSettingsChanged, this, &Qgs3DMapScene::createTerrain );
136
137 connect( &map, &Qgs3DMapSettings::terrainShadingChanged, this, &Qgs3DMapScene::createTerrain );
138 connect( &map, &Qgs3DMapSettings::lightSourcesChanged, this, &Qgs3DMapScene::updateLights );
139 connect( &map, &Qgs3DMapSettings::showLightSourceOriginsChanged, this, &Qgs3DMapScene::updateLights );
140 connect( &map, &Qgs3DMapSettings::fieldOfViewChanged, this, &Qgs3DMapScene::updateCameraLens );
141 connect( &map, &Qgs3DMapSettings::projectionTypeChanged, this, &Qgs3DMapScene::updateCameraLens );
142 connect( &map, &Qgs3DMapSettings::skyboxSettingsChanged, this, &Qgs3DMapScene::onSkyboxSettingsChanged );
143 connect( &map, &Qgs3DMapSettings::shadowSettingsChanged, this, &Qgs3DMapScene::onShadowSettingsChanged );
144 connect( &map, &Qgs3DMapSettings::ambientOcclusionSettingsChanged, this, &Qgs3DMapScene::onAmbientOcclusionSettingsChanged );
145 connect( &map, &Qgs3DMapSettings::eyeDomeLightingEnabledChanged, this, &Qgs3DMapScene::onEyeDomeShadingSettingsChanged );
146 connect( &map, &Qgs3DMapSettings::eyeDomeLightingStrengthChanged, this, &Qgs3DMapScene::onEyeDomeShadingSettingsChanged );
147 connect( &map, &Qgs3DMapSettings::eyeDomeLightingDistanceChanged, this, &Qgs3DMapScene::onEyeDomeShadingSettingsChanged );
148 connect( &map, &Qgs3DMapSettings::debugShadowMapSettingsChanged, this, &Qgs3DMapScene::onDebugShadowMapSettingsChanged );
149 connect( &map, &Qgs3DMapSettings::debugDepthMapSettingsChanged, this, &Qgs3DMapScene::onDebugDepthMapSettingsChanged );
151 connect( &map, &Qgs3DMapSettings::cameraMovementSpeedChanged, this, &Qgs3DMapScene::onCameraMovementSpeedChanged );
152 connect( &map, &Qgs3DMapSettings::cameraNavigationModeChanged, this, &Qgs3DMapScene::onCameraNavigationModeChanged );
153 connect( &map, &Qgs3DMapSettings::debugOverlayEnabledChanged, this, &Qgs3DMapScene::onDebugOverlayEnabledChanged );
154 connect( &map, &Qgs3DMapSettings::stopUpdatesChanged, this, &Qgs3DMapScene::onStopUpdatesChanged );
155
156 connect( &map, &Qgs3DMapSettings::axisSettingsChanged, this, &Qgs3DMapScene::on3DAxisSettingsChanged );
157
158 connect( &map, &Qgs3DMapSettings::originChanged, this, &Qgs3DMapScene::onOriginChanged );
159
160 connect( QgsApplication::sourceCache(), &QgsSourceCache::remoteSourceFetched, this, [=]( const QString &url ) {
161 const QList<QgsMapLayer *> modelVectorLayers = mModelVectorLayers;
162 for ( QgsMapLayer *layer : modelVectorLayers )
163 {
164 QgsAbstract3DRenderer *renderer = layer->renderer3D();
165 if ( renderer )
166 {
167 if ( renderer->type() == QLatin1String( "vector" ) )
168 {
169 const QgsPoint3DSymbol *pointSymbol = static_cast<const QgsPoint3DSymbol *>( static_cast<QgsVectorLayer3DRenderer *>( renderer )->symbol() );
170 if ( pointSymbol->shapeProperty( QStringLiteral( "model" ) ).toString() == url )
171 {
172 removeLayerEntity( layer );
173 addLayerEntity( layer );
174 }
175 }
176 else if ( renderer->type() == QLatin1String( "rulebased" ) )
177 {
178 const QgsRuleBased3DRenderer::RuleList rules = static_cast<QgsRuleBased3DRenderer *>( renderer )->rootRule()->descendants();
179 for ( auto rule : rules )
180 {
181 const QgsPoint3DSymbol *pointSymbol = dynamic_cast<const QgsPoint3DSymbol *>( rule->symbol() );
182 if ( pointSymbol->shapeProperty( QStringLiteral( "model" ) ).toString() == url )
183 {
184 removeLayerEntity( layer );
185 addLayerEntity( layer );
186 break;
187 }
188 }
189 }
190 }
191 }
192 } );
193
194 // listen to changes of layers in order to add/remove 3D renderer entities
195 connect( &map, &Qgs3DMapSettings::layersChanged, this, &Qgs3DMapScene::onLayersChanged );
196
197 connect( mCameraController, &QgsCameraController::cameraChanged, this, &Qgs3DMapScene::onCameraChanged );
198 connect( mEngine, &QgsAbstract3DEngine::sizeChanged, this, &Qgs3DMapScene::onCameraChanged );
199
200 onSkyboxSettingsChanged();
201
202 // force initial update of chunked entities
203 onCameraChanged();
204 // force initial update of eye dome shading
205 onEyeDomeShadingSettingsChanged();
206 // force initial update of debugging setting of preview quads
207 onDebugShadowMapSettingsChanged();
208 onDebugDepthMapSettingsChanged();
209 // force initial update of ambient occlusion settings
210 onAmbientOcclusionSettingsChanged();
211
212 onCameraMovementSpeedChanged();
213
214 on3DAxisSettingsChanged();
215}
216
218{
219 if ( mMap.sceneMode() == Qgis::SceneMode::Globe )
220 {
221 mCameraController->resetGlobe( 10'000'000 );
222 return;
223 }
224
225 const QgsDoubleRange zRange = elevationRange();
226 const QgsRectangle extent = sceneExtent();
227 const double side = std::max( extent.width(), extent.height() );
228 double d = side / 2 / std::tan( cameraController()->camera()->fieldOfView() / 2 * M_PI / 180 );
229 d += zRange.isInfinite() ? 0. : zRange.upper();
230 mCameraController->resetView( static_cast<float>( d ) );
231}
232
234{
235 QgsPointXY center = extent.center();
236 QgsVector3D centerWorld = mMap.mapToWorldCoordinates( QVector3D( center.x(), center.y(), 0 ) );
237 QgsVector3D p1 = mMap.mapToWorldCoordinates( QVector3D( extent.xMinimum(), extent.yMinimum(), 0 ) );
238 QgsVector3D p2 = mMap.mapToWorldCoordinates( QVector3D( extent.xMaximum(), extent.yMaximum(), 0 ) );
239
240 float xSide = std::abs( p1.x() - p2.x() );
241 float ySide = std::abs( p1.z() - p2.z() );
242 if ( xSide < ySide )
243 {
244 float fov = 2 * std::atan( std::tan( qDegreesToRadians( cameraController()->camera()->fieldOfView() ) / 2 ) * cameraController()->camera()->aspectRatio() );
245 float r = xSide / 2.0f / std::tan( fov / 2.0f );
246 mCameraController->setViewFromTop( centerWorld.x(), centerWorld.z(), r );
247 }
248 else
249 {
250 float fov = qDegreesToRadians( cameraController()->camera()->fieldOfView() );
251 float r = ySide / 2.0f / std::tan( fov / 2.0f );
252 mCameraController->setViewFromTop( centerWorld.x(), centerWorld.z(), r );
253 }
254}
255
256QVector<QgsPointXY> Qgs3DMapScene::viewFrustum2DExtent() const
257{
258 Qt3DRender::QCamera *camera = mCameraController->camera();
259 QVector<QgsPointXY> extent;
260 QVector<int> pointsOrder = { 0, 1, 3, 2 };
261 for ( int i : pointsOrder )
262 {
263 const QPoint p( ( ( i >> 0 ) & 1 ) ? 0 : mEngine->size().width(), ( ( i >> 1 ) & 1 ) ? 0 : mEngine->size().height() );
264 QgsRay3D ray = Qgs3DUtils::rayFromScreenPoint( p, mEngine->size(), camera );
265 QVector3D dir = ray.direction();
266 if ( dir.z() == 0.0 )
267 dir.setZ( 0.000001 );
268 double t = -ray.origin().z() / dir.z();
269 if ( t < 0 )
270 {
271 // If the projected point is on the back of the camera we choose the farthest point in the front
272 t = camera->farPlane();
273 }
274 else
275 {
276 // If the projected point is on the front of the camera we choose the closest between it and farthest point in the front
277 t = std::min<float>( t, camera->farPlane() );
278 }
279 QVector3D planePoint = ray.origin() + t * dir;
280 QgsVector3D pMap = mMap.worldToMapCoordinates( planePoint );
281 extent.push_back( QgsPointXY( pMap.x(), pMap.y() ) );
282 }
283 return extent;
284}
285
287{
288 int count = 0;
289 for ( Qgs3DMapSceneEntity *entity : std::as_const( mSceneEntities ) )
290 count += entity->pendingJobsCount();
291 return count;
292}
293
294float Qgs3DMapScene::worldSpaceError( float epsilon, float distance ) const
295{
296 Qt3DRender::QCamera *camera = mCameraController->camera();
297 float fov = camera->fieldOfView();
298 const QSize size = mEngine->size();
299 float screenSizePx = std::max( size.width(), size.height() ); // TODO: is this correct?
300
301 // see Qgs3DUtils::screenSpaceError() for the inverse calculation (world space error to screen space error)
302 // with explanation of the math.
303 float frustumWidthAtDistance = 2 * distance * tan( fov / 2 );
304 float err = frustumWidthAtDistance * epsilon / screenSizePx;
305 return err;
306}
307
308void Qgs3DMapScene::onCameraChanged()
309{
310 if ( mMap.projectionType() == Qt3DRender::QCameraLens::OrthographicProjection )
311 {
312 QRect viewportRect( QPoint( 0, 0 ), mEngine->size() );
313 const float viewWidthFromCenter = mCameraController->distance();
314 const float viewHeightFromCenter = viewportRect.height() * viewWidthFromCenter / viewportRect.width();
315 mEngine->camera()->lens()->setOrthographicProjection( -viewWidthFromCenter, viewWidthFromCenter, -viewHeightFromCenter, viewHeightFromCenter, mEngine->camera()->nearPlane(), mEngine->camera()->farPlane() );
316 }
317
318 updateScene( true );
319 updateCameraNearFarPlanes();
320
321 onShadowSettingsChanged();
322
323 QVector<QgsPointXY> extent2D = viewFrustum2DExtent();
324 emit viewed2DExtentFrom3DChanged( extent2D );
325
326 // The magic to make things work better in large scenes (e.g. more than 50km across)
327 // is here: we will simply move the origin of the scene, and update transforms
328 // of the camera and all other entities. That should ensure we will not need to deal
329 // with large coordinates in 32-bit floats (and if we do have large coordinates,
330 // because the scene is far from the camera, we don't care, because those errors
331 // end up being tiny when viewed from far away).
332 constexpr float ORIGIN_SHIFT_THRESHOLD = 10'000;
333 if ( mSceneOriginShiftEnabled && mEngine->camera()->position().length() > ORIGIN_SHIFT_THRESHOLD )
334 {
335 const QgsVector3D newOrigin = mMap.origin() + QgsVector3D( mEngine->camera()->position() );
336 QgsDebugMsgLevel( QStringLiteral( "Rebasing scene origin from %1 to %2" ).arg( mMap.origin().toString( 1 ), newOrigin.toString( 1 ) ), 2 );
337 mMap.setOrigin( newOrigin );
338 }
339}
340
341void Qgs3DMapScene::updateScene( bool forceUpdate )
342{
343 if ( !mSceneUpdatesEnabled )
344 {
345 QgsDebugMsgLevel( "Scene update skipped", 2 );
346 return;
347 }
348
349 QgsEventTracing::ScopedEvent traceEvent( QStringLiteral( "3D" ), forceUpdate ? QStringLiteral( "Force update scene" ) : QStringLiteral( "Update scene" ) );
350
351 Qgs3DMapSceneEntity::SceneContext sceneContext;
352 Qt3DRender::QCamera *camera = mEngine->camera();
353 sceneContext.cameraFov = camera->fieldOfView();
354 sceneContext.cameraPos = camera->position();
355 const QSize size = mEngine->size();
356 sceneContext.screenSizePx = std::max( size.width(), size.height() ); // TODO: is this correct?
357
358 // Make our own projection matrix so that frustum culling done by the
359 // entities isn't dependent on the current near/far planes, which would then
360 // require multiple steps to stabilize.
361 // The matrix is constructed just like in QMatrix4x4::perspective(), but for
362 // all elements involving the near and far plane, the limit of the expression
363 // with the far plane going to infinity is taken.
364 float fovRadians = ( camera->fieldOfView() / 2.0f ) * static_cast<float>( M_PI ) / 180.0f;
365 float fovCotan = std::cos( fovRadians ) / std::sin( fovRadians );
366 QMatrix4x4 projMatrix(
367 fovCotan / camera->aspectRatio(), 0, 0, 0,
368 0, fovCotan, 0, 0,
369 0, 0, -1, -2,
370 0, 0, -1, 0
371 );
372 sceneContext.viewProjectionMatrix = projMatrix * camera->viewMatrix();
373
374
375 for ( Qgs3DMapSceneEntity *entity : std::as_const( mSceneEntities ) )
376 {
377 if ( forceUpdate || ( entity->isEnabled() && entity->needsUpdate() ) )
378 {
379 entity->handleSceneUpdate( sceneContext );
380 if ( entity->hasReachedGpuMemoryLimit() )
382 }
383 }
384
385 updateSceneState();
386}
387
388bool Qgs3DMapScene::updateCameraNearFarPlanes()
389{
390 // Update near and far plane from the terrain.
391 // this needs to be done with great care as we have kind of circular dependency here:
392 // active nodes are culled based on the current frustum (which involves near + far plane)
393 // and then based on active nodes we set near and far plane.
394 //
395 // All of this is just heuristics assuming that all other stuff is being rendered somewhere
396 // around the area where the terrain is.
397 //
398 // Near/far plane is setup in order to make best use of the depth buffer to avoid:
399 // 1. precision errors - if the range is too great
400 // 2. unwanted clipping of scene - if the range is too small
401
402 Qt3DRender::QCamera *camera = cameraController()->camera();
403 QMatrix4x4 viewMatrix = camera->viewMatrix();
404 float fnear = 1e9;
405 float ffar = 0;
406
407 // Iterate all scene entities to make sure that they will not get
408 // clipped by the near or far plane
409 for ( Qgs3DMapSceneEntity *se : std::as_const( mSceneEntities ) )
410 {
411 const QgsRange<float> depthRange = se->getNearFarPlaneRange( viewMatrix );
412
413 fnear = std::min( fnear, depthRange.lower() );
414 ffar = std::max( ffar, depthRange.upper() );
415 }
416
417 if ( fnear < 1 )
418 fnear = 1; // does not really make sense to use negative far plane (behind camera)
419
420 // the update didn't work out... this can happen if the scene does not contain
421 // any Qgs3DMapSceneEntity. Use the scene extent to compute near and far planes
422 // as a fallback.
423 if ( fnear == 1e9 && ffar == 0 )
424 {
425 QgsDoubleRange sceneZRange = elevationRange();
426 sceneZRange = sceneZRange.isInfinite() ? QgsDoubleRange( 0.0, 0.0 ) : sceneZRange;
427 const QgsAABB sceneBbox = Qgs3DUtils::mapToWorldExtent( mMap.extent(), sceneZRange.lower(), sceneZRange.upper(), mMap.origin() );
428 Qgs3DUtils::computeBoundingBoxNearFarPlanes( sceneBbox, viewMatrix, fnear, ffar );
429 }
430
431 // when zooming in a lot, fnear can become smaller than ffar. This should not happen
432 if ( fnear > ffar )
433 std::swap( fnear, ffar );
434
435 // set near/far plane - with some tolerance in front/behind expected near/far planes
436 float newFar = ffar * 2;
437 float newNear = fnear / 2;
438 if ( !qgsFloatNear( newFar, camera->farPlane() ) || !qgsFloatNear( newNear, camera->nearPlane() ) )
439 {
440 camera->setFarPlane( newFar );
441 camera->setNearPlane( newNear );
442 return true;
443 }
444
445 return false;
446}
447
448void Qgs3DMapScene::onFrameTriggered( float dt )
449{
450 QgsEventTracing::addEvent( QgsEventTracing::EventType::Instant, QStringLiteral( "3D" ), QStringLiteral( "Frame begins" ) );
451
452 mCameraController->frameTriggered( dt );
453
454 updateScene();
455
456 // lock changing the FPS counter to 5 fps
457 static int frameCount = 0;
458 static float accumulatedTime = 0.0f;
459
460 if ( !mMap.isFpsCounterEnabled() )
461 {
462 frameCount = 0;
463 accumulatedTime = 0;
464 return;
465 }
466
467 frameCount++;
468 accumulatedTime += dt;
469 if ( accumulatedTime >= 0.2f )
470 {
471 float fps = ( float ) frameCount / accumulatedTime;
472 frameCount = 0;
473 accumulatedTime = 0.0f;
474 emit fpsCountChanged( fps );
475 }
476}
477
478void Qgs3DMapScene::createTerrain()
479{
480 if ( mTerrain )
481 {
482 mSceneEntities.removeOne( mTerrain );
483
484 delete mTerrain;
485 mTerrain = nullptr;
486 }
487
488 if ( mGlobe )
489 {
490 mSceneEntities.removeOne( mGlobe );
491
492 delete mGlobe;
493 mGlobe = nullptr;
494 }
495
496 if ( !mTerrainUpdateScheduled )
497 {
498 // defer re-creation of terrain: there may be multiple invocations of this slot, so create the new entity just once
499 QTimer::singleShot( 0, this, &Qgs3DMapScene::createTerrainDeferred );
500 mTerrainUpdateScheduled = true;
501 setSceneState( Updating );
502 }
503 else
504 {
506 }
507}
508
509void Qgs3DMapScene::createTerrainDeferred()
510{
511 QgsChunkedEntity *terrainOrGlobe = nullptr;
512
514 {
515 mGlobe = new QgsGlobeEntity( &mMap );
516 terrainOrGlobe = mGlobe;
517 }
518 else if ( mMap.sceneMode() == Qgis::SceneMode::Local && mMap.terrainRenderingEnabled() && mMap.terrainGenerator() )
519 {
520 double tile0width = mMap.terrainGenerator()->rootChunkExtent().width();
521 int maxZoomLevel = Qgs3DUtils::maxZoomLevel( tile0width, mMap.terrainSettings()->mapTileResolution(), mMap.terrainSettings()->maximumGroundError() );
522 const QgsBox3D rootBox3D = mMap.terrainGenerator()->rootChunkBox3D( mMap );
523 float rootError = mMap.terrainGenerator()->rootChunkError( mMap );
524 const QgsBox3D clippingBox3D( mMap.extent(), rootBox3D.zMinimum(), rootBox3D.zMaximum() );
525 mMap.terrainGenerator()->setupQuadtree( rootBox3D, rootError, maxZoomLevel, clippingBox3D );
526
527 mTerrain = new QgsTerrainEntity( &mMap );
528 terrainOrGlobe = mTerrain;
529 }
530
531 if ( terrainOrGlobe )
532 {
533 terrainOrGlobe->setParent( this );
534 terrainOrGlobe->setShowBoundingBoxes( mMap.showTerrainBoundingBoxes() );
535
536 mSceneEntities << terrainOrGlobe;
537
538 connect( terrainOrGlobe, &QgsChunkedEntity::pendingJobsCountChanged, this, &Qgs3DMapScene::totalPendingJobsCountChanged );
539 connect( terrainOrGlobe, &Qgs3DMapSceneEntity::newEntityCreated, this, [this]( Qt3DCore::QEntity *entity ) {
540 // let's make sure that any entity we're about to show has the right scene origin set
541 const QList<QgsGeoTransform *> transforms = entity->findChildren<QgsGeoTransform *>();
542 for ( QgsGeoTransform *transform : transforms )
543 {
544 transform->setOrigin( mMap.origin() );
545 }
546
547 // enable clipping on the terrain if necessary
548 handleClippingOnEntity( entity );
549 } );
550 }
551
552 // make sure that renderers for layers are re-created as well
553 const QList<QgsMapLayer *> layers = mMap.layers();
554 for ( QgsMapLayer *layer : layers )
555 {
556 // remove old entity - if any
557 removeLayerEntity( layer );
558
559 // add new entity - if any 3D renderer
560 addLayerEntity( layer );
561 }
562
564 onCameraChanged(); // force update of the new terrain
565 mTerrainUpdateScheduled = false;
566}
567
568void Qgs3DMapScene::onBackgroundColorChanged()
569{
570 mEngine->setClearColor( mMap.backgroundColor() );
571}
572
573void Qgs3DMapScene::updateLights()
574{
575 for ( Qt3DCore::QEntity *entity : std::as_const( mLightEntities ) )
576 entity->deleteLater();
577 mLightEntities.clear();
578
579 const QList<QgsLightSource *> newLights = mMap.lightSources();
580 for ( const QgsLightSource *source : newLights )
581 {
582 mLightEntities.append( source->createEntity( mMap, this ) );
583 }
584
585 onShadowSettingsChanged();
586}
587
588void Qgs3DMapScene::updateCameraLens()
589{
590 mEngine->camera()->lens()->setFieldOfView( mMap.fieldOfView() );
591 mEngine->camera()->lens()->setProjectionType( mMap.projectionType() );
592 onCameraChanged();
593}
594
595void Qgs3DMapScene::onLayerRenderer3DChanged()
596{
597 QgsMapLayer *layer = qobject_cast<QgsMapLayer *>( sender() );
598 Q_ASSERT( layer );
599
600 // remove old entity - if any
601 removeLayerEntity( layer );
602
603 // add new entity - if any 3D renderer
604 addLayerEntity( layer );
605}
606
607void Qgs3DMapScene::onLayersChanged()
608{
609 QSet<QgsMapLayer *> layersBefore = qgis::listToSet( mLayerEntities.keys() );
610 QList<QgsMapLayer *> layersAdded;
611 const QList<QgsMapLayer *> layers = mMap.layers();
612 for ( QgsMapLayer *layer : layers )
613 {
614 if ( !layersBefore.contains( layer ) )
615 {
616 layersAdded << layer;
617 }
618 else
619 {
620 layersBefore.remove( layer );
621 }
622 }
623
624 // what is left in layersBefore are layers that have been removed
625 for ( QgsMapLayer *layer : std::as_const( layersBefore ) )
626 {
627 removeLayerEntity( layer );
628 }
629
630 for ( QgsMapLayer *layer : std::as_const( layersAdded ) )
631 {
632 addLayerEntity( layer );
633 }
634}
635
637{
638 const QList<QgsMapLayer *> layers = mLayerEntities.keys();
639 for ( QgsMapLayer *layer : layers )
640 {
641 if ( QgsMapLayerTemporalProperties *temporalProperties = layer->temporalProperties() )
642 {
643 if ( temporalProperties->isActive() )
644 {
645 removeLayerEntity( layer );
646 addLayerEntity( layer );
647 }
648 }
649 }
650}
651
652void Qgs3DMapScene::addSceneEntity( Qgs3DMapSceneEntity *sceneNewEntity )
653{
654 Q_ASSERT( sceneNewEntity );
655
656 mSceneEntities.append( sceneNewEntity );
657
658 sceneNewEntity->setParent( this );
659
660 finalizeNewEntity( sceneNewEntity );
661
662 connect( sceneNewEntity, &Qgs3DMapSceneEntity::newEntityCreated, this, [this]( Qt3DCore::QEntity *entity ) {
663 finalizeNewEntity( entity );
664 // this ensures to update the near/far planes with the exact bounding box of the new entity.
665 updateCameraNearFarPlanes();
666 } );
667
668 connect( sceneNewEntity, &Qgs3DMapSceneEntity::pendingJobsCountChanged, this, &Qgs3DMapScene::totalPendingJobsCountChanged );
669
670 onCameraChanged(); // needed for chunked entities
671}
672
673void Qgs3DMapScene::removeSceneEntity( Qgs3DMapSceneEntity *sceneEntity )
674{
675 Q_ASSERT( sceneEntity );
676
677 mSceneEntities.removeOne( sceneEntity );
678
679 sceneEntity->deleteLater();
680}
681
682
683void Qgs3DMapScene::addLayerEntity( QgsMapLayer *layer )
684{
685 QgsAbstract3DRenderer *renderer = layer->renderer3D();
686 if ( renderer )
687 {
688 // Fix vector layer's renderer to make sure the renderer is pointing to its layer.
689 // It has happened before that renderer pointed to a different layer (probably after copying a style).
690 // This is a bit of a hack and it should be handled in QgsMapLayer::setRenderer3D() but in qgis_core
691 // the vector layer 3D renderer classes are not available.
692 if ( layer->type() == Qgis::LayerType::Vector && ( renderer->type() == QLatin1String( "vector" ) || renderer->type() == QLatin1String( "rulebased" ) ) )
693 {
694 static_cast<QgsAbstractVectorLayer3DRenderer *>( renderer )->setLayer( static_cast<QgsVectorLayer *>( layer ) );
695 if ( renderer->type() == QLatin1String( "vector" ) )
696 {
697 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
698 if ( vlayer->geometryType() == Qgis::GeometryType::Point )
699 {
700 const QgsPoint3DSymbol *pointSymbol = static_cast<const QgsPoint3DSymbol *>( static_cast<QgsVectorLayer3DRenderer *>( renderer )->symbol() );
701 if ( pointSymbol->shape() == Qgis::Point3DShape::Model )
702 {
703 mModelVectorLayers.append( layer );
704 }
705 }
706 }
707 else if ( renderer->type() == QLatin1String( "rulebased" ) )
708 {
709 const QgsRuleBased3DRenderer::RuleList rules = static_cast<QgsRuleBased3DRenderer *>( renderer )->rootRule()->descendants();
710 for ( auto rule : rules )
711 {
712 const QgsPoint3DSymbol *pointSymbol = dynamic_cast<const QgsPoint3DSymbol *>( rule->symbol() );
713 if ( pointSymbol && pointSymbol->shape() == Qgis::Point3DShape::Model )
714 {
715 mModelVectorLayers.append( layer );
716 break;
717 }
718 }
719 }
720 }
721 else if ( layer->type() == Qgis::LayerType::Mesh && renderer->type() == QLatin1String( "mesh" ) )
722 {
723 QgsMeshLayer3DRenderer *meshRenderer = static_cast<QgsMeshLayer3DRenderer *>( renderer );
724 meshRenderer->setLayer( static_cast<QgsMeshLayer *>( layer ) );
725
726 // Before entity creation, set the maximum texture size
727 // Not very clean, but for now, only place found in the workflow to do that simple
728 QgsMesh3DSymbol *sym = meshRenderer->symbol()->clone();
729 sym->setMaximumTextureSize( maximumTextureSize() );
730 meshRenderer->setSymbol( sym );
731 }
732 else if ( layer->type() == Qgis::LayerType::PointCloud && renderer->type() == QLatin1String( "pointcloud" ) )
733 {
734 QgsPointCloudLayer3DRenderer *pointCloudRenderer = static_cast<QgsPointCloudLayer3DRenderer *>( renderer );
735 pointCloudRenderer->setLayer( static_cast<QgsPointCloudLayer *>( layer ) );
736 }
737 else if ( layer->type() == Qgis::LayerType::TiledScene && renderer->type() == QLatin1String( "tiledscene" ) )
738 {
739 QgsTiledSceneLayer3DRenderer *tiledSceneRenderer = static_cast<QgsTiledSceneLayer3DRenderer *>( renderer );
740 tiledSceneRenderer->setLayer( static_cast<QgsTiledSceneLayer *>( layer ) );
741 }
742
743 Qt3DCore::QEntity *newEntity = renderer->createEntity( &mMap );
744 if ( newEntity )
745 {
746 mLayerEntities.insert( layer, newEntity );
747
748 if ( Qgs3DMapSceneEntity *sceneNewEntity = qobject_cast<Qgs3DMapSceneEntity *>( newEntity ) )
749 {
750 // also sets this scene as the entity's parent and finalizes it
751 addSceneEntity( sceneNewEntity );
752 }
753 else
754 {
755 newEntity->setParent( this );
756 finalizeNewEntity( newEntity );
757 }
758 }
759 }
760
761 connect( layer, &QgsMapLayer::request3DUpdate, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
762
763 if ( layer->type() == Qgis::LayerType::Vector )
764 {
765 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
766 connect( vlayer, &QgsVectorLayer::selectionChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
767 connect( vlayer, &QgsVectorLayer::layerModified, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
768 }
769
770 if ( layer->type() == Qgis::LayerType::Mesh )
771 {
772 connect( layer, &QgsMapLayer::rendererChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
773 }
774
775 if ( layer->type() == Qgis::LayerType::PointCloud )
776 {
777 QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( layer );
778 connect( pclayer, &QgsPointCloudLayer::renderer3DChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
779 connect( pclayer, &QgsPointCloudLayer::subsetStringChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
780 }
781}
782
783void Qgs3DMapScene::removeLayerEntity( QgsMapLayer *layer )
784{
785 Qt3DCore::QEntity *entity = mLayerEntities.take( layer );
786
787 if ( Qgs3DMapSceneEntity *sceneEntity = qobject_cast<Qgs3DMapSceneEntity *>( entity ) )
788 {
789 // also schedules the entity for deletion
790 removeSceneEntity( sceneEntity );
791 }
792 else
793 {
794 if ( entity )
795 entity->deleteLater();
796 }
797
798 disconnect( layer, &QgsMapLayer::request3DUpdate, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
799
800 if ( layer->type() == Qgis::LayerType::Vector )
801 {
802 QgsVectorLayer *vlayer = qobject_cast<QgsVectorLayer *>( layer );
803 disconnect( vlayer, &QgsVectorLayer::selectionChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
804 disconnect( vlayer, &QgsVectorLayer::layerModified, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
805 mModelVectorLayers.removeAll( layer );
806 }
807
808 if ( layer->type() == Qgis::LayerType::Mesh )
809 {
810 disconnect( layer, &QgsMapLayer::rendererChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
811 }
812
813 if ( layer->type() == Qgis::LayerType::PointCloud )
814 {
815 QgsPointCloudLayer *pclayer = qobject_cast<QgsPointCloudLayer *>( layer );
816 disconnect( pclayer, &QgsPointCloudLayer::renderer3DChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
817 disconnect( pclayer, &QgsPointCloudLayer::subsetStringChanged, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
818 disconnect( pclayer, &QgsPointCloudLayer::layerModified, this, &Qgs3DMapScene::onLayerRenderer3DChanged );
819 }
820}
821
822void Qgs3DMapScene::finalizeNewEntity( Qt3DCore::QEntity *newEntity )
823{
824 // let's make sure that any entity we're about to show has the right scene origin set
825 const QList<QgsGeoTransform *> transforms = newEntity->findChildren<QgsGeoTransform *>();
826 for ( QgsGeoTransform *transform : transforms )
827 {
828 transform->setOrigin( mMap.origin() );
829 }
830
831 // set clip planes on the new entity if necessary
832 handleClippingOnEntity( newEntity );
833
834 // this is probably not the best place for material-specific configuration,
835 // maybe this could be more generalized when other materials need some specific treatment
836 const QList<QgsLineMaterial *> childLineMaterials = newEntity->findChildren<QgsLineMaterial *>();
837 for ( QgsLineMaterial *lm : childLineMaterials )
838 {
839 connect( mEngine, &QgsAbstract3DEngine::sizeChanged, lm, [lm, this] {
840 lm->setViewportSize( mEngine->size() );
841 } );
842
843 lm->setViewportSize( mEngine->size() );
844 }
845 // configure billboard's viewport when the viewport is changed.
846 const QList<QgsPoint3DBillboardMaterial *> childBillboardMaterials = newEntity->findChildren<QgsPoint3DBillboardMaterial *>();
847 for ( QgsPoint3DBillboardMaterial *bm : childBillboardMaterials )
848 {
849 connect( mEngine, &QgsAbstract3DEngine::sizeChanged, bm, [bm, this] {
850 bm->setViewportSize( mEngine->size() );
851 } );
852
853 bm->setViewportSize( mEngine->size() );
854 }
855
856 // Finalize adding the 3D transparent objects by adding the layer components to the entities
857 QgsFrameGraph *frameGraph = mEngine->frameGraph();
858 Qt3DRender::QLayer *transparentLayer = frameGraph->forwardRenderView().transparentObjectLayer();
859 const QList<Qt3DRender::QMaterial *> childMaterials = newEntity->findChildren<Qt3DRender::QMaterial *>();
860 for ( Qt3DRender::QMaterial *material : childMaterials )
861 {
862 // This handles the phong material without data defined properties.
863 if ( Qt3DExtras::QDiffuseSpecularMaterial *ph = qobject_cast<Qt3DExtras::QDiffuseSpecularMaterial *>( material ) )
864 {
865 if ( ph->diffuse().value<QColor>().alphaF() != 1.0f )
866 {
867 Qt3DCore::QEntity *entity = qobject_cast<Qt3DCore::QEntity *>( ph->parent() );
868 if ( entity && !entity->components().contains( transparentLayer ) )
869 {
870 entity->addComponent( transparentLayer );
871 }
872 }
873 }
874 else
875 {
876 // This handles the phong material with data defined properties, the textured case and point (instanced) symbols.
877 Qt3DRender::QEffect *effect = material->effect();
878 if ( effect )
879 {
880 const QVector<Qt3DRender::QParameter *> parameters = effect->parameters();
881 for ( const Qt3DRender::QParameter *parameter : parameters )
882 {
883 if ( parameter->name() == "opacity" && parameter->value() != 1.0f )
884 {
885 Qt3DCore::QEntity *entity = qobject_cast<Qt3DCore::QEntity *>( material->parent() );
886 if ( entity && !entity->components().contains( transparentLayer ) )
887 {
888 entity->addComponent( transparentLayer );
889 }
890 break;
891 }
892 }
893 }
894 }
895 }
896}
897
898int Qgs3DMapScene::maximumTextureSize() const
899{
900 QSurface *surface = mEngine->surface();
901 QOpenGLContext context;
902 context.create();
903 bool success = context.makeCurrent( surface );
904
905 if ( success )
906 {
907 QOpenGLFunctions openglFunctions = QOpenGLFunctions( &context );
908
909 GLint size;
910 openglFunctions.initializeOpenGLFunctions();
911 openglFunctions.glGetIntegerv( GL_MAX_TEXTURE_SIZE, &size );
912 return int( size );
913 }
914 else
915 {
916 return 4096; //we can't have a context to defined the max texture size, we use this reasonable value
917 }
918}
919
920void Qgs3DMapScene::addCameraViewCenterEntity( Qt3DRender::QCamera *camera )
921{
922 mEntityCameraViewCenter = new Qt3DCore::QEntity;
923
924 Qt3DCore::QTransform *trCameraViewCenter = new Qt3DCore::QTransform;
925 mEntityCameraViewCenter->addComponent( trCameraViewCenter );
926 connect( camera, &Qt3DRender::QCamera::viewCenterChanged, this, [trCameraViewCenter, camera] {
927 trCameraViewCenter->setTranslation( camera->viewCenter() );
928 } );
929
930 Qt3DExtras::QPhongMaterial *materialCameraViewCenter = new Qt3DExtras::QPhongMaterial;
931 materialCameraViewCenter->setAmbient( Qt::red );
932 mEntityCameraViewCenter->addComponent( materialCameraViewCenter );
933
934 Qt3DExtras::QSphereMesh *rendererCameraViewCenter = new Qt3DExtras::QSphereMesh;
935 rendererCameraViewCenter->setRadius( 10 );
936 mEntityCameraViewCenter->addComponent( rendererCameraViewCenter );
937
938 mEntityCameraViewCenter->setEnabled( mMap.showCameraViewCenter() );
939 mEntityCameraViewCenter->setParent( this );
940
941 connect( &mMap, &Qgs3DMapSettings::showCameraViewCenterChanged, this, [this] {
942 mEntityCameraViewCenter->setEnabled( mMap.showCameraViewCenter() );
943 } );
944}
945
946void Qgs3DMapScene::setSceneState( Qgs3DMapScene::SceneState state )
947{
948 if ( mSceneState == state )
949 return;
950 mSceneState = state;
951 emit sceneStateChanged();
952}
953
954void Qgs3DMapScene::updateSceneState()
955{
956 if ( mTerrainUpdateScheduled )
957 {
958 setSceneState( Updating );
959 return;
960 }
961
962 for ( Qgs3DMapSceneEntity *entity : std::as_const( mSceneEntities ) )
963 {
964 if ( entity->isEnabled() && entity->pendingJobsCount() > 0 )
965 {
966 setSceneState( Updating );
967 return;
968 }
969 }
970
971 setSceneState( Ready );
972}
973
974void Qgs3DMapScene::onSkyboxSettingsChanged()
975{
976 QgsSkyboxSettings skyboxSettings = mMap.skyboxSettings();
977 if ( mSkybox )
978 {
979 mSkybox->deleteLater();
980 mSkybox = nullptr;
981 }
982
983 mEngine->setFrustumCullingEnabled( !mMap.isSkyboxEnabled() );
984
985 if ( mMap.isSkyboxEnabled() )
986 {
987 QMap<QString, QString> faces;
988 switch ( skyboxSettings.skyboxType() )
989 {
991 faces = skyboxSettings.cubeMapFacesPaths();
992 mSkybox = new QgsCubeFacesSkyboxEntity(
993 faces[QStringLiteral( "posX" )], faces[QStringLiteral( "posY" )], faces[QStringLiteral( "posZ" )],
994 faces[QStringLiteral( "negX" )], faces[QStringLiteral( "negY" )], faces[QStringLiteral( "negZ" )],
995 this
996 );
997 break;
999 mSkybox = new QgsPanoramicSkyboxEntity( skyboxSettings.panoramicTexturePath(), this );
1000 break;
1001 }
1002 }
1003}
1004
1005void Qgs3DMapScene::onShadowSettingsChanged()
1006{
1007 QgsFrameGraph *frameGraph = mEngine->frameGraph();
1008 frameGraph->updateShadowSettings( mMap.shadowSettings(), mMap.lightSources() );
1009}
1010
1011void Qgs3DMapScene::onAmbientOcclusionSettingsChanged()
1012{
1013 QgsFrameGraph *frameGraph = mEngine->frameGraph();
1014 QgsAmbientOcclusionSettings ambientOcclusionSettings = mMap.ambientOcclusionSettings();
1015 frameGraph->setAmbientOcclusionEnabled( ambientOcclusionSettings.isEnabled() );
1016 frameGraph->setAmbientOcclusionRadius( ambientOcclusionSettings.radius() );
1017 frameGraph->setAmbientOcclusionIntensity( ambientOcclusionSettings.intensity() );
1018 frameGraph->setAmbientOcclusionThreshold( ambientOcclusionSettings.threshold() );
1019}
1020
1021void Qgs3DMapScene::onDebugShadowMapSettingsChanged()
1022{
1024}
1025
1026void Qgs3DMapScene::onDebugDepthMapSettingsChanged()
1027{
1029}
1030
1031void Qgs3DMapScene::onDebugOverlayEnabledChanged()
1032{
1034 mEngine->renderSettings()->setRenderPolicy( mMap.isDebugOverlayEnabled() ? Qt3DRender::QRenderSettings::Always : Qt3DRender::QRenderSettings::OnDemand );
1035}
1036
1037void Qgs3DMapScene::onEyeDomeShadingSettingsChanged()
1038{
1039 bool edlEnabled = mMap.eyeDomeLightingEnabled();
1040 double edlStrength = mMap.eyeDomeLightingStrength();
1041 double edlDistance = mMap.eyeDomeLightingDistance();
1042 mEngine->frameGraph()->setupEyeDomeLighting( edlEnabled, edlStrength, edlDistance );
1043}
1044
1045void Qgs3DMapScene::onCameraMovementSpeedChanged()
1046{
1047 mCameraController->setCameraMovementSpeed( mMap.cameraMovementSpeed() );
1048}
1049
1050void Qgs3DMapScene::onCameraNavigationModeChanged()
1051{
1052 mCameraController->setCameraNavigationMode( mMap.cameraNavigationMode() );
1053}
1054
1056{
1057 QVector<QString> notParsedLayers;
1058 Qgs3DSceneExporter exporter;
1059
1060 exporter.setTerrainResolution( exportSettings.terrrainResolution() );
1061 exporter.setSmoothEdges( exportSettings.smoothEdges() );
1062 exporter.setExportNormals( exportSettings.exportNormals() );
1063 exporter.setExportTextures( exportSettings.exportTextures() );
1064 exporter.setTerrainTextureResolution( exportSettings.terrainTextureResolution() );
1065 exporter.setScale( exportSettings.scale() );
1066
1067 for ( auto it = mLayerEntities.constBegin(); it != mLayerEntities.constEnd(); ++it )
1068 {
1069 QgsMapLayer *layer = it.key();
1070 Qt3DCore::QEntity *rootEntity = it.value();
1071 Qgis::LayerType layerType = layer->type();
1072 switch ( layerType )
1073 {
1075 if ( !exporter.parseVectorLayerEntity( rootEntity, qobject_cast<QgsVectorLayer *>( layer ) ) )
1076 notParsedLayers.push_back( layer->name() );
1077 break;
1086 notParsedLayers.push_back( layer->name() );
1087 break;
1088 }
1089 }
1090
1091 if ( mTerrain )
1092 exporter.parseTerrain( mTerrain, "Terrain" );
1093
1094 const bool sceneSaved = exporter.save( exportSettings.sceneName(), exportSettings.sceneFolderPath() );
1095 if ( !sceneSaved )
1096 {
1097 return false;
1098 }
1099
1100 if ( !notParsedLayers.empty() )
1101 {
1102 QString message = tr( "The following layers were not exported:" ) + "\n";
1103 for ( const QString &layerName : notParsedLayers )
1104 message += layerName + "\n";
1105 QgsMessageOutput::showMessage( tr( "3D exporter warning" ), message, QgsMessageOutput::MessageText );
1106 }
1107
1108 return true;
1109}
1110
1111QVector<const QgsChunkNode *> Qgs3DMapScene::getLayerActiveChunkNodes( QgsMapLayer *layer )
1112{
1113 QVector<const QgsChunkNode *> chunks;
1114 if ( !mLayerEntities.contains( layer ) )
1115 return chunks;
1116 if ( QgsChunkedEntity *c = qobject_cast<QgsChunkedEntity *>( mLayerEntities[layer] ) )
1117 {
1118 const QList<QgsChunkNode *> activeNodes = c->activeNodes();
1119 for ( QgsChunkNode *n : activeNodes )
1120 chunks.push_back( n );
1121 }
1122 return chunks;
1123}
1124
1126{
1127 return mMap.extent();
1128}
1129
1130QgsDoubleRange Qgs3DMapScene::elevationRange( const bool ignoreTerrain ) const
1131{
1132 double zMin = std::numeric_limits<double>::max();
1133 double zMax = std::numeric_limits<double>::lowest();
1134 if ( mMap.terrainRenderingEnabled() && mTerrain && !ignoreTerrain )
1135 {
1136 const QgsBox3D box3D = mTerrain->rootNode()->box3D();
1137 zMin = std::min( zMin, box3D.zMinimum() );
1138 zMax = std::max( zMax, box3D.zMaximum() );
1139 }
1140
1141 for ( auto it = mLayerEntities.constBegin(); it != mLayerEntities.constEnd(); it++ )
1142 {
1143 QgsMapLayer *layer = it.key();
1144 switch ( layer->type() )
1145 {
1147 {
1148 QgsPointCloudLayer *pcl = qobject_cast<QgsPointCloudLayer *>( layer );
1149 QgsDoubleRange zRange = pcl->elevationProperties()->calculateZRange( pcl );
1150 zMin = std::min( zMin, zRange.lower() );
1151 zMax = std::max( zMax, zRange.upper() );
1152 break;
1153 }
1155 {
1156 QgsMeshLayer *meshLayer = qobject_cast<QgsMeshLayer *>( layer );
1157 QgsAbstract3DRenderer *renderer3D = meshLayer->renderer3D();
1158 if ( renderer3D )
1159 {
1160 QgsMeshLayer3DRenderer *meshLayerRenderer = static_cast<QgsMeshLayer3DRenderer *>( renderer3D );
1161 const int verticalGroupDatasetIndex = meshLayerRenderer->symbol()->verticalDatasetGroupIndex();
1162 const QgsMeshDatasetGroupMetadata verticalGroupMetadata = meshLayer->datasetGroupMetadata( verticalGroupDatasetIndex );
1163 const double verticalScale = meshLayerRenderer->symbol()->verticalScale();
1164 zMin = std::min( zMin, verticalGroupMetadata.minimum() * verticalScale );
1165 zMax = std::max( zMax, verticalGroupMetadata.maximum() * verticalScale );
1166 }
1167 break;
1168 }
1170 {
1171 QgsTiledSceneLayer *sceneLayer = qobject_cast<QgsTiledSceneLayer *>( layer );
1172 const QgsDoubleRange zRange = sceneLayer->elevationProperties()->calculateZRange( sceneLayer );
1173 if ( !zRange.isInfinite() && !zRange.isEmpty() )
1174 {
1175 zMin = std::min( zMin, zRange.lower() );
1176 zMax = std::max( zMax, zRange.upper() );
1177 }
1178 break;
1179 }
1186 break;
1187 }
1188 }
1189 const QgsDoubleRange zRange( std::min( zMin, std::numeric_limits<double>::max() ), std::max( zMax, std::numeric_limits<double>::lowest() ) );
1190 return zRange.isEmpty() ? QgsDoubleRange() : zRange;
1191}
1192
1193QMap<QString, Qgs3DMapScene *> Qgs3DMapScene::openScenes()
1194{
1195 return sOpenScenesFunction();
1196}
1197
1198void Qgs3DMapScene::addCameraRotationCenterEntity( QgsCameraController *controller )
1199{
1200 mEntityRotationCenter = new Qt3DCore::QEntity;
1201
1202 Qt3DCore::QTransform *trRotationCenter = new Qt3DCore::QTransform;
1203 mEntityRotationCenter->addComponent( trRotationCenter );
1204 Qt3DExtras::QPhongMaterial *materialRotationCenter = new Qt3DExtras::QPhongMaterial;
1205 materialRotationCenter->setAmbient( Qt::blue );
1206 mEntityRotationCenter->addComponent( materialRotationCenter );
1207 Qt3DExtras::QSphereMesh *rendererRotationCenter = new Qt3DExtras::QSphereMesh;
1208 rendererRotationCenter->setRadius( 10 );
1209 mEntityRotationCenter->addComponent( rendererRotationCenter );
1210 mEntityRotationCenter->setEnabled( false );
1211 mEntityRotationCenter->setParent( this );
1212
1213 connect( controller, &QgsCameraController::cameraRotationCenterChanged, this, [trRotationCenter]( QVector3D center ) {
1214 trRotationCenter->setTranslation( center );
1215 } );
1216
1217 connect( &mMap, &Qgs3DMapSettings::showCameraRotationCenterChanged, this, [this] {
1218 mEntityRotationCenter->setEnabled( mMap.showCameraRotationCenter() );
1219 } );
1220}
1221
1222void Qgs3DMapScene::on3DAxisSettingsChanged()
1223{
1224 if ( m3DAxis )
1225 {
1226 m3DAxis->onAxisSettingsChanged();
1227 }
1228 else
1229 {
1230 if ( QgsWindow3DEngine *engine = dynamic_cast<QgsWindow3DEngine *>( mEngine ) )
1231 {
1232 m3DAxis = new Qgs3DAxis( static_cast<Qgs3DMapCanvas *>( engine->window() ), engine->root(), this, mCameraController, &mMap );
1233 }
1234 }
1235}
1236
1237void Qgs3DMapScene::onOriginChanged()
1238{
1239 const QList<QgsGeoTransform *> geoTransforms = findChildren<QgsGeoTransform *>();
1240 for ( QgsGeoTransform *transform : geoTransforms )
1241 {
1242 transform->setOrigin( mMap.origin() );
1243 }
1244
1245 const QgsVector3D oldOrigin = mCameraController->origin();
1246 mCameraController->setOrigin( mMap.origin() );
1247
1248 if ( !mClipPlanesEquations.isEmpty() )
1249 {
1250 // how the math works - for a plane defined as (a,b,c,d), only "d" changes when
1251 // moving the origin - the plane normal vector (a,b,c) stays the same.
1252 // - line equation for old shift: a * (x - x0) + b * (y - y0) + c * (z - z0) + d0 = 0
1253 // - line equation for new shift: a * (x - x1) + b * (y - y1) + c * (z - z1) + d1 = 0
1254 // - we solve for d1:
1255 // d1 = a * (x1 - x0) + b * (y1 - y0) + c * (z1 - z0) + d0
1256
1257 QList<QVector4D> newPlanes;
1258 QgsVector3D originShift = mMap.origin() - oldOrigin;
1259 for ( QVector4D plane : std::as_const( mClipPlanesEquations ) )
1260 {
1261 plane.setW( originShift.x() * plane.x() + originShift.y() * plane.y() + originShift.z() * plane.z() + plane.w() );
1262 newPlanes.append( plane );
1263 }
1264 enableClipping( newPlanes );
1265 }
1266}
1267
1268void Qgs3DMapScene::handleClippingOnEntity( QEntity *entity ) const
1269{
1270 if ( mClipPlanesEquations.isEmpty() ) // no clip plane equations, disable clipping
1271 {
1272 for ( QgsMaterial *material : entity->componentsOfType<QgsMaterial>() )
1273 {
1274 material->disableClipping();
1275 }
1276 }
1277 else // enable clipping
1278 {
1279 for ( QgsMaterial *material : entity->componentsOfType<QgsMaterial>() )
1280 {
1281 material->enableClipping( mClipPlanesEquations );
1282 }
1283 }
1284
1285 // recursive call
1286 // enable or disable clipping on the children accordingly
1287 for ( QObject *child : entity->children() )
1288 {
1289 Qt3DCore::QEntity *childEntity = qobject_cast<Qt3DCore::QEntity *>( child );
1290 if ( childEntity )
1291 {
1292 handleClippingOnEntity( childEntity );
1293 }
1294 }
1295}
1296
1297void Qgs3DMapScene::handleClippingOnAllEntities() const
1298{
1299 // Need to loop mLayerEntities instead of mSceneEntities to handle entities
1300 // which do no inherit from Qgs3DMapSceneEntity. For example, mesh entities.
1301 for ( auto it = mLayerEntities.constBegin(); it != mLayerEntities.constEnd(); ++it )
1302 {
1303 handleClippingOnEntity( it.value() );
1304 }
1305 if ( mTerrain )
1306 {
1307 handleClippingOnEntity( mTerrain );
1308 }
1309 if ( mGlobe )
1310 {
1311 handleClippingOnEntity( mGlobe );
1312 }
1313}
1314
1315void Qgs3DMapScene::enableClipping( const QList<QVector4D> &clipPlaneEquations )
1316{
1317 if ( clipPlaneEquations.size() > mMaxClipPlanes )
1318 {
1319 QgsDebugMsgLevel( QStringLiteral( "Qgs3DMapScene::enableClipping: it is not possible to use more than %1 clipping planes." ).arg( mMaxClipPlanes ), 2 );
1320 }
1321 mClipPlanesEquations = clipPlaneEquations.mid( 0, mMaxClipPlanes );
1322
1323 // enable the clip planes on the framegraph
1324 QgsForwardRenderView &forwardRenderView = mEngine->frameGraph()->forwardRenderView();
1325 forwardRenderView.addClipPlanes( clipPlaneEquations.size() );
1326
1327 // Enable the clip planes for the material of each entity.
1328 handleClippingOnAllEntities();
1329}
1330
1332{
1333 mClipPlanesEquations.clear();
1334
1335 // disable the clip planes on the framegraph
1336 QgsForwardRenderView &forwardRenderView = mEngine->frameGraph()->forwardRenderView();
1337 forwardRenderView.removeClipPlanes();
1338
1339 // Disable the clip planes for the material of each entity.
1340 handleClippingOnAllEntities();
1341}
1342
1343void Qgs3DMapScene::onStopUpdatesChanged()
1344{
1345 mSceneUpdatesEnabled = !mMap.stopUpdates();
1346}
LayerType
Types of layers that can be added to a map.
Definition qgis.h:169
@ 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.
@ Vector
Vector layer.
@ VectorTile
Vector tile layer. Added in QGIS 3.14.
@ Mesh
Mesh layer. Added in QGIS 3.2.
@ Raster
Raster layer.
@ PointCloud
Point cloud layer. Added in QGIS 3.18.
@ Globe
Scene is represented as a globe using a geocentric CRS.
@ Local
Local scene based on a projected CRS.
Display 3D ortho axis in the main 3D view.
Definition qgs3daxis.h:55
void onAxisSettingsChanged()
Force update of the axis and the viewport when a setting has changed.
Convenience wrapper to simplify the creation of a 3D window ready to be used with QGIS.
Manages the various settings the user can choose from when exporting a 3D scene.
bool exportNormals() const
Returns whether normals will be exported.
int terrrainResolution() const
Returns the terrain resolution.
QString sceneFolderPath() const
Returns the scene folder path.
float scale() const
Returns the scale of the exported model.
int terrainTextureResolution() const
Returns the terrain texture resolution.
QString sceneName() const
Returns the scene name.
bool smoothEdges() const
Returns whether triangles edges will look smooth.
bool exportTextures() const
Returns whether textures will be exported.
QVector< const QgsChunkNode * > getLayerActiveChunkNodes(QgsMapLayer *layer)
Returns the active chunk nodes of layer.
void viewed2DExtentFrom3DChanged(QVector< QgsPointXY > extent)
Emitted when the viewed 2D extent seen by the 3D camera has changed.
static std::function< QMap< QString, Qgs3DMapScene * >()> sOpenScenesFunction
Static function for returning open 3D map scenes.
void fpsCountChanged(float fpsCount)
Emitted when the FPS count changes.
void setViewFrom2DExtent(const QgsRectangle &extent)
Resets camera view to show the extent extent (top view)
void disableClipping()
Disables OpenGL clipping.
Qgs3DMapScene(Qgs3DMapSettings &map, QgsAbstract3DEngine *engine)
Constructs a 3D scene based on map settings and Qt 3D renderer configuration.
QgsAbstract3DEngine * engine() const
Returns the abstract 3D engine.
void gpuMemoryLimitReached()
Emitted when one of the entities reaches its GPU memory limit and it is not possible to lower the GPU...
QgsCameraController * cameraController() const
Returns camera controller.
SceneState
Enumeration of possible states of the 3D scene.
@ Ready
The scene is fully loaded/updated.
@ Updating
The scene is still being loaded/updated.
bool exportScene(const Qgs3DMapExportSettings &exportSettings)
Exports the scene according to the scene export settings Returns false if the operation failed.
int totalPendingJobsCount() const
Returns number of pending jobs for all chunked entities.
void updateTemporal()
Updates the temporale entities.
static Q_DECL_DEPRECATED QMap< QString, Qgs3DMapScene * > openScenes()
Returns a map of 3D map scenes (by name) open in the QGIS application.
void totalPendingJobsCountChanged()
Emitted when the total number of pending jobs changes.
void fpsCounterEnabledChanged(bool fpsCounterEnabled)
Emitted when the FPS counter is activated or deactivated.
QgsDoubleRange elevationRange(bool ignoreTerrain=false) const
Returns the scene's elevation range.
QgsRectangle sceneExtent() const
Returns the scene extent in the map's CRS.
void sceneStateChanged()
Emitted when the scene's state has changed.
void removeSceneEntity(Qgs3DMapSceneEntity *entity)
Removes a 3D scene entity for the scene.
QList< QgsMapLayer * > layers() const
Returns the layers that contain chunked entities.
QVector< QgsPointXY > viewFrustum2DExtent() const
Calculates the 2D extent viewed by the 3D camera as the vertices of the viewed trapezoid.
void enableClipping(const QList< QVector4D > &clipPlaneEquations)
Enables OpenGL clipping based on the planes equations defined in clipPlaneEquations.
float worldSpaceError(float epsilon, float distance) const
Given screen error (in pixels) and distance from camera (in 3D world coordinates),...
void terrainEntityChanged()
Emitted when the current terrain entity is replaced by a new one.
void addSceneEntity(Qgs3DMapSceneEntity *entity)
Adds a 3D map scene entity to the scene.
void viewZoomFull()
Resets camera view to show the whole scene (top view)
Definition of the world.
void extentChanged()
Emitted when the 3d view's 2d extent has changed.
bool isDebugOverlayEnabled() const
Returns whether debug overlay is enabled.
Qt::Corner debugDepthMapCorner() const
Returns the corner where the shadow map preview is displayed.
void originChanged()
Emitted when the world's origin point has been shifted.
void eyeDomeLightingDistanceChanged()
Emitted when the eye dome lighting distance has changed.
void terrainShadingChanged()
Emitted when terrain shading enabled flag or terrain shading material has changed.
QgsVector3D mapToWorldCoordinates(const QgsVector3D &mapCoords) const
Converts map coordinates to 3D world coordinates (applies offset and turns (x,y,z) into (x,...
double cameraMovementSpeed() const
Returns the camera movement speed.
Qt3DRender::QCameraLens::ProjectionType projectionType() const
Returns the camera lens' projection type.
bool debugDepthMapEnabled() const
Returns whether the shadow map debugging is enabled.
bool isSkyboxEnabled() const
Returns whether the skybox is enabled.
void debugDepthMapSettingsChanged()
Emitted when depth map debugging has changed.
Qgis::NavigationMode cameraNavigationMode() const
Returns the navigation mode used by the camera.
double eyeDomeLightingStrength() const
Returns the eye dome lighting strength value.
void backgroundColorChanged()
Emitted when the background color has changed.
Qt::Corner debugShadowMapCorner() const
Returns the corner where the shadow map preview is displayed.
bool showCameraViewCenter() const
Returns whether to show camera's view center as a sphere (for debugging)
void showCameraRotationCenterChanged()
Emitted when the flag whether camera's rotation center is shown has changed.
const QgsAbstractTerrainSettings * terrainSettings() const
Returns the terrain settings.
void cameraNavigationModeChanged()
Emitted when the camera navigation mode was changed.
void shadowSettingsChanged()
Emitted when shadow rendering settings are changed.
bool stopUpdates() const
Returns whether the scene updates on camera movement.
void eyeDomeLightingEnabledChanged()
Emitted when the flag whether eye dome lighting is used has changed.
void debugOverlayEnabledChanged(bool debugOverlayEnabled)
Emitted when the debug overaly is enabled or disabled.
Qgis::SceneMode sceneMode() const
Returns mode of the 3D scene - whether it is represented as a globe (when using Geocentric CRS such a...
void setOrigin(const QgsVector3D &origin)
Sets coordinates in map CRS at which our 3D world has origin (0,0,0)
void skyboxSettingsChanged()
Emitted when skybox settings are changed.
QgsShadowSettings shadowSettings() const
Returns the current configuration of shadows.
QList< QgsLightSource * > lightSources() const
Returns list of directional light sources defined in the scene.
double debugDepthMapSize() const
Returns the size of the shadow map preview.
void projectionTypeChanged()
Emitted when the camera lens projection type changes.
float fieldOfView() const
Returns the camera lens' field of view.
QgsAmbientOcclusionSettings ambientOcclusionSettings() const
Returns the current configuration of screen space ambient occlusion.
QgsRectangle extent() const
Returns the 3D scene's 2D extent in the 3D scene's CRS.
void stopUpdatesChanged()
Emitted when the flag whether to keep updating scene has changed.
int eyeDomeLightingDistance() const
Returns the eye dome lighting distance value (contributes to the contrast of the image)
void lightSourcesChanged()
Emitted when any of the light source settings in the map changes.
void showLightSourceOriginsChanged()
Emitted when the flag whether light source origins are shown has changed.
QgsTerrainGenerator * terrainGenerator() const
Returns the terrain generator.
QColor backgroundColor() const
Returns background color of the 3D map view.
void terrainSettingsChanged()
Emitted when the terrain settings are changed.
double debugShadowMapSize() const
Returns the size of the shadow map preview.
QgsVector3D worldToMapCoordinates(const QgsVector3D &worldCoords) const
Converts 3D world coordinates to map coordinates (applies offset and turns (x,y,z) into (x,...
bool showTerrainBoundingBoxes() const
Returns whether to display bounding boxes of terrain tiles (for debugging)
bool debugShadowMapEnabled() const
Returns whether the shadow map debugging is enabled.
bool terrainRenderingEnabled() const
Returns whether the 2D terrain surface will be rendered.
void fpsCounterEnabledChanged(bool fpsCounterEnabled)
Emitted when the FPS counter is enabled or disabled.
void axisSettingsChanged()
Emitted when 3d axis rendering settings are changed.
void ambientOcclusionSettingsChanged()
Emitted when ambient occlusion rendering settings are changed.
void layersChanged()
Emitted when the list of map layers for 3d rendering has changed.
void eyeDomeLightingStrengthChanged()
Emitted when the eye dome lighting strength has changed.
QgsSkyboxSettings skyboxSettings() const
Returns the current configuration of the skybox.
void cameraMovementSpeedChanged()
Emitted when the camera movement speed was changed.
bool eyeDomeLightingEnabled() const
Returns whether eye dome lighting is used.
bool isFpsCounterEnabled() const
Returns whether FPS counter label is enabled.
void fieldOfViewChanged()
Emitted when the camera lens field of view changes.
QList< QgsMapLayer * > layers() const
Returns the list of 3D map layers to be rendered in the scene.
void terrainGeneratorChanged()
Emitted when the terrain generator has changed.
void debugShadowMapSettingsChanged()
Emitted when shadow map debugging has changed.
void showCameraViewCenterChanged()
Emitted when the flag whether camera's view center is shown has changed.
QgsVector3D origin() const
Returns coordinates in map CRS at which 3D scene has origin (0,0,0).
bool showCameraRotationCenter() const
Returns whether to show camera's rotation center as a sphere (for debugging)
Entity that handles the exporting of 3D scenes.
bool save(const QString &sceneName, const QString &sceneFolderPath, int precision=6)
Saves the scene to a .obj file Returns false if the operation failed.
void setExportTextures(bool exportTextures)
Sets whether the textures will be exported.
void parseTerrain(QgsTerrainEntity *terrain, const QString &layer)
Creates terrain export objects from the terrain entity.
void setTerrainResolution(int resolution)
Sets the terrain resolution.
void setTerrainTextureResolution(int resolution)
Sets the terrain texture resolution.
bool parseVectorLayerEntity(Qt3DCore::QEntity *entity, QgsVectorLayer *layer)
Creates necessary export objects from entity if it represents valid vector layer entity Returns false...
void setScale(float scale)
Sets the scale of the exported 3D model.
void setExportNormals(bool exportNormals)
Sets whether the normals will be exported.
void setSmoothEdges(bool smoothEdges)
Sets whether the triangles will look smooth.
static int maxZoomLevel(double tile0width, double tileResolution, double maxError)
Calculates the highest needed zoom level for tiles in quad-tree given width of the base tile (zoom le...
static QgsAABB mapToWorldExtent(const QgsRectangle &extent, double zMin, double zMax, const QgsVector3D &mapOrigin)
Converts map extent to axis aligned bounding box in 3D world coordinates.
static void computeBoundingBoxNearFarPlanes(const QgsAABB &bbox, const QMatrix4x4 &viewMatrix, float &fnear, float &ffar)
This routine computes nearPlane farPlane from the closest and farthest corners point of bounding box ...
static QgsRay3D rayFromScreenPoint(const QPoint &point, const QSize &windowSize, Qt3DRender::QCamera *camera)
Convert from clicked point on the screen to a ray in world coordinates.
static int openGlMaxClipPlanes(QSurface *surface)
Gets the maximum number of clip planes that can be used.
Axis-aligned bounding box - in world coords.
Definition qgsaabb.h:35
Base class for 3D engine implementation.
void sizeChanged()
Emitted after a call to setSize()
virtual QSurface * surface() const =0
Returns the surface of the engine.
virtual Qt3DRender::QCamera * camera()=0
Returns pointer to the engine's camera entity.
virtual void setClearColor(const QColor &color)=0
Sets background color of the scene.
virtual void setFrustumCullingEnabled(bool enabled)=0
Sets whether frustum culling is enabled (this should make rendering faster by not rendering entities ...
QgsFrameGraph * frameGraph()
Returns the shadow rendering frame graph object used to render the scene.
virtual QSize size() const =0
Returns size of the engine's rendering area in pixels.
virtual Qt3DRender::QRenderSettings * renderSettings()=0
Returns access to the engine's render settings (the frame graph can be accessed from here)
Base class for all renderers that participate in 3D views.
virtual QString type() const =0
Returns unique identifier of the renderer class (used to identify subclass)
virtual Qt3DCore::QEntity * createEntity(Qgs3DMapSettings *map) const =0
Returns a 3D entity that will be used to show renderer's data in 3D scene.
double maximumGroundError() const
Returns the maximum ground error of terrain tiles in world units.
int mapTileResolution() const
Returns the resolution (in pixels) of the texture of a terrain tile.
Base class for 3D renderers that are based on vector layers.
Contains the configuration of ambient occlusion rendering.
float radius() const
Returns the radius parameter of the ambient occlusion effect.
bool isEnabled() const
Returns whether ambient occlusion effect is enabled.
float intensity() const
Returns the shading factor of the ambient occlusion effect.
float threshold() const
Returns at what amount of occlusion the effect will kick in.
static QgsSourceCache * sourceCache()
Returns the application's source cache, used for caching embedded and remote source strings as local ...
A 3-dimensional box composed of x, y, z coordinates.
Definition qgsbox3d.h:43
double zMaximum() const
Returns the maximum z value.
Definition qgsbox3d.h:259
double zMinimum() const
Returns the minimum z value.
Definition qgsbox3d.h:252
Object that controls camera movement based on user input.
Qt3DRender::QCamera * camera() const
Returns camera that is being controlled.
const QgsVector3D origin() const
Returns the origin of the scene in map coordinates.
float distance() const
Returns distance of the camera from the point it is looking at.
void setCameraNavigationMode(Qgis::NavigationMode navigationMode)
Sets the navigation mode used by the camera controller.
void cameraChanged()
Emitted when camera has been updated.
void frameTriggered(float dt)
Called internally from 3D scene when a new frame is generated. Updates camera according to keyboard/m...
void resetView(float distance)
Move camera back to the initial position (looking down towards origin of world's coordinates)
void setViewFromTop(float worldX, float worldY, float distance, float yaw=0)
Sets camera to look down towards given point in world coordinate, in given distance from plane with z...
void resetGlobe(float distance, double lat=0, double lon=0)
Resets view of the globe to look at a particular location given as latitude and longitude (in degrees...
void setOrigin(const QgsVector3D &origin)
Reacts to the shift of origin of the scene, updating camera pose and any other member variables so th...
void setCameraMovementSpeed(double movementSpeed)
Sets the camera movement speed.
void cameraRotationCenterChanged(QVector3D position)
Emitted when the camera rotation center changes.
A skybox constructed from 6 cube faces.
QgsRange which stores a range of double values.
Definition qgsrange.h:233
bool isInfinite() const
Returns true if the range consists of all possible values.
Definition qgsrange.h:287
Container class that holds different objects related to forward rendering.
Qt3DRender::QLayer * transparentObjectLayer()
Returns a layer object used to indicate that the object is transparent.
void addClipPlanes(int nrClipPlanes)
Setups nrClipPlanes clip planes in the forward pass to enable OpenGL clipping.
void removeClipPlanes()
Disables OpenGL clipping.
Container class that holds different objects related to frame graphs of 3D scenes.
void updateShadowSettings(const QgsShadowSettings &shadowSettings, const QList< QgsLightSource * > &lightSources)
Updates shadow bias, light and texture size according to shadowSettings and lightSources.
void setupShadowMapDebugging(bool enabled, Qt::Corner corner, double size)
Sets the shadow map debugging view port.
void setAmbientOcclusionIntensity(float intensity)
Sets the ambient occlusion intensity.
void setDebugOverlayEnabled(bool enabled)
Sets whether debug overlay is enabled.
void setupDepthMapDebugging(bool enabled, Qt::Corner corner, double size)
Sets the depth map debugging view port.
QgsForwardRenderView & forwardRenderView()
Returns forward renderview.
void setAmbientOcclusionThreshold(float threshold)
Sets the ambient occlusion threshold.
void setupEyeDomeLighting(bool enabled, double strength, int distance)
Sets eye dome lighting shading related settings.
void setAmbientOcclusionEnabled(bool enabled)
Sets whether Screen Space Ambient Occlusion will be enabled.
void setAmbientOcclusionRadius(float radius)
Sets the ambient occlusion radius.
Base class for light sources in 3d scenes.
virtual QgsDoubleRange calculateZRange(QgsMapLayer *layer) const
Attempts to calculate the overall elevation or z range for the specified layer, using the settings de...
Base class for storage of map layer temporal properties.
Base class for all map layer types.
Definition qgsmaplayer.h:77
QString name
Definition qgsmaplayer.h:81
QgsAbstract3DRenderer * renderer3D() const
Returns 3D renderer associated with the layer.
void request3DUpdate()
Signal emitted when a layer requires an update in any 3D maps.
void renderer3DChanged()
Signal emitted when 3D renderer associated with the layer has changed.
Qgis::LayerType type
Definition qgsmaplayer.h:87
void rendererChanged()
Signal emitted when renderer is changed.
virtual QgsMapLayerTemporalProperties * temporalProperties()
Returns the layer's temporal properties.
void layerModified()
Emitted when modifications has been done on layer.
Base class for all materials used within QGIS 3D views.
Definition qgsmaterial.h:39
3D symbol that draws mesh geometry as planar triangles.
double verticalScale() const
Returns mesh vertical scale.
int verticalDatasetGroupIndex() const
Returns the index of the dataset group that will be used to render the vertical component of the 3D m...
void setMaximumTextureSize(int maximumTextureSize)
Sets the maximum texture size supported by the hardware Used to store the GL_MAX_TEXTURE_SIZE value t...
QgsMesh3DSymbol * clone() const override SIP_FACTORY
Returns a new instance of the symbol with the same settings.
A collection of dataset group metadata such as whether the data is vector or scalar,...
double minimum() const
Returns minimum scalar value/vector magnitude present for whole dataset group.
double maximum() const
Returns maximum scalar value/vector magnitude present for whole dataset group.
3D renderer that renders all mesh triangles of a mesh layer.
void setSymbol(QgsMesh3DSymbol *symbol)
Sets 3D symbol associated with the renderer.
const QgsMesh3DSymbol * symbol() const
Returns 3D symbol associated with the renderer.
void setLayer(QgsMeshLayer *layer)
Sets vector layer associated with the renderer.
Represents a mesh layer supporting display of data on structured or unstructured meshes.
QgsMeshDatasetGroupMetadata datasetGroupMetadata(const QgsMeshDatasetIndex &index) const
Returns the dataset groups metadata.
virtual void showMessage(bool blocking=true)=0
display the message to the user and deletes itself
A skybox constructed from a panoramic image.
Material of the billboard rendering for points in 3D map view.
void setViewportSize(const QSizeF size)
Set the size of the view port.
3D symbol that draws point geometries as 3D objects using one of the predefined shapes.
Qgis::Point3DShape shape() const
Returns 3D shape for points.
QVariant shapeProperty(const QString &property) const
Returns the value for a specific shape property.
3D renderer that renders all points from a point cloud layer.
void setLayer(QgsPointCloudLayer *layer)
Sets point cloud layer associated with the renderer.
Represents a map layer supporting display of point clouds.
QgsMapLayerElevationProperties * elevationProperties() override
Returns the layer's elevation properties.
void subsetStringChanged()
Emitted when the layer's subset string has changed.
Represents a 2D point.
Definition qgspointxy.h:60
double y
Definition qgspointxy.h:64
double x
Definition qgspointxy.h:63
A template based class for storing ranges (lower to upper values).
Definition qgsrange.h:46
T lower() const
Returns the lower bound of the range.
Definition qgsrange.h:78
T upper() const
Returns the upper bound of the range.
Definition qgsrange.h:85
bool isEmpty() const
Returns true if the range is empty, ie the lower bound equals (or exceeds) the upper bound and either...
Definition qgsrange.h:125
A representation of a ray in 3D.
Definition qgsray3d.h:31
A rectangle specified with double values.
double xMinimum
double yMinimum
double xMaximum
double yMaximum
QgsPointXY center
Rule-based 3D renderer.
QList< QgsRuleBased3DRenderer::Rule * > RuleList
Contains the configuration of a skybox entity.
QMap< QString, QString > cubeMapFacesPaths() const
Returns a map containing the path of each texture specified by the user.
QgsSkyboxEntity::SkyboxType skyboxType() const
Returns the type of the skybox.
QString panoramicTexturePath() const
Returns the panoramic texture path of a skybox of type "Panormaic skybox".
void remoteSourceFetched(const QString &url)
Emitted when the cache has finished retrieving a 3D model from a remote url.
virtual QgsBox3D rootChunkBox3D(const Qgs3DMapSettings &map) const
Returns 3D box (in map coordinates) of the root chunk.
virtual float rootChunkError(const Qgs3DMapSettings &map) const
Returns error of the root chunk in world coordinates.
virtual QgsRectangle rootChunkExtent() const =0
extent of the terrain's root chunk in terrain's CRS
3D renderer that renders content of a tiled scene layer.
void setLayer(QgsTiledSceneLayer *layer)
Sets tiled scene layer associated with the renderer.
Represents a map layer supporting display of tiled scene objects.
QgsMapLayerElevationProperties * elevationProperties() override
Returns the layer's elevation properties.
A 3D vector (similar to QVector3D) with the difference that it uses double precision instead of singl...
Definition qgsvector3d.h:30
double y() const
Returns Y coordinate.
Definition qgsvector3d.h:49
double z() const
Returns Z coordinate.
Definition qgsvector3d.h:51
QString toString(int precision=17) const
Returns a string representation of the 3D vector.
double x() const
Returns X coordinate.
Definition qgsvector3d.h:47
3D renderer that renders all features of a vector layer with the same 3D symbol.
Represents a vector layer which manages a vector based dataset.
Q_INVOKABLE Qgis::GeometryType geometryType() const
Returns point, line or polygon.
void selectionChanged(const QgsFeatureIds &selected, const QgsFeatureIds &deselected, bool clearAndSelect)
Emitted when selection was changed.
On-screen 3D engine: it creates an OpenGL window (QWindow) and displays rendered 3D scenes there.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c
bool qgsFloatNear(float a, float b, float epsilon=4 *FLT_EPSILON)
Compare two floats (but allow some difference)
Definition qgis.h:6297
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:41