QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskBasicLinesNode.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskBasicLinesNode.h"
7#include "QskInternalMacros.h"
8
9#include <qsgmaterial.h>
10#include <qsggeometry.h>
11#include <QTransform>
12
13QSK_QT_PRIVATE_BEGIN
14#include <private/qsgnode_p.h>
15QSK_QT_PRIVATE_END
16
17static inline QVector4D qskColorVector( const QColor& c, qreal opacity)
18{
19 const auto a = c.alphaF() * opacity;
20 return QVector4D( c.redF() * a, c.greenF() * a, c.blueF() * a, a );
21}
22
23static inline QVector2D qskOrigin(
24 const QRect& rect, Qt::Orientations orientations )
25{
26 return QVector2D(
27 ( orientations & Qt::Horizontal ) ? 0.5 * rect.width() : 0.0,
28 ( orientations & Qt::Vertical ) ? 0.5 * rect.height() : 0.0
29 );
30}
31
32#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
33 #include <QSGMaterialRhiShader>
34 using RhiShader = QSGMaterialRhiShader;
35#else
36 using RhiShader = QSGMaterialShader;
37#endif
38
39namespace
40{
41 class Material final : public QSGMaterial
42 {
43 public:
44 Material();
45
46#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
47 QSGMaterialShader* createShader() const override;
48#else
49 QSGMaterialShader* createShader( QSGRendererInterface::RenderMode ) const override;
50#endif
51
52 QSGMaterialType* type() const override;
53
54 int compare( const QSGMaterial* other ) const override;
55
56 QColor m_color = QColor( 255, 255, 255 );
57 Qt::Orientations m_pixelAlignment;
58 };
59
60 class ShaderRhi final : public RhiShader
61 {
62 public:
63
64 ShaderRhi()
65 {
66 const QString root( ":/qskinny/shaders/" );
67
68 setShaderFileName( VertexStage, root + "crisplines.vert.qsb" );
69 setShaderFileName( FragmentStage, root + "crisplines.frag.qsb" );
70 }
71
72 bool updateUniformData( RenderState& state,
73 QSGMaterial* newMaterial, QSGMaterial* oldMaterial ) override
74 {
75 auto matOld = static_cast< Material* >( oldMaterial );
76 auto matNew = static_cast< Material* >( newMaterial );
77
78 Q_ASSERT( state.uniformData()->size() >= 88 );
79
80 auto data = state.uniformData()->data();
81 bool changed = false;
82
83 const auto matrix = state.combinedMatrix();
84
85 if ( state.isMatrixDirty() )
86 {
87 memcpy( data + 0, matrix.constData(), 64 );
88 changed = true;
89 }
90
91 if ( ( matOld == nullptr ) || ( matNew->m_color != matOld->m_color )
92 || state.isOpacityDirty() )
93 {
94 const auto v4 = qskColorVector( matNew->m_color, state.opacity() );
95 memcpy( data + 64, &v4, 16 );
96 changed = true;
97 }
98
99 if ( state.isMatrixDirty() || ( matOld == nullptr )
100 || ( matNew->m_pixelAlignment != matOld->m_pixelAlignment ) )
101 {
102 /*
103 The shaders work with coordinates in the range[-1,1]. When knowing
104 the device coordinates corresponding to [0.0] we can scale a vertex
105 into device coordinates.
106
107 coordinates <= 0.0 indicate, that no rounding should be done.
108 */
109 const auto origin = qskOrigin(
110 state.viewportRect(), matNew->m_pixelAlignment );
111
112 memcpy( data + 80, &origin, 8 );
113 changed = true;
114 }
115
116 return changed;
117 }
118 };
119}
120
121#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
122
123namespace
124{
125 // the old type of shader - spcific for OpenGL
126
127 class ShaderGL final : public QSGMaterialShader
128 {
129 public:
130 ShaderGL()
131 {
132 const QString root( ":/qskinny/shaders/" );
133
134 setShaderSourceFile( QOpenGLShader::Vertex,
135 ":/qskinny/shaders/crisplines.vert" );
136
137 setShaderSourceFile( QOpenGLShader::Fragment,
138 ":/qt-project.org/scenegraph/shaders/flatcolor.frag" );
139 }
140
141 char const* const* attributeNames() const override
142 {
143 static char const* const names[] = { "in_vertex", nullptr };
144 return names;
145 }
146
147 void initialize() override
148 {
149 QSGMaterialShader::initialize();
150
151 auto p = program();
152
153 m_matrixId = p->uniformLocation( "matrix" );
154 m_colorId = p->uniformLocation( "color" );
155 m_originId = p->uniformLocation( "origin" );
156 }
157
158 void updateState( const QSGMaterialShader::RenderState& state,
159 QSGMaterial* newMaterial, QSGMaterial* oldMaterial) override
160 {
161 auto p = program();
162
163 const auto matrix = state.combinedMatrix();
164
165 if ( state.isMatrixDirty() )
166 p->setUniformValue( m_matrixId, matrix );
167
168 bool updateMaterial = ( oldMaterial == nullptr )
169 || newMaterial->compare( oldMaterial ) != 0;
170
171 updateMaterial |= state.isCachedMaterialDataDirty();
172
173 if ( updateMaterial )
174 {
175 auto material = static_cast< const Material* >( newMaterial );
176
177 p->setUniformValue( m_colorId,
178 qskColorVector( material->m_color, state.opacity() ) );
179
180 const auto origin = qskOrigin(
181 state.viewportRect(), material->m_pixelAlignment );;
182 p->setUniformValue( m_originId, origin );
183 }
184 }
185
186 private:
187 int m_matrixId = -1;
188 int m_colorId = -1;
189 int m_originId = -1;
190 };
191}
192
193#endif
194
195Material::Material()
196{
197#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
198 setFlag( QSGMaterial::SupportsRhiShader, true );
199#endif
200}
201
202#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
203
204QSGMaterialShader* Material::createShader() const
205{
206 if ( !( flags() & QSGMaterial::RhiShaderWanted ) )
207 return new ShaderGL();
208
209 return new ShaderRhi();
210}
211
212#else
213
214QSGMaterialShader* Material::createShader( QSGRendererInterface::RenderMode ) const
215{
216 return new ShaderRhi();
217}
218
219#endif
220
221QSGMaterialType* Material::type() const
222{
223 static QSGMaterialType staticType;
224 return &staticType;
225}
226
227int Material::compare( const QSGMaterial* other ) const
228{
229 auto material = static_cast< const Material* >( other );
230
231 if ( ( material->m_color == m_color )
232 && ( material->m_pixelAlignment == m_pixelAlignment ) )
233 {
234 return 0;
235 }
236
237 return QSGMaterial::compare( other );
238}
239
240class QskBasicLinesNodePrivate final : public QSGGeometryNodePrivate
241{
242 public:
243 QskBasicLinesNodePrivate()
244 : geometry( QSGGeometry::defaultAttributes_Point2D(), 0 )
245 {
246 geometry.setDrawingMode( QSGGeometry::DrawLines );
247 }
248
249 QSGGeometry geometry;
250 Material material;
251};
252
253QskBasicLinesNode::QskBasicLinesNode()
254 : QSGGeometryNode( *new QskBasicLinesNodePrivate )
255{
256 Q_D( QskBasicLinesNode );
257
258 setGeometry( &d->geometry );
259 setMaterial( &d->material );
260}
261
262QskBasicLinesNode::~QskBasicLinesNode()
263{
264}
265
266void QskBasicLinesNode::setPixelAlignment( Qt::Orientations orientations )
267{
268 Q_D( QskBasicLinesNode );
269
270 if ( orientations != d->material.m_pixelAlignment )
271 {
272 d->material.m_pixelAlignment = orientations;
273 markDirty( QSGNode::DirtyMaterial );
274 }
275}
276
277Qt::Orientations QskBasicLinesNode::pixelAlignment() const
278{
279 return d_func()->material.m_pixelAlignment;
280}
281
282void QskBasicLinesNode::setColor( const QColor& color )
283{
284 Q_D( QskBasicLinesNode );
285
286 const auto c = color.toRgb();
287 if ( c != d->material.m_color )
288 {
289 d->material.m_color = c;
290 markDirty( QSGNode::DirtyMaterial );
291 }
292}
293
294QColor QskBasicLinesNode::color() const
295{
296 return d_func()->material.m_color;
297}
298
299void QskBasicLinesNode::setLineWidth( float lineWidth )
300{
301 Q_D( QskBasicLinesNode );
302
303 lineWidth = std::max( lineWidth, 0.0f );
304 if( lineWidth != d->geometry.lineWidth() )
305 d->geometry.setLineWidth( lineWidth );
306}
307
308float QskBasicLinesNode::lineWidth() const
309{
310 return d_func()->geometry.lineWidth();
311}
312