Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qquickshapecurverenderer.cpp
Go to the documentation of this file.
1// Copyright (C) 2024 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3// Qt-Security score:significant reason:default
4
7
8#if QT_CONFIG(thread)
9#include <QtCore/qthreadpool.h>
10#endif
11
12#include <QtGui/qvector2d.h>
13#include <QtGui/qvector4d.h>
14#include <QtGui/private/qtriangulator_p.h>
15#include <QtGui/private/qtriangulatingstroker_p.h>
16#include <QtGui/private/qrhi_p.h>
17
18#include <QtQuick/private/qsgcurvefillnode_p.h>
19#include <QtQuick/private/qsgcurvestrokenode_p.h>
20#include <QtQuick/private/qquadpath_p.h>
21#include <QtQuick/private/qsgcurveprocessor_p.h>
22#include <QtQuick/qsgmaterial.h>
23
25
26Q_LOGGING_CATEGORY(lcShapeCurveRenderer, "qt.shape.curverenderer");
27
28namespace {
29
30/*! \internal
31 Choice of vertex shader to use for the wireframe node:
32 * \c SimpleWFT is for when vertices are already in logical coordinates
33 * \c StrokeWFT chooses the stroke shader, which moves vertices according to the stroke width uniform
34*/
35enum WireFrameType { SimpleWFT, StrokeWFT };
36
37class QQuickShapeWireFrameMaterialShader : public QSGMaterialShader
38{
39public:
40 QQuickShapeWireFrameMaterialShader(WireFrameType wft, int viewCount) : m_wftype(wft)
41 {
42 setShaderFileName(VertexStage, wft == StrokeWFT ?
43 QStringLiteral(":/qt-project.org/scenegraph/shaders_ng/shapestroke_wireframe.vert.qsb") :
44 QStringLiteral(":/qt-project.org/shapes/shaders_ng/wireframe.vert.qsb"), viewCount);
45 setShaderFileName(FragmentStage,
46 wft == StrokeWFT ?
47 QStringLiteral(":/qt-project.org/scenegraph/shaders_ng/shapestroke_wireframe.frag.qsb") :
48 QStringLiteral(":/qt-project.org/shapes/shaders_ng/wireframe.frag.qsb"), viewCount);
49 }
50
51 bool updateUniformData(RenderState &state, QSGMaterial *newMaterial, QSGMaterial *) override;
52
53 WireFrameType m_wftype;
54};
55
56class QQuickShapeWireFrameMaterial : public QSGMaterial
57{
58public:
59 QQuickShapeWireFrameMaterial(WireFrameType wft) : m_wftype(wft)
60 {
61 setFlag(Blending, true);
62 }
63
64 int compare(const QSGMaterial *other) const override
65 {
66 return (type() - other->type());
67 }
68
69 void setCosmeticStroke(bool c)
70 {
71 m_cosmeticStroke = c;
72 }
73
74 void setStrokeWidth(float width)
75 {
76 m_strokeWidth = width;
77 }
78
79 float strokeWidth()
80 {
81 return (m_cosmeticStroke ? -1.0 : 1.0) * qAbs(m_strokeWidth);
82 }
83
84protected:
85 QSGMaterialType *type() const override
86 {
87 static QSGMaterialType t;
88 return &t;
89 }
90 QSGMaterialShader *createShader(QSGRendererInterface::RenderMode) const override
91 {
92 return new QQuickShapeWireFrameMaterialShader(m_wftype, viewCount());
93 }
94
95 WireFrameType m_wftype;
96 bool m_cosmeticStroke = false;
97 float m_strokeWidth = 1.0f;
98};
99
100bool QQuickShapeWireFrameMaterialShader::updateUniformData(RenderState &state, QSGMaterial *newMaterial, QSGMaterial *)
101{
102 QByteArray *buf = state.uniformData();
103 Q_ASSERT(buf->size() >= 64);
104 const int matrixCount = qMin(state.projectionMatrixCount(), newMaterial->viewCount());
105 bool changed = false;
106 float localScale = /* newNode != nullptr ? newNode->localScale() : */ 1.0f;
107
108 for (int viewIndex = 0; viewIndex < matrixCount; ++viewIndex) {
109 if (state.isMatrixDirty()) {
110 QMatrix4x4 m = state.combinedMatrix(viewIndex);
111 if (m_wftype == StrokeWFT)
112 m.scale(localScale);
113 memcpy(buf->data() + 64 * viewIndex, m.constData(), 64);
114 changed = true;
115 }
116 }
117 // determinant is xscale * yscale, as long as Item.transform does not include shearing or rotation
118 const float matrixScale = qSqrt(qAbs(state.determinant())) * state.devicePixelRatio() * localScale;
119 memcpy(buf->data() + matrixCount * 64, &matrixScale, 4);
120 const float dpr = state.devicePixelRatio();
121 memcpy(buf->data() + matrixCount * 64 + 8, &dpr, 4);
122 const float opacity = 1.0; // don't fade the wireframe
123 memcpy(buf->data() + matrixCount * 64 + 4, &opacity, 4);
124 const float strokeWidth = static_cast<QQuickShapeWireFrameMaterial *>(newMaterial)->strokeWidth();
125 memcpy(buf->data() + matrixCount * 64 + 12, &strokeWidth, 4);
126 changed = true;
127 // shapestroke_wireframe.vert doesn't use the strokeColor and debug uniforms, so we don't bother setting them
128
129 return changed;
130}
131
132template <WireFrameType wftype>
133class QQuickShapeWireFrameNode : public QSGCurveAbstractNode
134{
135public:
136 struct WireFrameVertex
137 {
138 float x, y, u, v, w, nx, ny, sw;
139 };
140
141 QQuickShapeWireFrameNode()
142 {
143 isDebugNode = true;
144 setFlag(OwnsGeometry, true);
145 setGeometry(new QSGGeometry(attributes(), 0, 0));
146 activateMaterial();
147 }
148
149 void setColor(QColor col) override
150 {
151 Q_UNUSED(col);
152 }
153
154 void setUseStandardDerivatives(bool useStandardDerivatives) override
155 {
156 Q_UNUSED(useStandardDerivatives);
157 }
158
159 void setCosmeticStroke(bool c)
160 {
161 m_material->setCosmeticStroke(c);
162 }
163
164 void setStrokeWidth(float width)
165 {
166 m_material->setStrokeWidth(width);
167 }
168
169 void activateMaterial()
170 {
171 m_material.reset(new QQuickShapeWireFrameMaterial(wftype));
172 setMaterial(m_material.data());
173 }
174
175 static const QSGGeometry::AttributeSet &attributes()
176 {
177 static QSGGeometry::Attribute data[] = {
178 QSGGeometry::Attribute::createWithAttributeType(0, 2, QSGGeometry::FloatType, QSGGeometry::PositionAttribute),
179 QSGGeometry::Attribute::createWithAttributeType(1, 3, QSGGeometry::FloatType, QSGGeometry::TexCoordAttribute),
180 QSGGeometry::Attribute::createWithAttributeType(2, 3, QSGGeometry::FloatType, QSGGeometry::TexCoordAttribute),
181 };
182 static QSGGeometry::AttributeSet attrs = { 3, sizeof(WireFrameVertex), data };
183 return attrs;
184 }
185
186 void cookGeometry() override
187 {
188 // Intentionally empty
189 }
190
191protected:
192 QScopedPointer<QQuickShapeWireFrameMaterial> m_material;
193};
194}
195
196QQuickShapeCurveRenderer::~QQuickShapeCurveRenderer()
197{
198 for (const PathData &pd : std::as_const(m_paths)) {
199 if (pd.currentRunner) {
200 pd.currentRunner->orphaned = true;
201 if (!pd.currentRunner->isAsync || pd.currentRunner->isDone)
202 delete pd.currentRunner;
203 }
204 }
205}
206
207void QQuickShapeCurveRenderer::beginSync(int totalCount, bool *countChanged)
208{
209 if (countChanged != nullptr && totalCount != m_paths.size())
210 *countChanged = true;
211 for (int i = totalCount; i < m_paths.size(); i++) { // Handle removal of paths
212 setFillTextureProvider(i, nullptr); // deref window
213 m_removedPaths.append(m_paths.at(i));
214 }
215 m_paths.resize(totalCount);
216}
217
218void QQuickShapeCurveRenderer::setPath(int index, const QPainterPath &path, QQuickShapePath::PathHints pathHints)
219{
220 auto &pathData = m_paths[index];
221 pathData.originalPath = path;
222 pathData.pathHints = pathHints;
223 pathData.m_dirty |= PathDirty;
224}
225
226void QQuickShapeCurveRenderer::setStrokeColor(int index, const QColor &color)
227{
228 auto &pathData = m_paths[index];
229 const bool wasVisible = pathData.isStrokeVisible();
230 pathData.pen.setColor(color);
231 if (pathData.isStrokeVisible() != wasVisible)
232 pathData.m_dirty |= StrokeDirty;
233 else
234 pathData.m_dirty |= UniformsDirty;
235}
236
237void QQuickShapeCurveRenderer::setStrokeWidth(int index, qreal w)
238{
239 auto &pathData = m_paths[index];
240 if (w > 0) {
241 pathData.validPenWidth = true;
242 pathData.pen.setWidthF(w);
243 } else {
244 pathData.validPenWidth = false;
245 }
246 pathData.m_dirty |= StrokeDirty;
247}
248
249void QQuickShapeCurveRenderer::setCosmeticStroke(int index, bool c)
250{
251 auto &pathData = m_paths[index];
252 pathData.pen.setCosmetic(c);
253 pathData.m_dirty |= StrokeDirty;
254}
255
256void QQuickShapeCurveRenderer::setFillColor(int index, const QColor &color)
257{
258 auto &pathData = m_paths[index];
259 const bool wasVisible = pathData.isFillVisible();
260 pathData.fillColor = color;
261 if (pathData.isFillVisible() != wasVisible)
262 pathData.m_dirty |= FillDirty;
263 else
264 pathData.m_dirty |= UniformsDirty;
265}
266
267void QQuickShapeCurveRenderer::setFillRule(int index, QQuickShapePath::FillRule fillRule)
268{
269 auto &pathData = m_paths[index];
270 pathData.fillRule = Qt::FillRule(fillRule);
271 pathData.m_dirty |= PathDirty;
272}
273
274void QQuickShapeCurveRenderer::setJoinStyle(int index,
275 QQuickShapePath::JoinStyle joinStyle,
276 int miterLimit)
277{
278 auto &pathData = m_paths[index];
279 pathData.pen.setJoinStyle(Qt::PenJoinStyle(joinStyle));
280 pathData.pen.setMiterLimit(miterLimit);
281 pathData.m_dirty |= StrokeDirty;
282}
283
284void QQuickShapeCurveRenderer::setCapStyle(int index, QQuickShapePath::CapStyle capStyle)
285{
286 auto &pathData = m_paths[index];
287 pathData.pen.setCapStyle(Qt::PenCapStyle(capStyle));
288 pathData.m_dirty |= StrokeDirty;
289}
290
291void QQuickShapeCurveRenderer::setStrokeStyle(int index,
292 QQuickShapePath::StrokeStyle strokeStyle,
293 qreal dashOffset,
294 const QList<qreal> &dashPattern)
295{
296 auto &pathData = m_paths[index];
297 pathData.pen.setStyle(Qt::PenStyle(strokeStyle));
298 if (strokeStyle == QQuickShapePath::DashLine) {
299 pathData.pen.setDashPattern(dashPattern);
300 pathData.pen.setDashOffset(dashOffset);
301 }
302 pathData.m_dirty |= StrokeDirty;
303}
304
305static QGradient::Type copyCurveGradient(const QQuickShapeGradient *gradient,
306 QSGGradientCache::GradientDesc *dst)
307{
308 QGradient::Type gradientType = QGradient::NoGradient;
309 if (const QQuickShapeLinearGradient *g = qobject_cast<const QQuickShapeLinearGradient *>(gradient)) {
310 gradientType = QGradient::LinearGradient;
311 dst->a = QPointF(g->x1(), g->y1());
312 dst->b = QPointF(g->x2(), g->y2());
313 } else if (const QQuickShapeRadialGradient *g = qobject_cast<const QQuickShapeRadialGradient *>(gradient)) {
314 gradientType = QGradient::RadialGradient;
315 dst->a = QPointF(g->centerX(), g->centerY());
316 dst->b = QPointF(g->focalX(), g->focalY());
317 dst->v0 = g->centerRadius();
318 dst->v1 = g->focalRadius();
319 } else if (const QQuickShapeConicalGradient *g = qobject_cast<const QQuickShapeConicalGradient *>(gradient)) {
320 gradientType = QGradient::ConicalGradient;
321 dst->a = QPointF(g->centerX(), g->centerY());
322 dst->v0 = g->angle();
323 } else if (gradient != nullptr) {
324 static bool warned = false;
325 if (!warned) {
326 warned = true;
327 qCWarning(lcShapeCurveRenderer) << "Unsupported gradient";
328 }
329 }
330
331 if (gradientType != QGradient::NoGradient) {
332 dst->stops = gradient->gradientStops();
333 dst->spread = QGradient::Spread(gradient->spread());
334 }
335
336 return gradientType;
337}
338
339void QQuickShapeCurveRenderer::setFillGradient(int index, QQuickShapeGradient *gradient)
340{
341 PathData &pd(m_paths[index]);
342 const bool wasVisible = pd.isFillVisible();
343 pd.gradientType = copyCurveGradient(gradient, &pd.gradient);
344 pd.m_dirty |= (pd.isFillVisible() != wasVisible) ? FillDirty : UniformsDirty;
345}
346
347void QQuickShapeCurveRenderer::setStrokeGradient(int index, QQuickShapeGradient *gradient)
348{
349 PathData &pd(m_paths[index]);
350 const bool wasVisible = pd.isStrokeVisible();
351 pd.strokeGradientType = copyCurveGradient(gradient, &pd.strokeGradient);
352 pd.m_dirty |= (pd.isStrokeVisible() != wasVisible) ? StrokeDirty : UniformsDirty;
353}
354
355void QQuickShapeCurveRenderer::setFillTransform(int index, const QSGTransform &transform)
356{
357 auto &pathData = m_paths[index];
358 pathData.fillTransform = transform;
359 pathData.m_dirty |= UniformsDirty;
360}
361
362void QQuickShapeCurveRenderer::setFillTextureProvider(int index, QQuickItem *textureProviderItem)
363{
364 auto &pathData = m_paths[index];
365 const bool wasVisible = pathData.isFillVisible();
366 if (pathData.fillTextureProviderItem != nullptr)
367 QQuickItemPrivate::get(pathData.fillTextureProviderItem)->derefWindow();
368 pathData.fillTextureProviderItem = textureProviderItem;
369 if (pathData.fillTextureProviderItem != nullptr)
370 QQuickItemPrivate::get(pathData.fillTextureProviderItem)->refWindow(m_item->window());
371 pathData.m_dirty |= (pathData.isFillVisible() != wasVisible) ? FillDirty : UniformsDirty;
372}
373
374void QQuickShapeCurveRenderer::handleSceneChange(QQuickWindow *window)
375{
376 for (auto &pathData : m_paths) {
377 if (pathData.fillTextureProviderItem != nullptr) {
378 if (window == nullptr)
379 QQuickItemPrivate::get(pathData.fillTextureProviderItem)->derefWindow();
380 else
381 QQuickItemPrivate::get(pathData.fillTextureProviderItem)->refWindow(window);
382 }
383 }
384}
385
386void QQuickShapeCurveRenderer::setAsyncCallback(void (*callback)(void *), void *data)
387{
388 m_asyncCallback = callback;
389 m_asyncCallbackData = data;
390}
391
392void QQuickShapeCurveRenderer::endSync(bool async)
393{
394 bool asyncThreadsRunning = false;
395
396 for (PathData &pathData : m_paths) {
397 if (!pathData.m_dirty)
398 continue;
399
400 if (pathData.m_dirty == UniformsDirty) {
401 // Requires no curve node computation, gets handled directly in updateNode()
402 continue;
403 }
404
405 if (pathData.currentRunner) {
406 // We are in a new sync round before updateNode() has been called to commit the results
407 // of the previous sync and processing round
408 if (pathData.currentRunner->isAsync) {
409 // Already performing async processing. A new run of the runner will be started in
410 // updateNode() to take care of the new dirty flags
411 asyncThreadsRunning = true;
412 continue;
413 } else {
414 // Throw away outdated results and start a new processing
415 delete pathData.currentRunner;
416 pathData.currentRunner = nullptr;
417 }
418 }
419
420 pathData.currentRunner = new QQuickShapeCurveRunnable;
421 setUpRunner(&pathData);
422
423#if QT_CONFIG(thread)
424 if (async) {
425 pathData.currentRunner->isAsync = true;
426 QThreadPool::globalInstance()->start(pathData.currentRunner);
427 asyncThreadsRunning = true;
428 } else
429#endif
430 {
431 pathData.currentRunner->run();
432 }
433 }
434
435 if (async && !asyncThreadsRunning && m_asyncCallback)
436 m_asyncCallback(m_asyncCallbackData);
437}
438
439void QQuickShapeCurveRenderer::setUpRunner(PathData *pathData)
440{
441 Q_ASSERT(pathData->currentRunner);
442 QQuickShapeCurveRunnable *runner = pathData->currentRunner;
443 runner->isDone = false;
444 runner->pathData = *pathData;
445 runner->pathData.fillNodes.clear();
446 runner->pathData.strokeNodes.clear();
447 runner->pathData.currentRunner = nullptr;
448 pathData->m_dirty = 0;
449 if (!runner->isInitialized) {
450 runner->isInitialized = true;
451 runner->setAutoDelete(false);
452 QObject::connect(runner, &QQuickShapeCurveRunnable::done, qApp,
453 [this](QQuickShapeCurveRunnable *r) {
454 r->isDone = true;
455 if (r->orphaned) {
456 delete r; // Renderer was destroyed
457 } else if (r->isAsync) {
458 maybeUpdateAsyncItem();
459 }
460 });
461 }
462}
463
464void QQuickShapeCurveRenderer::maybeUpdateAsyncItem()
465{
466 for (const PathData &pd : std::as_const(m_paths)) {
467 if (pd.currentRunner && !pd.currentRunner->isDone)
468 return;
469 }
470 if (m_item)
471 m_item->update();
472 if (m_asyncCallback)
473 m_asyncCallback(m_asyncCallbackData);
474}
475
476QQuickShapeCurveRunnable::~QQuickShapeCurveRunnable()
477{
478 qDeleteAll(pathData.fillNodes);
479 qDeleteAll(pathData.strokeNodes);
480}
481
483{
484 QQuickShapeCurveRenderer::processPath(&pathData);
485 emit done(this);
486}
487
489{
490 static bool d = qEnvironmentVariableIntValue("QT_QUICKSHAPES_DISABLE_STANDARD_DERIVATIVES") != 0;
491 return d;
492}
493
494void QQuickShapeCurveRenderer::updateNode()
495{
496 if (!m_rootNode)
497 return;
498
499 auto updateUniforms = [](const PathData &pathData) {
500 for (auto &pathNode : std::as_const(pathData.fillNodes)) {
501 if (pathNode->isDebugNode)
502 continue;
503 QSGCurveFillNode *fillNode = static_cast<QSGCurveFillNode *>(pathNode);
504 fillNode->setColor(pathData.fillColor);
505 fillNode->setGradientType(pathData.gradientType);
506 fillNode->setFillGradient(pathData.gradient);
507 fillNode->setFillTransform(pathData.fillTransform);
508 fillNode->setFillTextureProvider(pathData.fillTextureProviderItem != nullptr
509 ? pathData.fillTextureProviderItem->textureProvider()
510 : nullptr);
511 }
512 for (QSGCurveAbstractNode *pathNode : std::as_const(pathData.strokeNodes)) {
513 pathNode->setColor(pathData.pen.color());
514 if (pathNode->isDebugNode) {
515 auto *wfNode = static_cast<QQuickShapeWireFrameNode<StrokeWFT> *>(pathNode);
516 wfNode->setStrokeWidth(pathData.pen.widthF());
517 wfNode->setCosmeticStroke(pathData.pen.isCosmetic());
518 } else {
519 auto *strokeNode = static_cast<QSGCurveStrokeNode *>(pathNode);
520 strokeNode->setStrokeWidth(pathData.pen.widthF());
521 strokeNode->setCosmeticStroke(pathData.pen.isCosmetic());
522 strokeNode->setStrokeGradient(pathData.strokeGradient);
523 strokeNode->setGradientType(pathData.strokeGradientType);
524 }
525 }
526 };
527
528 NodeList toBeDeleted;
529
530 for (const PathData &pathData : std::as_const(m_removedPaths)) {
531 toBeDeleted += pathData.fillNodes;
532 toBeDeleted += pathData.strokeNodes;
533 }
534 m_removedPaths.clear();
535
536 const bool supportsDerivatives = m_item != nullptr
537 && m_item->window() != nullptr
538 && m_item->window()->rhi() != nullptr
539 && !disableScreenSpaceDerivativeShader()
540 ? m_item->window()->rhi()->isFeatureSupported(QRhi::ScreenSpaceDerivatives)
541 : false;
542
543 for (int i = 0; i < m_paths.size(); i++) {
544 PathData &pathData = m_paths[i];
545 if (pathData.currentRunner) {
546 if (!pathData.currentRunner->isDone)
547 continue;
548 // Find insertion point for new nodes. Default is the first stroke node of this path
549 QSGNode *nextNode = pathData.strokeNodes.value(0);
550 // If that is 0, use the first node (stroke or fill) of later paths, if any
551 for (int j = i + 1; !nextNode && j < m_paths.size(); j++) {
552 const PathData &pd = m_paths[j];
553 nextNode = pd.fillNodes.isEmpty() ? pd.strokeNodes.value(0) : pd.fillNodes.value(0);
554 }
555
556 PathData &newData = pathData.currentRunner->pathData;
557 if (newData.m_dirty & PathDirty)
558 pathData.path = newData.path;
559 if (newData.m_dirty & FillDirty) {
560 pathData.fillPath = newData.fillPath;
561 for (auto *node : std::as_const(newData.fillNodes)) {
562 node->setUseStandardDerivatives(supportsDerivatives);
563 if (nextNode)
564 m_rootNode->insertChildNodeBefore(node, nextNode);
565 else
566 m_rootNode->appendChildNode(node);
567 }
568 toBeDeleted += pathData.fillNodes;
569 pathData.fillNodes = newData.fillNodes;
570 }
571 if (newData.m_dirty & StrokeDirty) {
572 for (auto *node : std::as_const(newData.strokeNodes)) {
573 node->setUseStandardDerivatives(supportsDerivatives);
574 if (nextNode)
575 m_rootNode->insertChildNodeBefore(node, nextNode);
576 else
577 m_rootNode->appendChildNode(node);
578 }
579 toBeDeleted += pathData.strokeNodes;
580 pathData.strokeNodes = newData.strokeNodes;
581 }
582 if (newData.m_dirty & UniformsDirty)
583 updateUniforms(newData);
584
585 // Ownership of new nodes have been transferred to root node
586 newData.fillNodes.clear();
587 newData.strokeNodes.clear();
588
589#if QT_CONFIG(thread)
590 if (pathData.currentRunner->isAsync && (pathData.m_dirty & ~UniformsDirty)) {
591 // New changes have arrived while runner was computing; restart it to handle them
592 setUpRunner(&pathData);
593 QThreadPool::globalInstance()->start(pathData.currentRunner);
594 } else
595#endif
596 {
597 pathData.currentRunner->deleteLater();
598 pathData.currentRunner = nullptr;
599 }
600 }
601
602 if (pathData.m_dirty == UniformsDirty && !pathData.currentRunner) {
603 // Simple case so no runner was created in endSync(); handle it directly here
604 updateUniforms(pathData);
605 pathData.m_dirty = 0;
606 }
607 }
608 qDeleteAll(toBeDeleted); // also removes them from m_rootNode's child list
609}
610
611void QQuickShapeCurveRenderer::processPath(PathData *pathData)
612{
613 static const bool doOverlapSolving = !qEnvironmentVariableIntValue("QT_QUICKSHAPES_DISABLE_OVERLAP_SOLVER");
614 static const bool doIntersetionSolving = !qEnvironmentVariableIntValue("QT_QUICKSHAPES_DISABLE_INTERSECTION_SOLVER");
615 static const bool useTriangulatingStroker = qEnvironmentVariableIntValue("QT_QUICKSHAPES_TRIANGULATING_STROKER");
616 static const bool simplifyPath = qEnvironmentVariableIntValue("QT_QUICKSHAPES_SIMPLIFY_PATHS");
617
618 int &dirtyFlags = pathData->m_dirty;
619
620 if (dirtyFlags & PathDirty) {
621 if (simplifyPath)
622 pathData->path = QQuadPath::fromPainterPath(pathData->originalPath.simplified(), QQuadPath::PathLinear | QQuadPath::PathNonIntersecting | QQuadPath::PathNonOverlappingControlPointTriangles);
623 else
624 pathData->path = QQuadPath::fromPainterPath(pathData->originalPath, QQuadPath::PathHints(int(pathData->pathHints)));
625 pathData->path.setFillRule(pathData->fillRule);
626 pathData->fillPath = {};
627 dirtyFlags |= (FillDirty | StrokeDirty);
628 }
629
630 if (dirtyFlags & FillDirty) {
631 if (pathData->isFillVisible()) {
632 if (pathData->fillPath.isEmpty()) {
633 pathData->fillPath = pathData->path.subPathsClosed();
634 if (doIntersetionSolving)
635 QSGCurveProcessor::solveIntersections(pathData->fillPath);
636 pathData->fillPath.addCurvatureData();
637 if (doOverlapSolving)
638 QSGCurveProcessor::solveOverlaps(pathData->fillPath);
639 }
640 pathData->fillNodes = addFillNodes(pathData->fillPath);
641 dirtyFlags |= (StrokeDirty | UniformsDirty);
642 }
643 }
644
645 if (dirtyFlags & StrokeDirty) {
646 if (pathData->isStrokeVisible()) {
647 const QPen &pen = pathData->pen;
648 const bool solid = (pen.style() == Qt::SolidLine);
649 const QQuadPath &strokePath = solid ? pathData->path
650 : pathData->path.dashed(pen.widthF(),
651 pen.dashPattern(),
652 pen.dashOffset());
653 if (useTriangulatingStroker)
654 pathData->strokeNodes = addTriangulatingStrokerNodes(strokePath, pen);
655 else
656 pathData->strokeNodes = addCurveStrokeNodes(strokePath, pen);
657 dirtyFlags |= UniformsDirty;
658 }
659 }
660}
661
662QQuickShapeCurveRenderer::NodeList QQuickShapeCurveRenderer::addFillNodes(const QQuadPath &path)
663{
664 NodeList ret;
665 std::unique_ptr<QSGCurveFillNode> node(new QSGCurveFillNode);
666 std::unique_ptr<QQuickShapeWireFrameNode<SimpleWFT>> wfNode;
667
668 const qsizetype approxDataCount = 20 * path.elementCount();
669 node->reserve(approxDataCount);
670
671 const int debugFlags = debugVisualization();
672 const bool wireFrame = debugFlags & DebugWireframe;
673
674 if (Q_LIKELY(!wireFrame)) {
675 QSGCurveProcessor::processFill(path,
676 path.fillRule(),
677 [&node](const std::array<QVector2D, 3> &v,
678 const std::array<QVector2D, 3> &n,
679 QSGCurveProcessor::uvForPointCallback uvForPoint)
680 {
681 node->appendTriangle(v, n, uvForPoint);
682 });
683 } else {
684 QList<QQuickShapeWireFrameNode<SimpleWFT>::WireFrameVertex> wfVertices;
685 wfVertices.reserve(approxDataCount);
686 QSGCurveProcessor::processFill(path,
687 path.fillRule(),
688 [&wfVertices, &node](const std::array<QVector2D, 3> &v,
689 const std::array<QVector2D, 3> &n,
690 QSGCurveProcessor::uvForPointCallback uvForPoint)
691 {
692 node->appendTriangle(v, n, uvForPoint);
693
694 wfVertices.append({v.at(0).x(), v.at(0).y(), 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f }); // 0
695 wfVertices.append({v.at(1).x(), v.at(1).y(), 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 1.0f }); // 1
696 wfVertices.append({v.at(2).x(), v.at(2).y(), 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f }); // 2
697 });
698
699 wfNode.reset(new QQuickShapeWireFrameNode<SimpleWFT>);
700 const QList<quint32> indices = node->uncookedIndexes();
701 QSGGeometry *wfg = new QSGGeometry(QQuickShapeWireFrameNode<SimpleWFT>::attributes(),
702 wfVertices.size(),
703 indices.size(),
704 QSGGeometry::UnsignedIntType);
705 wfNode->setGeometry(wfg);
706
707 wfg->setDrawingMode(QSGGeometry::DrawTriangles);
708 memcpy(wfg->indexData(),
709 indices.data(),
710 indices.size() * wfg->sizeOfIndex());
711 memcpy(wfg->vertexData(),
712 wfVertices.data(),
713 wfg->vertexCount() * wfg->sizeOfVertex());
714 }
715
716 if (Q_UNLIKELY(debugFlags & DebugCurves))
717 node->setDebug(0.5f);
718
719 if (node->uncookedIndexes().size() > 0) {
720 node->cookGeometry();
721 ret.append(node.release());
722 if (wireFrame)
723 ret.append(wfNode.release());
724 }
725
726 return ret;
727}
728
729QQuickShapeCurveRenderer::NodeList QQuickShapeCurveRenderer::addTriangulatingStrokerNodes(const QQuadPath &path, const QPen &pen)
730{
731 NodeList ret;
732 const QColor &color = pen.color();
733
734 QList<QQuickShapeWireFrameNode<StrokeWFT>::WireFrameVertex> wfVertices;
735
736 QTriangulatingStroker stroker;
737 const auto painterPath = path.toPainterPath();
738 const QVectorPath &vp = qtVectorPathForPath(painterPath);
739 stroker.process(vp, pen, {}, {});
740
741 auto *node = new QSGCurveFillNode;
742
743 auto uvForPoint = [](QVector2D v1, QVector2D v2, QVector2D p)
744 {
745 double divisor = v1.x() * v2.y() - v2.x() * v1.y();
746
747 float u = (p.x() * v2.y() - p.y() * v2.x()) / divisor;
748 float v = (p.y() * v1.x() - p.x() * v1.y()) / divisor;
749
750 return QVector2D(u, v);
751 };
752
753 // Find uv coordinates for the point p, for a quadratic curve from p0 to p2 with control point p1
754 // also works for a line from p0 to p2, where p1 is on the inside of the path relative to the line
755 auto curveUv = [uvForPoint](QVector2D p0, QVector2D p1, QVector2D p2, QVector2D p)
756 {
757 QVector2D v1 = 2 * (p1 - p0);
758 QVector2D v2 = p2 - v1 - p0;
759 return uvForPoint(v1, v2, p - p0);
760 };
761
762 auto findPointOtherSide = [](const QVector2D &startPoint, const QVector2D &endPoint, const QVector2D &referencePoint){
763
764 QVector2D baseLine = endPoint - startPoint;
765 QVector2D insideVector = referencePoint - startPoint;
766 QVector2D normal = QVector2D(-baseLine.y(), baseLine.x()); // TODO: limit size of triangle
767
768 bool swap = QVector2D::dotProduct(insideVector, normal) < 0;
769
770 return swap ? startPoint + normal : startPoint - normal;
771 };
772
773 static bool disableExtraTriangles = qEnvironmentVariableIntValue("QT_QUICKSHAPES_WIP_DISABLE_EXTRA_STROKE_TRIANGLES");
774
775 auto addStrokeTriangle = [&](const QVector2D &p1, const QVector2D &p2, const QVector2D &p3){
776 if (p1 == p2 || p2 == p3) {
777 return;
778 }
779
780 auto uvForPoint = [&p1, &p2, &p3, curveUv](QVector2D p) {
781 auto uv = curveUv(p1, p2, p3, p);
782 return QVector3D(uv.x(), uv.y(), 0.0f); // Line
783 };
784
785 node->appendTriangle(p1, p2, p3, uvForPoint);
786
787
788 wfVertices.append({p1.x(), p1.y(), 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f}); // 0
789 wfVertices.append({p2.x(), p2.y(), 0.0f, 0.1f, 0.0f, 0.0f, 0.0f, 1.0f}); // 1
790 wfVertices.append({p3.x(), p3.y(), 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f}); // 2
791
792 if (!disableExtraTriangles) {
793 // Add a triangle on the outer side of the line to get some more AA
794 // The new point replaces p2 (currentVertex+1)
795 QVector2D op = findPointOtherSide(p1, p3, p2);
796 node->appendTriangle(p1, op, p3, uvForPoint);
797
798 wfVertices.append({p1.x(), p1.y(), 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f});
799 wfVertices.append({op.x(), op.y(), 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 1.0f}); // replacing p2
800 wfVertices.append({p3.x(), p3.y(), 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 1.0f});
801 }
802 };
803
804 const int vertCount = stroker.vertexCount() / 2;
805 const float *verts = stroker.vertices();
806 for (int i = 0; i < vertCount - 2; ++i) {
807 QVector2D p[3];
808 for (int j = 0; j < 3; ++j) {
809 p[j] = QVector2D(verts[(i+j)*2], verts[(i+j)*2 + 1]);
810 }
811 addStrokeTriangle(p[0], p[1], p[2]);
812 }
813
814 QList<quint32> indices = node->uncookedIndexes();
815 if (indices.size() > 0) {
816 node->setColor(color);
817
818 node->cookGeometry();
819 ret.append(node);
820 }
821 const bool wireFrame = debugVisualization() & DebugWireframe;
822 if (wireFrame) {
823 QQuickShapeWireFrameNode<StrokeWFT> *wfNode = new QQuickShapeWireFrameNode<StrokeWFT>;
824 QSGGeometry *wfg = new QSGGeometry(QQuickShapeWireFrameNode<StrokeWFT>::attributes(),
825 wfVertices.size(),
826 indices.size(),
827 QSGGeometry::UnsignedIntType);
828 wfNode->setGeometry(wfg);
829
830 wfg->setDrawingMode(QSGGeometry::DrawTriangles);
831 memcpy(wfg->indexData(),
832 indices.data(),
833 indices.size() * wfg->sizeOfIndex());
834 memcpy(wfg->vertexData(),
835 wfVertices.data(),
836 wfg->vertexCount() * wfg->sizeOfVertex());
837
838 ret.append(wfNode);
839 }
840
841 return ret;
842}
843
844void QQuickShapeCurveRenderer::setRootNode(QSGNode *node)
845{
846 clearNodeReferences();
847 m_rootNode = node;
848}
849
850void QQuickShapeCurveRenderer::clearNodeReferences()
851{
852 for (PathData &pd : m_paths) {
853 pd.fillNodes.clear();
854 pd.strokeNodes.clear();
855 }
856}
857
858int QQuickShapeCurveRenderer::debugVisualizationFlags = QQuickShapeCurveRenderer::NoDebug;
859
860int QQuickShapeCurveRenderer::debugVisualization()
861{
862 static const int envFlags = qEnvironmentVariableIntValue("QT_QUICKSHAPES_DEBUG");
863 return debugVisualizationFlags | envFlags;
864}
865
866void QQuickShapeCurveRenderer::setDebugVisualization(int options)
867{
868 if (debugVisualizationFlags == options)
869 return;
870 debugVisualizationFlags = options;
871}
872
873/*! \internal
874 Convert \a path to QSGCurveAbstractNodes with vertices ready to send to the GPU.
875 The given \a path is assumed to be a stroke centerline: it may be continuous or dashed.
876 Also create the wireframe node if enabled.
877*/
878QQuickShapeCurveRenderer::NodeList QQuickShapeCurveRenderer::addCurveStrokeNodes(const QQuadPath &path, const QPen &pen)
879{
880 NodeList ret;
881
882 const bool debug = debugVisualization() & DebugCurves;
883 auto *node = new QSGCurveStrokeNode;
884 node->setDebug(0.2f * debug);
885 QList<QQuickShapeWireFrameNode<StrokeWFT>::WireFrameVertex> wfVertices;
886
887 const float penWidth = pen.widthF();
888
889 static const int subdivisions = qEnvironmentVariable("QT_QUICKSHAPES_STROKE_SUBDIVISIONS", QStringLiteral("3")).toInt();
890
891 const bool wireFrame = debugVisualization() & DebugWireframe;
892 QSGCurveProcessor::processStroke(path,
893 pen.miterLimit(),
894 penWidth, pen.isCosmetic(),
895 pen.joinStyle(),
896 pen.capStyle(),
897 // addStrokeTriangleCallback (see qsgcurveprocessor_p.h):
898 [&wfVertices, &node, &wireFrame](const std::array<QVector2D, 3> &vtx, // triangle corners
899 const std::array<QVector2D, 3> &ctl, // curve control points
900 const std::array<QVector2D, 3> &n, // normals
901 const std::array<float, 3> &ex, // extrusions
902 QSGCurveStrokeNode::TriangleFlags flags)
903 {
904 const QVector2D &v0 = vtx.at(0);
905 const QVector2D &v1 = vtx.at(1);
906 const QVector2D &v2 = vtx.at(2);
907 if (flags.testFlag(QSGCurveStrokeNode::TriangleFlag::Line))
908 node->appendTriangle(vtx, std::array<QVector2D, 2>{ctl.at(0), ctl.at(2)}, n, ex);
909 else
910 node->appendTriangle(vtx, ctl, n, ex);
911
912 if (Q_UNLIKELY(wireFrame)) {
913 wfVertices.append({v0.x(), v0.y(), 1.0f, 0.0f, 0.0f, n.at(0).x(), n.at(0).y(), ex.at(0)});
914 wfVertices.append({v1.x(), v1.y(), 0.0f, 1.0f, 0.0f, n.at(1).x(), n.at(1).y(), ex.at(1)});
915 wfVertices.append({v2.x(), v2.y(), 0.0f, 0.0f, 1.0f, n.at(2).x(), n.at(2).y(), ex.at(2)});
916 }
917 },
918 subdivisions);
919
920 auto indexCopy = node->uncookedIndexes(); // uncookedIndexes get deleted on cooking
921
922 node->setColor(pen.color());
923 node->setStrokeWidth(penWidth);
924 node->setCosmeticStroke(pen.isCosmetic());
925 node->cookGeometry();
926 ret.append(node);
927
928 if (Q_UNLIKELY(wireFrame)) {
929 QQuickShapeWireFrameNode<StrokeWFT> *wfNode = new QQuickShapeWireFrameNode<StrokeWFT>;
930
931 QSGGeometry *wfg = new QSGGeometry(QQuickShapeWireFrameNode<StrokeWFT>::attributes(),
932 wfVertices.size(),
933 indexCopy.size(),
934 QSGGeometry::UnsignedIntType);
935 wfNode->setGeometry(wfg);
936 wfNode->setCosmeticStroke(pen.isCosmetic());
937 wfNode->setStrokeWidth(penWidth);
938
939 wfg->setDrawingMode(QSGGeometry::DrawTriangles);
940 memcpy(wfg->indexData(),
941 indexCopy.data(),
942 indexCopy.size() * wfg->sizeOfIndex());
943 memcpy(wfg->vertexData(),
944 wfVertices.data(),
945 wfg->vertexCount() * wfg->sizeOfVertex());
946
947 ret.append(wfNode);
948 }
949
950 return ret;
951}
952
953QT_END_NAMESPACE
void run() override
Implement this pure virtual function in your subclass.
Combined button and popup list for selecting options.
static bool disableScreenSpaceDerivativeShader()
static QGradient::Type copyCurveGradient(const QQuickShapeGradient *gradient, QSGGradientCache::GradientDesc *dst)