QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskStrokeNode.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskStrokeNode.h"
7#include "QskVertex.h"
8#include "QskGradient.h"
9#include "QskRgbValue.h"
10#include "QskInternalMacros.h"
11
12#include <qpainterpath.h>
13
14QSK_QT_PRIVATE_BEGIN
15#include <private/qtriangulatingstroker_p.h>
16QSK_QT_PRIVATE_END
17
18static inline bool qskIsPenVisible( const QPen& pen )
19{
20 if ( pen.style() == Qt::NoPen )
21 return false;
22
23 if ( pen.brush().gradient() )
24 {
25 // TODO ...
26 }
27 else
28 {
29 if ( !QskRgb::isVisible( pen.color() ) )
30 return false;
31 }
32
33 return true;
34}
35
36QskStrokeNode::QskStrokeNode()
37{
38}
39
40QskStrokeNode::~QskStrokeNode() = default;
41
42void QskStrokeNode::updatePath( const QPainterPath& path, const QPen& pen )
43{
44 updatePath( path, QTransform(), pen );
45}
46
47void QskStrokeNode::updatePath(
48 const QPainterPath& path, const QTransform& transform, const QPen& pen )
49{
50 if ( path.isEmpty() || !qskIsPenVisible( pen ) )
51 {
52 resetGeometry();
53 return;
54 }
55
56 if ( auto qGradient = pen.brush().gradient() )
57 {
58 const auto r = transform.mapRect( path.boundingRect() );
59
60 QskGradient gradient( *qGradient );
61 gradient.setStretchMode( QskGradient::StretchToSize );
62
63 setColoring( r, gradient );
64 }
65 else
66 setColoring( pen.color() );
67
68 if ( true ) // For the moment we always update the geometry. TODO ...
69 {
70 /*
71 Unfortunately QTriangulatingStroker does not offer on the fly
72 transformations - like with qTriangulate. TODO ...
73 */
74 const auto scaledPath = transform.map( path );
75
76 auto effectivePen = pen;
77
78 if ( !effectivePen.isCosmetic() )
79 {
80 const auto scaleFactor = qMin( transform.m11(), transform.m22() );
81 if ( scaleFactor != 1.0 )
82 {
83 effectivePen.setWidth( effectivePen.widthF() * scaleFactor );
84 effectivePen.setCosmetic( false );
85 }
86 }
87
88 QTriangulatingStroker stroker;
89
90 if ( pen.style() == Qt::SolidLine )
91 {
92 // clipRect, renderHint are ignored in QTriangulatingStroker::process
93 stroker.process( qtVectorPathForPath( scaledPath ), effectivePen, {}, {} );
94 }
95 else
96 {
97 constexpr QRectF clipRect; // empty rect: no clipping
98
99 QDashedStrokeProcessor dashStroker;
100 dashStroker.process( qtVectorPathForPath( scaledPath ),
101 effectivePen, clipRect, {} );
102
103 const QVectorPath dashedVectorPath( dashStroker.points(),
104 dashStroker.elementCount(), dashStroker.elementTypes(), 0 );
105
106 stroker.process( dashedVectorPath, effectivePen, {}, {} );
107 }
108
109 auto& geometry = *this->geometry();
110
111 // 2 vertices for each point
112 geometry.setDrawingMode( QSGGeometry::DrawTriangleStrip );
113 geometry.allocate( stroker.vertexCount() / 2 );
114
115 if ( isGeometryColored() )
116 {
117 const QskVertex::Color c( pen.color() );
118
119 const auto v = stroker.vertices();
120 auto points = geometry.vertexDataAsColoredPoint2D();
121
122 for ( int i = 0; i < geometry.vertexCount(); i++ )
123 {
124 const auto j = 2 * i;
125 points[i].set( v[j], v[j + 1], c.r, c.g, c.b, c.a );
126 }
127 }
128 else
129 {
130 memcpy( geometry.vertexData(), stroker.vertices(),
131 stroker.vertexCount() * sizeof( float ) );
132 }
133
134 geometry.markVertexDataDirty();
135 markDirty( QSGNode::DirtyGeometry );
136 }
137}