QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskTextureRenderer.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskTextureRenderer.h"
7#include "QskQuick.h"
8#include "QskInternalMacros.h"
9
10#include <qopenglcontext.h>
11#include <qopenglframebufferobject.h>
12#include <qopenglpaintdevice.h>
13
14#include <qimage.h>
15#include <qpainter.h>
16
17#include <qquickwindow.h>
18
19QSK_QT_PRIVATE_BEGIN
20#include <private/qsgplaintexture_p.h>
21#include <private/qopenglframebufferobject_p.h>
22QSK_QT_PRIVATE_END
23
24#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
25 #include <qquickopenglutils.h>
26#endif
27
28static GLuint qskTakeTexture( QOpenGLFramebufferObject& fbo )
29{
30 /*
31 See https://bugreports.qt.io/browse/QTBUG-103929
32
33 As we create a FBO for each update of a node we can't live
34 without having this ( ugly ) workaround.
35 */
36 class MyFBO
37 {
38 public:
39 virtual ~MyFBO() = default;
40 QScopedPointer< QOpenGLFramebufferObjectPrivate > d_ptr;
41 };
42
43 static_assert( sizeof( MyFBO ) == sizeof( QOpenGLFramebufferObject ),
44 "Bad cast: QOpenGLFramebufferObject does not match" );
45
46 auto& attachment = reinterpret_cast< MyFBO* >( &fbo )->d_ptr->colorAttachments[0];
47 auto guard = attachment.guard;
48
49 const auto textureId = fbo.takeTexture();
50
51 if ( guard )
52 {
53 class MyGuard : public QOpenGLSharedResourceGuard
54 {
55 public:
56 void invalidateTexture() { invalidateResource(); }
57 };
58
59 reinterpret_cast< MyGuard* >( guard )->invalidateTexture();
60 }
61
62 attachment.guard = guard;
63
64 return textureId;
65}
66
67bool QskTextureRenderer::isOpenGLWindow( const QQuickWindow* window )
68{
69 if ( window == nullptr )
70 return false;
71
72 const auto renderer = window->rendererInterface();
73 switch( renderer->graphicsApi() )
74 {
75 case QSGRendererInterface::OpenGL:
76#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
77 case QSGRendererInterface::OpenGLRhi:
78#endif
79 return true;
80
81 default:
82 return false;
83 }
84}
85
86void QskTextureRenderer::setTextureId( QQuickWindow* window,
87 quint32 textureId, const QSize& size, QSGTexture* texture )
88{
89 auto plainTexture = qobject_cast< QSGPlainTexture* >( texture );
90 if ( plainTexture == nullptr )
91 return;
92
93 auto rhi = qskRenderingHardwareInterface( window );
94
95#if QT_VERSION >= QT_VERSION_CHECK( 6, 4, 0 )
96
97 const uint nativeFormat = 0;
98 plainTexture->setTextureFromNativeTexture(
99 rhi, quint64( textureId ), 0, nativeFormat, size, {}, {} );
100
101#elif QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
102 plainTexture->setTextureFromNativeTexture(
103 rhi, quint64( textureId ), 0, size, {}, {} );
104#else
105 if ( rhi )
106 {
107 // enabled with: "export QSG_RHI=1"
108 plainTexture->setTextureFromNativeObject( rhi,
109 QQuickWindow::NativeObjectTexture, &textureId, 0, size, false );
110 }
111 else
112 {
113 plainTexture->setTextureId( textureId );
114 plainTexture->setTextureSize( size );
115 }
116#endif
117}
118
119quint32 QskTextureRenderer::createPaintedTextureGL(
120 QQuickWindow* window, const QSize& size, QskTextureRenderer::PaintHelper* helper )
121{
122 /*
123 Binding GL_ARRAY_BUFFER/GL_ELEMENT_ARRAY_BUFFER to 0 seems to be enough.
124
125 However - as we do not know what is finally painted and what the
126 OpenGL paint engine is doing we better reinitialize everything.
127
128 Hope this has no side effects as the context will leave the function
129 in a modified state. Otherwise we could try to change the buffers
130 only and reset them, before leaving.
131
132 QQuickFramebufferObject does the FBO rendering early
133 ( QQuickWindow::beforeRendering ). But so far doing it below updatePaintNode
134 seems to work as well. Let's see if we run into issues ...
135 */
136
137 window->beginExternalCommands();
138
139#if QT_VERSION >= QT_VERSION_CHECK( 6, 0, 0 )
140 QQuickOpenGLUtils::resetOpenGLState();
141#else
142 window->resetOpenGLState();
143#endif
144
145 auto context = QOpenGLContext::currentContext();
146
147 QOpenGLFramebufferObjectFormat format1;
148 format1.setAttachment( QOpenGLFramebufferObject::CombinedDepthStencil );
149
150 format1.setSamples( context->format().samples() );
151
152 QOpenGLFramebufferObject multisampledFbo( size, format1 );
153
154 QOpenGLPaintDevice pd( size );
155 pd.setPaintFlipped( true );
156
157 {
158 QPainter painter( &pd );
159
160 painter.setCompositionMode( QPainter::CompositionMode_Source );
161 painter.fillRect( 0, 0, size.width(), size.height(), Qt::transparent );
162 painter.setCompositionMode( QPainter::CompositionMode_SourceOver );
163
164 const auto ratio = window->effectiveDevicePixelRatio();
165
166 painter.scale( ratio, ratio );
167 helper->paint( &painter, size / ratio );
168
169#if 1
170 if ( format1.samples() > 0 )
171 {
172 /*
173 Multisampling in the window surface might get lost
174 as a side effect of rendering to the FBO.
175 weired, needs to be investigated more
176 */
177 painter.setRenderHint( QPainter::Antialiasing, true );
178 }
179#endif
180 }
181
182 QOpenGLFramebufferObjectFormat format2;
183 format2.setAttachment( QOpenGLFramebufferObject::NoAttachment );
184
185 QOpenGLFramebufferObject fbo( size, format2 );
186
187 const QRect fboRect( 0, 0, size.width(), size.height() );
188
189 QOpenGLFramebufferObject::blitFramebuffer(
190 &fbo, fboRect, &multisampledFbo, fboRect );
191
192 window->endExternalCommands();
193
194 return qskTakeTexture( fbo );
195}
196
197static QSGTexture* qskCreateTextureRaster( QQuickWindow* window,
198 const QSize& size, QskTextureRenderer::PaintHelper* helper )
199{
200 const auto ratio = window ? window->effectiveDevicePixelRatio() : 1.0;
201
202 QImage image( size * ratio, QImage::Format_RGBA8888_Premultiplied );
203 image.fill( Qt::transparent );
204
205 {
206 QPainter painter( &image );
207
208 /*
209 setting a devicePixelRatio for the image only works for
210 value >= 1.0. So we have to scale manually.
211 */
212 painter.scale( ratio, ratio );
213
214 helper->paint( &painter, size );
215 }
216
217 return window->createTextureFromImage( image, QQuickWindow::TextureHasAlphaChannel );
218}
219
220QSGTexture* QskTextureRenderer::createPaintedTexture(
221 QQuickWindow* window, const QSize& size, PaintHelper* helper )
222{
223 if ( isOpenGLWindow( window ) )
224 {
225 const auto textureId = createPaintedTextureGL( window, size, helper );
226
227 auto texture = new QSGPlainTexture;
228 texture->setHasAlphaChannel( true );
229 texture->setOwnsTexture( true );
230
231 setTextureId( window, textureId, size, texture );
232
233 return texture;
234 }
235 else
236 {
237 return qskCreateTextureRaster( window, size, helper );
238 }
239}