QGIS API Documentation 4.0.0-Norrköping (1ddcee3d0e4)
Loading...
Searching...
No Matches
qgs3daxisrenderview.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgs3daxisrenderview.cpp
3 --------------------------------------
4 Date : June 2024
5 Copyright : (C) 2024 by Benoit De Mezzo
6 Email : benoit dot de dot mezzo at oslandia 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 "qgs3daxisrenderview.h"
17
18#include "qgs3daxis.h"
19#include "qgs3dmapcanvas.h"
20#include "qgs3dmapsettings.h"
21#include "qgscameracontroller.h"
22#include "qgsmapsettings.h"
23
24#include <Qt3DRender/QCamera>
25#include <Qt3DRender/QCameraSelector>
26#include <Qt3DRender/QClearBuffers>
27#include <Qt3DRender/QLayer>
28#include <Qt3DRender/QLayerFilter>
29#include <Qt3DRender/QSortPolicy>
30#include <Qt3DRender/QViewport>
31#include <Qt3DRender/qsubtreeenabler.h>
32
34 const QString &viewName,
35 Qgs3DMapCanvas *canvas, //
36 QgsCameraController *cameraCtrl,
37 Qgs3DMapSettings *settings, //
38 Qgs3DAxis *axis3D
39)
40 : QgsAbstractRenderView( viewName )
41 , mCanvas( canvas )
42 , mMapSettings( settings )
43 , m3DAxis( axis3D )
44{
45 mViewport = new Qt3DRender::QViewport( mRendererEnabler );
46 mViewport->setObjectName( mViewName + "::Viewport" );
47
48 mObjectLayer = new Qt3DRender::QLayer;
49 mObjectLayer->setObjectName( mViewName + "::ObjectLayer" );
50 mObjectLayer->setRecursive( true );
51
52 mLabelLayer = new Qt3DRender::QLayer;
53 mLabelLayer->setObjectName( mViewName + "::LabelLayer" );
54 mLabelLayer->setRecursive( true );
55
56 // render pass for the object (axis or cube)
57 Qt3DRender::QLayerFilter *objectFilter = new Qt3DRender::QLayerFilter( mViewport );
58 objectFilter->addLayer( mObjectLayer );
59
60 mObjectCamera = new Qt3DRender::QCamera;
61 mObjectCamera->setObjectName( mViewName + "::ObjectCamera" );
62 mObjectCamera->setProjectionType( cameraCtrl->camera()->projectionType() );
63 mObjectCamera->lens()->setFieldOfView( cameraCtrl->camera()->lens()->fieldOfView() * 0.5f );
64
65 Qt3DRender::QCameraSelector *cameraSelector = new Qt3DRender::QCameraSelector( objectFilter );
66 cameraSelector->setCamera( mObjectCamera );
67
68 // This ensures to have the label (Text2DEntity) rendered after the other objects and therefore
69 // avoid any transparency issue on the label.
70 Qt3DRender::QSortPolicy *objectSortPolicy = new Qt3DRender::QSortPolicy( cameraSelector );
71 QVector<Qt3DRender::QSortPolicy::SortType> objectSortTypes = QVector<Qt3DRender::QSortPolicy::SortType>();
72 objectSortTypes << Qt3DRender::QSortPolicy::BackToFront;
73 objectSortPolicy->setSortTypes( objectSortTypes );
74
75 Qt3DRender::QClearBuffers *objectClearBuffers = new Qt3DRender::QClearBuffers( objectSortPolicy );
76 objectClearBuffers->setBuffers( Qt3DRender::QClearBuffers::DepthBuffer );
77
78 Qt3DRender::QLayerFilter *labelFilter = new Qt3DRender::QLayerFilter( mViewport );
79 labelFilter->addLayer( mLabelLayer );
80
81 mLabelCamera = new Qt3DRender::QCamera;
82 mLabelCamera->setObjectName( mViewName + "::LabelCamera" );
83 mLabelCamera->setProjectionType( Qt3DRender::QCameraLens::ProjectionType::OrthographicProjection );
84
85 Qt3DRender::QCameraSelector *labelCameraSelector = new Qt3DRender::QCameraSelector( labelFilter );
86 labelCameraSelector->setCamera( mLabelCamera );
87
88 // this ensures to have the label (Text2DEntity) rendered after the other objects and therefore
89 // avoid any transparency issue on the label.
90 Qt3DRender::QSortPolicy *labelSortPolicy = new Qt3DRender::QSortPolicy( labelCameraSelector );
91 QVector<Qt3DRender::QSortPolicy::SortType> labelSortTypes = QVector<Qt3DRender::QSortPolicy::SortType>();
92 labelSortTypes << Qt3DRender::QSortPolicy::BackToFront;
93 labelSortPolicy->setSortTypes( labelSortTypes );
94
95 Qt3DRender::QClearBuffers *clearBuffers = new Qt3DRender::QClearBuffers( labelSortPolicy );
96 clearBuffers->setBuffers( Qt3DRender::QClearBuffers::DepthBuffer );
97
98 // update viewport size
100}
101
102Qt3DRender::QViewport *Qgs3DAxisRenderView::viewport() const
103{
104 return mViewport;
105}
106
107Qt3DRender::QLayer *Qgs3DAxisRenderView::objectLayer() const
108{
109 return mObjectLayer;
110}
111
112Qt3DRender::QLayer *Qgs3DAxisRenderView::labelLayer() const
113{
114 return mLabelLayer;
115}
116
117Qt3DRender::QCamera *Qgs3DAxisRenderView::objectCamera() const
118{
119 return mObjectCamera;
120}
121
122Qt3DRender::QCamera *Qgs3DAxisRenderView::labelCamera() const
123{
124 return mLabelCamera;
125}
126
127void Qgs3DAxisRenderView::updateWindowResize( int width, int height )
128{
129 onViewportSizeUpdate( width, height );
130}
131
132
133void Qgs3DAxisRenderView::onViewportSizeUpdate( int width, int height )
134{
135 Qgs3DAxisSettings settings = mMapSettings->get3DAxisSettings();
136 double windowWidth = static_cast<double>( width < 0 ? mCanvas->width() : width );
137 double windowHeight = static_cast<double>( height < 0 ? mCanvas->height() : height );
138
139 if ( 2 <= QgsLogger::debugLevel() )
140 {
141 QgsMapSettings set;
142 QgsDebugMsgLevel( QString( "onViewportSizeUpdate window w/h: %1px / %2px" ).arg( windowWidth ).arg( windowHeight ), 2 );
144 QString( "onViewportSizeUpdate window physicalDpi %1 (%2, %3)" )
145 .arg( mCanvas->screen()->physicalDotsPerInch() )
146 .arg( mCanvas->screen()->physicalDotsPerInchX() )
147 .arg( mCanvas->screen()->physicalDotsPerInchY() ),
148 2
149 );
151 QString( "onViewportSizeUpdate window logicalDotsPerInch %1 (%2, %3)" )
152 .arg( mCanvas->screen()->logicalDotsPerInch() )
153 .arg( mCanvas->screen()->logicalDotsPerInchX() )
154 .arg( mCanvas->screen()->logicalDotsPerInchY() ),
155 2
156 );
157
158 QgsDebugMsgLevel( QString( "onViewportSizeUpdate window pixel ratio %1" ).arg( mCanvas->screen()->devicePixelRatio() ), 2 );
159
160 QgsDebugMsgLevel( QString( "onViewportSizeUpdate set pixel ratio %1" ).arg( set.devicePixelRatio() ), 2 );
161 QgsDebugMsgLevel( QString( "onViewportSizeUpdate set outputDpi %1" ).arg( set.outputDpi() ), 2 );
162 QgsDebugMsgLevel( QString( "onViewportSizeUpdate set dpiTarget %1" ).arg( set.dpiTarget() ), 2 );
163 }
164
165 // default viewport size in pixel according to 92 dpi
166 double defaultViewportPixelSize = ( ( double ) settings.defaultViewportSize() / 25.4 ) * 92.0;
167
168 // computes the viewport size according to screen dpi but as the viewport size growths too fast
169 // then we limit the growth by using a factor on the dpi difference.
170 double viewportPixelSize = defaultViewportPixelSize + ( ( double ) settings.defaultViewportSize() / 25.4 ) * ( mCanvas->screen()->physicalDotsPerInch() - 92.0 ) * 0.7;
171 QgsDebugMsgLevel( QString( "onViewportSizeUpdate viewportPixelSize %1" ).arg( viewportPixelSize ), 2 );
172 double widthRatio = viewportPixelSize / windowWidth;
173 double heightRatio = widthRatio * windowWidth / windowHeight;
174
175 QgsDebugMsgLevel( QString( "3DAxis viewport ratios width: %1% / height: %2%" ).arg( widthRatio * 100.0 ).arg( heightRatio * 100.0 ), 2 );
176
177 if ( heightRatio * windowHeight < viewportPixelSize )
178 {
179 heightRatio = viewportPixelSize / windowHeight;
180 widthRatio = heightRatio * windowHeight / windowWidth;
181 QgsDebugMsgLevel( QString( "3DAxis viewport, height too small, ratios adjusted to width: %1% / height: %2%" ).arg( widthRatio * 100.0 ).arg( heightRatio * 100.0 ), 2 );
182 }
183
184 if ( heightRatio > settings.maxViewportRatio() || widthRatio > settings.maxViewportRatio() )
185 {
186 QgsDebugMsgLevel( QString( "3DAxis viewport takes too much place into the 3d view, disabling it (maxViewportRatio: %1)." ).arg( settings.maxViewportRatio() ), 2 );
187 // take too much place into the 3d view
188 mViewport->setEnabled( false );
189 m3DAxis->onViewportScaleFactorChanged( 0.0 );
190 }
191 else
192 {
193 if ( !mViewport->isEnabled() )
194 {
195 // will be used to adjust the axis label translations/sizes
196 m3DAxis->onViewportScaleFactorChanged( viewportPixelSize / defaultViewportPixelSize );
197 }
198 mViewport->setEnabled( true );
199
200 float xRatio = 1.0f;
201 float yRatio = 1.0f;
202 if ( settings.horizontalPosition() == Qt::AnchorPoint::AnchorLeft )
203 xRatio = 0.0f;
204 else if ( settings.horizontalPosition() == Qt::AnchorPoint::AnchorHorizontalCenter )
205 xRatio = 0.5f - static_cast<float>( widthRatio ) / 2.0f;
206 else
207 xRatio = 1.0f - static_cast<float>( widthRatio );
208
209 if ( settings.verticalPosition() == Qt::AnchorPoint::AnchorTop )
210 yRatio = 0.0f;
211 else if ( settings.verticalPosition() == Qt::AnchorPoint::AnchorVerticalCenter )
212 yRatio = 0.5f - static_cast<float>( heightRatio ) / 2.0f;
213 else
214 yRatio = 1.0f - static_cast<float>( heightRatio );
215
216 QgsDebugMsgLevel( QString( "Qgs3DAxis: update viewport: %1 x %2 x %3 x %4" ).arg( xRatio ).arg( yRatio ).arg( widthRatio ).arg( heightRatio ), 2 );
217 mViewport->setNormalizedRect( QRectF( xRatio, yRatio, widthRatio, heightRatio ) );
218
219 if ( settings.mode() == Qgs3DAxisSettings::Mode::Crs )
220 {
221 const float halfWidthSize = static_cast<float>( windowWidth * widthRatio / 2.0 );
222 const float halfHeightSize = static_cast<float>( windowWidth * widthRatio / 2.0 );
223 // clang-format off
224 mLabelCamera->lens()->setOrthographicProjection(
225 -halfWidthSize, halfWidthSize,
226 -halfHeightSize, halfHeightSize,
227 mLabelCamera->lens()->nearPlane(), mLabelCamera->lens()->farPlane()
228 );
229 // clang-format on
230 }
231 }
232}
233
235{
236 Qgs3DAxisSettings axisSettings = mMapSettings->get3DAxisSettings();
237 axisSettings.setHorizontalPosition( pos );
238 mMapSettings->set3DAxisSettings( axisSettings );
240}
241
243{
244 Qgs3DAxisSettings axisSettings = mMapSettings->get3DAxisSettings();
245 axisSettings.setVerticalPosition( pos );
246 mMapSettings->set3DAxisSettings( axisSettings );
248}
Qt3DRender::QLayer * labelLayer() const
Returns the layer to be used by entities to be included in the label renderpass.
void onVerticalPositionChanged(Qt::AnchorPoint position)
Updates viewport vertical position.
Qt3DRender::QCamera * labelCamera() const
Returns camera used for billboarded labels.
Qt3DRender::QLayer * objectLayer() const
Returns main object layer.
Qgs3DAxisRenderView(const QString &viewName, Qgs3DMapCanvas *canvas, QgsCameraController *cameraCtrl, Qgs3DMapSettings *settings, Qgs3DAxis *axis3D)
Constructor for Qgs3DAxisRenderView with the specified parent object.
Qt3DRender::QViewport * viewport() const
Returns the viewport associated to this renderview.
void onViewportSizeUpdate(int width=-1, int height=-1)
Updates viewport size. Uses canvas size by default.
Qt3DRender::QCamera * objectCamera() const
Returns main object camera (used for axis or cube).
void updateWindowResize(int width, int height) override
Called when 3D window is resized.
void onHorizontalPositionChanged(Qt::AnchorPoint position)
Updates viewport horizontal position.
Contains the configuration of a 3d axis.
double maxViewportRatio() const
Returns the maximal axis viewport ratio (see Qt3DRender::QViewport::normalizedRect()).
@ Crs
Respect CRS directions.
Qt::AnchorPoint verticalPosition() const
Returns the vertical position for the 3d axis.
void setHorizontalPosition(Qt::AnchorPoint position)
Sets the horizontal position for the 3d axis.
int defaultViewportSize() const
Returns the default axis viewport size in millimeters.
Qgs3DAxisSettings::Mode mode() const
Returns the type of the 3daxis.
Qt::AnchorPoint horizontalPosition() const
Returns the horizontal position for the 3d axis.
void setVerticalPosition(Qt::AnchorPoint position)
Sets the vertical position for the 3d axis.
Display 3D ortho axis in the main 3D view.
Definition qgs3daxis.h:68
Convenience wrapper to simplify the creation of a 3D window ready to be used with QGIS.
Definition of the world.
Qt3DRender::QSubtreeEnabler * mRendererEnabler
QgsAbstractRenderView(const QString &viewName)
Constructor for QgsAbstractRenderView with the specified parent object.
Object that controls camera movement based on user input.
Qt3DRender::QCamera * camera() const
Returns camera that is being controlled.
static int debugLevel()
Reads the environment variable QGIS_DEBUG and converts it to int.
Definition qgslogger.h:144
Contains configuration for rendering maps.
double dpiTarget() const
Returns the target DPI (dots per inch) to be taken into consideration when rendering.
float devicePixelRatio() const
Returns the device pixel ratio.
double outputDpi() const
Returns the DPI (dots per inch) used for conversion between real world units (e.g.
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63