6#include "QskBoxShadowNode.h"
7#include "QskBoxShapeMetrics.h"
8#include "QskInternalMacros.h"
11#include <qsgmaterialshader.h>
12#include <qsgmaterial.h>
15#include <private/qsgnode_p.h>
20#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
21 #include <QSGMaterialRhiShader>
22 using RhiShader = QSGMaterialRhiShader;
24 using RhiShader = QSGMaterialShader;
29 class Material final :
public QSGMaterial
34#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
35 QSGMaterialShader* createShader()
const override;
37 QSGMaterialShader* createShader( QSGRendererInterface::RenderMode )
const override;
40 QSGMaterialType* type()
const override;
42 int compare(
const QSGMaterial* other )
const override;
44 QVector2D m_aspectRatio = QVector2D{ 1, 1 };
45 QVector4D m_radius = QVector4D{ 0, 0, 0, 0 };
46 QVector4D m_color = QVector4D{ 0, 0, 0, 1 };
47 float m_blurExtent = 0.0;
53 class ShaderRhi final :
public RhiShader
58 const QString root(
":/qskinny/shaders/" );
60 setShaderFileName( VertexStage, root +
"boxshadow.vert.qsb" );
61 setShaderFileName( FragmentStage, root +
"boxshadow.frag.qsb" );
64 bool updateUniformData( RenderState& state,
65 QSGMaterial* newMaterial, QSGMaterial* oldMaterial )
override
67 const auto matOld =
static_cast< Material*
>( oldMaterial );
68 const auto matNew =
static_cast< Material*
>( newMaterial );
70 Q_ASSERT( state.uniformData()->size() >= 112 );
72 auto data = state.uniformData()->data();
75 if ( state.isMatrixDirty() )
77 const auto matrix = state.combinedMatrix();
78 memcpy( data + 0, matrix.constData(), 64 );
83 if ( matOld ==
nullptr || matNew->m_color != matOld->m_color )
85 memcpy( data + 64, &matNew->m_color, 16 );
89 if ( matOld ==
nullptr || matNew->m_radius != matOld->m_radius )
91 memcpy( data + 80, &matNew->m_radius, 16 );
95 if ( matOld ==
nullptr || matNew->m_aspectRatio != matOld->m_aspectRatio )
97 memcpy( data + 96, &matNew->m_aspectRatio, 8 );
101 if ( matOld ==
nullptr || matNew->m_blurExtent != matOld->m_blurExtent )
103 memcpy( data + 104, &matNew->m_blurExtent, 4 );
107 if ( state.isOpacityDirty() )
109 const float opacity = state.opacity();
110 memcpy( data + 108, &opacity, 4 );
120#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
126 class ShaderGL final :
public QSGMaterialShader
131 const QString root(
":/qskinny/shaders/" );
133 setShaderSourceFile( QOpenGLShader::Vertex, root +
"boxshadow.vert" );
134 setShaderSourceFile( QOpenGLShader::Fragment, root +
"boxshadow.frag" );
137 char const*
const* attributeNames()
const override
139 static char const*
const names[] = {
"in_vertex",
"in_coord",
nullptr };
143 void initialize()
override
145 QSGMaterialShader::initialize();
149 m_matrixId = p->uniformLocation(
"matrix" );
150 m_aspectRatioId = p->uniformLocation(
"aspectRatio" );
151 m_opacityId = p->uniformLocation(
"opacity" );
152 m_blurExtentId = p->uniformLocation(
"blurExtent" );
153 m_radiusId = p->uniformLocation(
"radius" );
154 m_colorId = p->uniformLocation(
"color" );
157 void updateState(
const QSGMaterialShader::RenderState& state,
158 QSGMaterial* newMaterial, QSGMaterial* oldMaterial)
override
162 if ( state.isMatrixDirty() )
163 p->setUniformValue( m_matrixId, state.combinedMatrix() );
165 if ( state.isOpacityDirty() )
166 p->setUniformValue( m_opacityId, state.opacity() );
168 bool updateMaterial = ( oldMaterial == nullptr )
169 || newMaterial->compare( oldMaterial ) != 0;
171 updateMaterial |= state.isCachedMaterialDataDirty();
173 if ( updateMaterial )
175 auto material =
static_cast< const Material*
>( newMaterial );
177 p->setUniformValue( m_aspectRatioId, material->m_aspectRatio );
178 p->setUniformValue( m_blurExtentId, material->m_blurExtent);
179 p->setUniformValue( m_radiusId, material->m_radius );
180 p->setUniformValue( m_colorId, material->m_color );
186 int m_opacityId = -1;
187 int m_aspectRatioId = -1;
188 int m_blurExtentId = -1;
198 setFlag( QSGMaterial::Blending,
true );
200#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
201 setFlag( QSGMaterial::SupportsRhiShader,
true );
205#if QT_VERSION < QT_VERSION_CHECK( 6, 0, 0 )
207QSGMaterialShader* Material::createShader()
const
209 if ( !( flags() & QSGMaterial::RhiShaderWanted ) )
210 return new ShaderGL();
212 return new ShaderRhi();
217QSGMaterialShader* Material::createShader( QSGRendererInterface::RenderMode )
const
219 return new ShaderRhi();
224QSGMaterialType* Material::type()
const
226 static QSGMaterialType staticType;
230int Material::compare(
const QSGMaterial* other )
const
232 auto material =
static_cast< const Material*
>( other );
234 if ( ( material->m_color == m_color )
235 && ( material->m_aspectRatio == m_aspectRatio )
236 && qFuzzyCompare(material->m_blurExtent, m_blurExtent)
237 && qFuzzyCompare(material->m_radius, m_radius) )
242 return QSGMaterial::compare( other );
245class QskBoxShadowNodePrivate final :
public QSGGeometryNodePrivate
248 QskBoxShadowNodePrivate()
249 : geometry( QSGGeometry::defaultAttributes_TexturedPoint2D(), 4 )
253 QSGGeometry geometry;
259QskBoxShadowNode::QskBoxShadowNode()
260 : QSGGeometryNode( *new QskBoxShadowNodePrivate )
264 setGeometry( &d->geometry );
265 setMaterial( &d->material );
268QskBoxShadowNode::~QskBoxShadowNode()
272void QskBoxShadowNode::setShadowData(
274 qreal blurRadius,
const QColor& color )
278 if ( rect != d->rect )
282 QSGGeometry::updateTexturedRectGeometry(
283 &d->geometry, d->rect, QRectF( -0.5, -0.5, 1.0, 1.0 ) );
285 d->geometry.markVertexDataDirty();
286 markDirty( QSGNode::DirtyGeometry );
288 QVector2D aspectRatio( 1.0, 1.0 );
290 if ( rect.width() >= rect.height() )
291 aspectRatio.setX( rect.width() / rect.height() );
293 aspectRatio.setY( rect.height() / rect.width() );
295 if ( d->material.m_aspectRatio != aspectRatio )
297 d->material.m_aspectRatio = aspectRatio;
298 markDirty( QSGNode::DirtyMaterial );
303 const auto shape = shapeMetrics.toAbsolute( rect.size() );
305 const float t = std::min( d->rect.width(), d->rect.height() );
307 const float r1 = shape.radius( Qt::BottomRightCorner ).width();
308 const float r2 = shape.radius( Qt::TopRightCorner ).width();
309 const float r3 = shape.radius( Qt::BottomLeftCorner ).width();
310 const float r4 = shape.radius( Qt::TopLeftCorner ).width();
312 const auto uniformRadius = QVector4D(
313 std::min( r1 / t, 1.0f ), std::min( r2 / t, 1.0f ),
314 std::min( r3 / t, 1.0f ), std::min( r4 / t, 1.0f ) );
316 if ( d->material.m_radius != uniformRadius )
318 d->material.m_radius = uniformRadius;
319 markDirty( QSGNode::DirtyMaterial );
324 if ( blurRadius <= 0.0 )
327 const float t = 0.5 * std::min( d->rect.width(), d->rect.height() );
328 const float uniformExtent = blurRadius / t;
330 if ( !qFuzzyCompare( d->material.m_blurExtent, uniformExtent ) )
332 d->material.m_blurExtent = uniformExtent;
333 markDirty( QSGNode::DirtyMaterial );
338 const auto a = color.alphaF();
340 const QVector4D c( color.redF() * a, color.greenF() * a, color.blueF() * a, a );
342 if ( d->material.m_color != c )
344 d->material.m_color = c;
345 markDirty( QSGNode::DirtyMaterial );