QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskSceneTexture.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskSceneTexture.h"
7#include "QskTreeNode.h"
8#include "QskInternalMacros.h"
9
10#include <qmath.h>
11
12QSK_QT_PRIVATE_BEGIN
13#include <private/qquickwindow_p.h>
14#include <private/qsgtexture_p.h>
15
16#define QT_BUILD_QUICK_LIB // suppress Qt5 warnings
17#include <private/qsgbatchrenderer_p.h>
18#undef QT_BUILD_QUICK_LIB
19
20QSK_QT_PRIVATE_END
21
22/*
23 With Qt 5.15 Rhi can optionally be enbled by setting "export QSG_RHI=1".
24 So we need to have a native QOpenGL implementation and one using
25 the Rhi abstraction layer. For Qt6 we can rely on Rhi.
26 Once Qt5 support has been dropped we can eliminate this #ifdef jungle
27 */
28
29#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
30#include <qopenglframebufferobject.h>
31#endif
32
33static int qskRenderOrderCompare( const QSGNode* rootNode,
34 const QSGNode* node1, const QSGNode* node2 )
35{
36 if ( rootNode == node1 )
37 return 1;
38
39 if ( rootNode == node2 )
40 return -1;
41
42 for ( auto node = rootNode->firstChild();
43 node != nullptr; node = node->nextSibling() )
44 {
45 const auto ret = qskRenderOrderCompare( node, node1, node2 );
46 if ( ret )
47 return ret;
48 }
49
50 return 0;
51}
52
53namespace
54{
55#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
56 inline QSGRendererInterface::RenderMode contextRenderMode(
57 QSGDefaultRenderContext* context )
58 {
59 return context->useDepthBufferFor2D()
60 ? QSGRendererInterface::RenderMode2D
61 : QSGRendererInterface::RenderMode2DNoDepthBuffer;
62 }
63#endif
64
65 class Renderer final : public QSGBatchRenderer::Renderer
66 {
67 using Inherited = QSGBatchRenderer::Renderer;
68
69 public:
70 Renderer( QskSceneTexture*, QSGDefaultRenderContext* );
71 ~Renderer() override;
72
73#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
74 inline int textureId() const { return m_fbo ? m_fbo->texture() : 0; }
75
76 inline void renderScene()
77 {
78 class Bindable : public QSGBindable
79 {
80 public:
81 Bindable( QOpenGLFramebufferObject* fbo ) : m_fbo( fbo ) {}
82 void bind() const override { m_fbo->bind(); }
83 private:
84 QOpenGLFramebufferObject* m_fbo;
85 };
86
87 Inherited::renderScene( Bindable( m_fbo ) );
88 }
89#endif
90
91 inline QRhiTexture* rhiTexture() const { return m_rhiTexture; }
92
93 inline bool isDirty() const { return m_dirty; }
94
95 void setFinalNode( QSGTransformNode* );
96
97 void setProjection( const QRectF& );
98 void setTextureSize( const QSize& );
99 QSize textureSize() const;
100
101 protected:
102 void nodeChanged( QSGNode*, QSGNode::DirtyState ) override;
103 void render() override;
104
105 private:
106 void createTarget( const QSize& );
107 void clearTarget();
108 void markDirty();
109
110 QSGTransformNode* m_finalNode = nullptr;
111 QskSceneTexture* m_texture = nullptr;
112
113#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
114 QOpenGLFramebufferObject* m_fbo = nullptr;
115#endif
116
117#if QT_VERSION < QT_VERSION_CHECK( 6, 4, 0 )
118 struct RenderTarget
119 {
120 QRhiRenderTarget* rt = nullptr;
121 QRhiRenderPassDescriptor* rpDesc = nullptr;
122 QRhiCommandBuffer* cb = nullptr;
123 } m_rt;
124#endif
125 QRhiTexture* m_rhiTexture = nullptr;
126
127 bool m_dirty = true;
128 };
129
130 Renderer::Renderer( QskSceneTexture* texture, QSGDefaultRenderContext* context )
131#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
132 : Inherited( context )
133#else
134 : Inherited( context, contextRenderMode( context ) )
135#endif
136 , m_texture( texture )
137 {
138 setClearColor( Qt::transparent );
139
140 connect( this, &QSGRenderer::sceneGraphChanged,
141 this, &Renderer::markDirty );
142 }
143
144 Renderer::~Renderer()
145 {
146 clearTarget();
147 }
148
149 void Renderer::setFinalNode( QSGTransformNode* node )
150 {
151 if ( node != m_finalNode )
152 {
153 m_finalNode = node;
154 markDirty();
155 }
156 }
157
158 void Renderer::setProjection( const QRectF& rect )
159 {
160 bool flipFramebuffer = true;
161 bool flipMatrix = false;
162
163 if ( const auto rhi = context()->rhi() )
164 {
165 flipFramebuffer = rhi->isYUpInFramebuffer();
166 flipMatrix = !rhi->isYUpInNDC();
167 }
168
169 auto r = rect;
170
171 if ( flipFramebuffer )
172 {
173 r.moveTop( r.bottom() );
174 r.setHeight( -r.height() );
175 }
176
177 MatrixTransformFlags matrixFlags;
178
179 if ( flipMatrix )
180 matrixFlags |= QSGAbstractRenderer::MatrixTransformFlipY;
181
182 setProjectionMatrixToRect( r, matrixFlags );
183 }
184
185 void Renderer::setTextureSize( const QSize& size )
186 {
187 if ( const auto rhi = context()->rhi() )
188 {
189 if ( m_rt.rt && m_rt.rt->pixelSize() != size )
190 clearTarget();
191
192 if ( m_rt.rt == nullptr )
193 createTarget( size );
194 }
195 else
196 {
197#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
198 if ( m_fbo && m_fbo->size() != size )
199 clearTarget();
200
201 if ( m_fbo == nullptr )
202 createTarget( size );
203#endif
204 }
205
206 const QRect r( 0, 0, size.width(), size.height() );
207
208 setDeviceRect( r );
209 setViewportRect( r );
210 }
211
212 QSize Renderer::textureSize() const
213 {
214#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
215 if ( m_fbo ) return m_fbo->size();
216#else
217 if( m_rt.rt ) return m_rt.rt->pixelSize();
218#endif
219 return QSize();
220 }
221
222 void Renderer::render()
223 {
224 m_dirty = false;
225
226 qskTryBlockTrailingNodes( m_finalNode, rootNode(), true, false );
227
228#if 0
229 static int counter = 0;
230 qDebug() << ++counter;
231 QSGNodeDumper::dump( rootNode() );
232#endif
233 Inherited::render();
234 qskTryBlockTrailingNodes( m_finalNode, rootNode(), false, false );
235 }
236
237 void Renderer::nodeChanged( QSGNode* node, QSGNode::DirtyState state )
238 {
239 /*
240 No need to update the texture for changes of nodes behind
241 the final node.
242
243 Unfortunately QQuickWindow does not update the scene graph in
244 rendering order and we might be called for relevant nodes after
245 the texture has already been updated. In these situations we
246 update the texture twice. Not so good ...
247 */
248 if ( qskRenderOrderCompare( rootNode(), node, m_finalNode ) > 0 )
249 {
250 // triggering QSGRenderer::sceneGraphChanged signals
251 Inherited::nodeChanged( node, state );
252 }
253 }
254
255 void Renderer::markDirty()
256 {
257 if ( !m_dirty )
258 {
259 m_dirty = true;
260 Q_EMIT m_texture->updateRequested();
261 }
262 }
263
264 void Renderer::createTarget( const QSize& size )
265 {
266 if ( const auto rhi = context()->rhi() )
267 {
268 auto flags = QRhiTexture::RenderTarget | QRhiTexture::UsedAsTransferSource;
269
270 m_rhiTexture = rhi->newTexture( QRhiTexture::RGBA8, size, 1, flags );
271#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
272 m_rhiTexture->build();
273#else
274 m_rhiTexture->create();
275#endif
276
277 QRhiColorAttachment color0( m_rhiTexture );
278 auto target = rhi->newTextureRenderTarget( { color0 } );
279
280 target->setRenderPassDescriptor(
281 target->newCompatibleRenderPassDescriptor() );
282
283#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
284 target->build();
285#else
286 target->create();
287#endif
288
289 m_rt.rt = target;
290 m_rt.rpDesc = target->renderPassDescriptor();
291
292 auto defaultContext = qobject_cast< QSGDefaultRenderContext* >( context() );
293 m_rt.cb = defaultContext->currentFrameCommandBuffer();
294
295#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
296 setRenderTarget( m_rt.rt );
297 setCommandBuffer( m_rt.cb );
298 setRenderPassDescriptor( m_rt.rpDesc );
299#endif
300 }
301 else
302 {
303#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
304 QOpenGLFramebufferObjectFormat format;
305 format.setInternalTextureFormat( GL_RGBA8 );
306 format.setSamples( 0 );
307 format.setAttachment( QOpenGLFramebufferObject::CombinedDepthStencil );
308
309 m_fbo = new QOpenGLFramebufferObject( size, format );
310#endif
311 }
312 }
313
314 void Renderer::clearTarget()
315 {
316 if ( const auto rhi = context()->rhi() )
317 {
318 delete m_rt.rt;
319 m_rt.rt = nullptr;
320
321 delete m_rt.rpDesc;
322 m_rt.rpDesc = nullptr;
323
324 delete m_rhiTexture;
325 m_rhiTexture = nullptr;
326 }
327 else
328 {
329#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
330 delete m_fbo;
331 m_fbo = nullptr;
332#endif
333 }
334 }
335}
336
337class QskSceneTexturePrivate final : public QSGTexturePrivate
338{
339 public:
340 QskSceneTexturePrivate( const QQuickWindow* window, QskSceneTexture* texture )
341#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
342 : QSGTexturePrivate()
343#else
344 : QSGTexturePrivate( texture )
345#endif
346 , devicePixelRatio( window->effectiveDevicePixelRatio() )
347 {
348 Q_UNUSED( texture );
349
350 // Qt5 needs the extra const_cast
351 auto dw = QQuickWindowPrivate::get( const_cast< QQuickWindow* >( window ) );
352 context = dynamic_cast< QSGDefaultRenderContext* >( dw->context );
353 }
354
355#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
356 int comparisonKey() const override
357 {
358 if ( renderer )
359 {
360 if ( renderer->textureId() )
361 return renderer->textureId();
362
363 if ( renderer->rhiTexture() )
364 return int( qintptr( renderer->rhiTexture() ) );
365 }
366
367 return int( qintptr( this ) );
368 }
369
370 QRhiTexture *rhiTexture() const override
371 { return renderer ? renderer->rhiTexture() : nullptr; }
372#endif
373
374 QSize pixelSize() const
375 {
376 QSize size( qCeil( rect.width() ), qCeil( rect.height() ) );
377 size *= devicePixelRatio;
378
379 const QSize minSize = context->sceneGraphContext()->minimumFBOSize();
380
381 while ( size.width() < minSize.width() )
382 size.rwidth() *= 2;
383
384 while ( size.height() < minSize.height() )
385 size.rheight() *= 2;
386
387 return size;
388 }
389
390 QRectF rect;
391 const qreal devicePixelRatio;
392
393 Renderer* renderer = nullptr;
394 QSGDefaultRenderContext* context = nullptr;
395};
396
397QskSceneTexture::QskSceneTexture( const QQuickWindow* window )
398 : Inherited( *new QskSceneTexturePrivate( window, this ) )
399{
400 Q_ASSERT( d_func()->context );
401}
402
403QskSceneTexture::~QskSceneTexture()
404{
405 delete d_func()->renderer;
406}
407
408QSize QskSceneTexture::textureSize() const
409{
410 Q_D( const QskSceneTexture );
411 return d->renderer ? d->renderer->textureSize() : QSize();
412}
413
414void QskSceneTexture::render( const QSGRootNode* rootNode,
415 const QSGTransformNode* finalNode, const QRectF& rect )
416{
417 Q_D( QskSceneTexture );
418
419 d->rect = rect;
420
421 if ( d->renderer == nullptr )
422 {
423 d->renderer = new Renderer( this, d->context );
424 d->renderer->setDevicePixelRatio( d->devicePixelRatio );
425 }
426
427 d->renderer->setRootNode( const_cast< QSGRootNode* >( rootNode ) );
428 d->renderer->setFinalNode( const_cast< QSGTransformNode* >( finalNode ) );
429
430 d->renderer->setProjection( d->rect );
431 d->renderer->setTextureSize( d->pixelSize() );
432 d->renderer->renderScene();
433}
434
435bool QskSceneTexture::isDirty() const
436{
437 Q_D( const QskSceneTexture );
438 return d->renderer ? d->renderer->isDirty() : true;
439}
440
441QRectF QskSceneTexture::normalizedTextureSubRect() const
442{
443 return QRectF( 0, 1, 1, -1 );
444}
445
446bool QskSceneTexture::hasAlphaChannel() const
447{
448 return false;
449}
450
451bool QskSceneTexture::hasMipmaps() const
452{
453 return false;
454}
455
456#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
457
458void QskSceneTexture::bind()
459{
460 if ( d_func()->rhiTexture() == nullptr )
461 {
462 auto funcs = QOpenGLContext::currentContext()->functions();
463 funcs->glBindTexture( GL_TEXTURE_2D, textureId() );
464
465 updateBindOptions();
466 }
467}
468
469int QskSceneTexture::textureId() const
470{
471 Q_D( const QskSceneTexture );
472 return d->renderer ? d->renderer->textureId() : 0;
473}
474
475#else
476
477qint64 QskSceneTexture::comparisonKey() const
478{
479 return qint64( rhiTexture() );
480}
481
482QRhiTexture* QskSceneTexture::rhiTexture() const
483{
484 Q_D( const QskSceneTexture );
485 return d->renderer ? d->renderer->rhiTexture() : nullptr;
486}
487
488#endif
489
490#include "moc_QskSceneTexture.cpp"