QSkinny 0.8.0
C++/Qt UI toolkit
Loading...
Searching...
No Matches
QskGraphic.cpp
1/******************************************************************************
2 * QSkinny - Copyright (C) The authors
3 * SPDX-License-Identifier: BSD-3-Clause
4 *****************************************************************************/
5
6#include "QskGraphic.h"
7#include "QskColorFilter.h"
8#include "QskGraphicPaintEngine.h"
9#include "QskPainterCommand.h"
10#include "QskInternalMacros.h"
11
12#include <qguiapplication.h>
13#include <qimage.h>
14#include <qmath.h>
15#include <qpaintengine.h>
16#include <qpainter.h>
17#include <qpainterpath.h>
18#include <qpixmap.h>
19#include <qhashfunctions.h>
20
21QSK_QT_PRIVATE_BEGIN
22#include <private/qpainter_p.h>
23QSK_QT_PRIVATE_END
24
25static inline qreal qskDevicePixelRatio()
26{
27 return qGuiApp ? qGuiApp->devicePixelRatio() : 1.0;
28}
29
30static bool qskHasScalablePen( const QPainter* painter )
31{
32 const QPen pen = painter->pen();
33
34 bool scalablePen = false;
35
36 if ( pen.style() != Qt::NoPen && pen.brush().style() != Qt::NoBrush )
37 {
38 scalablePen = !pen.isCosmetic();
39 }
40
41 return scalablePen;
42}
43
44static QRectF qskStrokedPathRect(
45 const QPainter* painter, const QPainterPath& path )
46{
47 const auto pen = painter->pen();
48
49 QPainterPathStroker stroker;
50 stroker.setWidth( pen.widthF() );
51 stroker.setCapStyle( pen.capStyle() );
52 stroker.setJoinStyle( pen.joinStyle() );
53 stroker.setMiterLimit( pen.miterLimit() );
54
55 QRectF rect;
56 if ( qskHasScalablePen( painter ) )
57 {
58 const QPainterPath stroke = stroker.createStroke( path );
59 rect = painter->transform().map( stroke ).boundingRect();
60 }
61 else
62 {
63 QPainterPath mappedPath = painter->transform().map( path );
64 mappedPath = stroker.createStroke( mappedPath );
65
66 rect = mappedPath.boundingRect();
67 }
68
69 return rect;
70}
71
72static inline void qskExecCommand(
73 QPainter* painter, const QskPainterCommand& cmd,
74 const QskColorFilter& colorFilter,
75 QskGraphic::RenderHints renderHints,
76 const QTransform& transform,
77 const QTransform* initialTransform )
78{
79 switch ( cmd.type() )
80 {
81 case QskPainterCommand::Path:
82 {
83 bool doMap = false;
84
85 if ( painter->transform().isScaling() )
86 {
87 if ( painter->pen().isCosmetic() )
88 {
89 // OpenGL2 seems to be buggy for cosmetic pens.
90 // It interpolates curves in too rough steps then
91
92 doMap = painter->paintEngine()->type() == QPaintEngine::OpenGL2;
93 }
94 else
95 {
96 doMap = renderHints.testFlag( QskGraphic::RenderPensUnscaled );
97 }
98 }
99
100 if ( doMap )
101 {
102 const QTransform tr = painter->transform();
103
104 painter->resetTransform();
105
106 QPainterPath path = tr.map( *cmd.path() );
107 if ( initialTransform )
108 {
109 painter->setTransform( *initialTransform );
110 path = initialTransform->inverted().map( path );
111 }
112
113 painter->drawPath( path );
114
115 painter->setTransform( tr );
116 }
117 else
118 {
119 painter->drawPath( *cmd.path() );
120 }
121 break;
122 }
123 case QskPainterCommand::Pixmap:
124 {
125 const auto data = cmd.pixmapData();
126 painter->drawPixmap( data->rect, data->pixmap, data->subRect );
127 break;
128 }
129 case QskPainterCommand::Image:
130 {
131 const auto data = cmd.imageData();
132 painter->drawImage( data->rect, data->image,
133 data->subRect, data->flags );
134 break;
135 }
136 case QskPainterCommand::State:
137 {
138 const auto data = cmd.stateData();
139
140 if ( data->flags & QPaintEngine::DirtyPen )
141 painter->setPen( colorFilter.substituted( data->pen ) );
142
143 if ( data->flags & QPaintEngine::DirtyBrush )
144 painter->setBrush( colorFilter.substituted( data->brush ) );
145
146 if ( data->flags & QPaintEngine::DirtyBrushOrigin )
147 painter->setBrushOrigin( data->brushOrigin );
148
149 if ( data->flags & QPaintEngine::DirtyFont )
150 painter->setFont( data->font );
151
152 if ( data->flags & QPaintEngine::DirtyBackground )
153 {
154 painter->setBackgroundMode( data->backgroundMode );
155 painter->setBackground( colorFilter.substituted( data->backgroundBrush ) );
156 }
157
158 if ( data->flags & QPaintEngine::DirtyTransform )
159 {
160 painter->setTransform( data->transform * transform );
161 }
162
163 if ( data->flags & QPaintEngine::DirtyClipEnabled )
164 painter->setClipping( data->isClipEnabled );
165
166 if ( data->flags & QPaintEngine::DirtyClipRegion )
167 {
168 painter->setClipRegion( data->clipRegion,
169 data->clipOperation );
170 }
171
172 if ( data->flags & QPaintEngine::DirtyClipPath )
173 {
174 painter->setClipPath( data->clipPath, data->clipOperation );
175 }
176
177 if ( data->flags & QPaintEngine::DirtyHints )
178 {
179#if 1
180 auto& state = QPainterPrivate::get( painter )->state;
181 state->renderHints = data->renderHints;
182
183 // to trigger internal updates we have to set at least one flag
184 const auto hint = QPainter::SmoothPixmapTransform;
185 painter->setRenderHint( hint, data->renderHints.testFlag( hint ) );
186#else
187 for ( int i = 0; i < 8; i++ )
188 {
189 const auto hint = static_cast< QPainter::RenderHint >( 1 << i );
190 painter->setRenderHint( hint, data->renderHints.testFlag( hint ) );
191 }
192#endif
193 }
194
195 if ( data->flags & QPaintEngine::DirtyCompositionMode )
196 painter->setCompositionMode( data->compositionMode );
197
198 if ( data->flags & QPaintEngine::DirtyOpacity )
199 painter->setOpacity( data->opacity );
200
201 break;
202 }
203 default:
204 break;
205 }
206}
207
208/*
209 To avoid subobject-linkage warnings, when including the source code in
210 svg2qvg we don't use an anonymous namespace here
211 */
212namespace QskGraphicPrivate
213{
214 class PathInfo
215 {
216 public:
217 PathInfo()
218 : m_scalablePen( false )
219 {
220 // QVector needs a default constructor
221 }
222
223 PathInfo( const QRectF& pointRect,
224 const QRectF& boundingRect, bool scalablePen )
225 : m_pointRect( pointRect )
226 , m_boundingRect( boundingRect )
227 , m_scalablePen( scalablePen )
228 {
229 }
230
231 inline QRectF scaledBoundingRect( qreal sx, qreal sy, bool scalePens ) const
232 {
233 if ( sx == 1.0 && sy == 1.0 )
234 return m_boundingRect;
235
236 QTransform transform;
237 transform.scale( sx, sy );
238
239 QRectF rect;
240 if ( scalePens && m_scalablePen )
241 {
242 rect = transform.mapRect( m_boundingRect );
243 }
244 else
245 {
246 rect = transform.mapRect( m_pointRect );
247
248 const qreal l = qAbs( m_pointRect.left() - m_boundingRect.left() );
249 const qreal r = qAbs( m_pointRect.right() - m_boundingRect.right() );
250 const qreal t = qAbs( m_pointRect.top() - m_boundingRect.top() );
251 const qreal b = qAbs( m_pointRect.bottom() - m_boundingRect.bottom() );
252
253 rect.adjust( -l, -t, r, b );
254 }
255
256 return rect;
257 }
258
259 inline double scaleFactorX( const QRectF& pathRect,
260 const QRectF& targetRect, const QRectF& graphicBoundingRect, bool scalePens ) const
261 {
262 if ( pathRect.width() <= 0.0 )
263 return 0.0;
264
265 const QPointF p0 = m_pointRect.center();
266
267 const auto p = pathRect.united( m_boundingRect );
268
269 const qreal l = qAbs( p.left() - p0.x() );
270 const qreal r = qAbs( p.right() - p0.x() );
271
272 const double w = 2.0 * qMin( l, r ) *
273 targetRect.width() / graphicBoundingRect.width();
274
275 double sx;
276 if ( scalePens && m_scalablePen )
277 {
278 sx = w / m_boundingRect.width();
279 }
280 else
281 {
282 const qreal pw = qMax(
283 qAbs( m_boundingRect.left() - m_pointRect.left() ),
284 qAbs( m_boundingRect.right() - m_pointRect.right() ) );
285
286 sx = ( w - 2 * pw ) / m_pointRect.width();
287 }
288
289 return sx;
290 }
291
292 inline double scaleFactorY( const QRectF& pathRect,
293 const QRectF& targetRect, const QRectF& graphicBoundingRect, bool scalePens ) const
294 {
295 if ( pathRect.height() <= 0.0 )
296 return 0.0;
297
298 const QPointF p0 = m_pointRect.center();
299
300 const auto p = pathRect.united( m_boundingRect );
301
302 const qreal t = qAbs( p.top() - p0.y() );
303 const qreal b = qAbs( p.bottom() - p0.y() );
304
305 const double h = 2.0 * qMin( t, b ) *
306 targetRect.height() / graphicBoundingRect.height();
307
308 double sy;
309 if ( scalePens && m_scalablePen )
310 {
311 sy = h / m_boundingRect.height();
312 }
313 else
314 {
315 const double pw =
316 qMax( qAbs( m_boundingRect.top() - m_pointRect.top() ),
317 qAbs( m_boundingRect.bottom() - m_pointRect.bottom() ) );
318
319 sy = ( h - 2 * pw ) / m_pointRect.height();
320 }
321
322 return sy;
323 }
324
325 private:
326 QRectF m_pointRect;
327 QRectF m_boundingRect;
328 bool m_scalablePen;
329 };
330}
331
332class QskGraphic::PrivateData : public QSharedData
333{
334 public:
335 PrivateData()
336 : commandTypes( 0 )
337 , renderHints( 0 )
338 {
339 }
340
341 PrivateData( const PrivateData& other )
342 : QSharedData( other )
343 , viewBox( other.viewBox )
344 , commands( other.commands )
345 , pathInfos( other.pathInfos )
346 , boundingRect( other.boundingRect )
347 , pointRect( other.pointRect )
348 , modificationId( other.modificationId )
349 , commandTypes( other.commandTypes )
350 , renderHints( other.renderHints )
351 {
352 }
353
354 inline bool operator==( const PrivateData& other ) const
355 {
356 return ( modificationId == other.modificationId ) &&
357 ( renderHints == other.renderHints ) &&
358 ( viewBox == other.viewBox );
359 }
360
361 void resetCommands()
362 {
363 commands.clear();
364 pathInfos.clear();
365
366 commandTypes = 0;
367 boundingRect = pointRect = { 0.0, 0.0, -1.0, -1.0 };
368
369 modificationId = 0;
370 }
371
372 inline void addCommand( const QskPainterCommand& command )
373 {
374 commands += command;
375
376 static QAtomicInteger< quint64 > nextId( 1 );
377 modificationId = nextId.fetchAndAddRelaxed( 1 );
378 }
379
380 QRectF viewBox = { 0.0, 0.0, -1.0, -1.0 };
381 QVector< QskPainterCommand > commands;
382 QVector< QskGraphicPrivate::PathInfo > pathInfos;
383
384 QRectF boundingRect = { 0.0, 0.0, -1.0, -1.0 };
385 QRectF pointRect = { 0.0, 0.0, -1.0, -1.0 };
386
387 quint64 modificationId = 0;
388
389 uint commandTypes : 4;
390 uint renderHints : 4;
391};
392
393QskGraphic::QskGraphic()
394 : m_data( new PrivateData() )
395 , m_paintEngine( nullptr )
396{
397}
398
399QskGraphic::QskGraphic( const QskGraphic& other )
400 : QPaintDevice()
401 , m_data( other.m_data )
402 , m_paintEngine( nullptr )
403{
404}
405
406QskGraphic::QskGraphic( QskGraphic&& other )
407 : m_data( std::move( other.m_data ) )
408 , m_paintEngine( nullptr )
409{
410}
411
412QskGraphic::~QskGraphic()
413{
414 delete m_paintEngine;
415}
416
417QskGraphic& QskGraphic::operator=( const QskGraphic& other )
418{
419 delete m_paintEngine;
420 m_data = other.m_data;
421 m_paintEngine = nullptr;
422
423 return *this;
424}
425
426QskGraphic& QskGraphic::operator=( QskGraphic&& other )
427{
428 m_data = std::move( other.m_data );
429 m_paintEngine = nullptr;
430 return *this;
431}
432
433bool QskGraphic::operator==( const QskGraphic& other ) const
434{
435 return *m_data == *other.m_data;
436}
437
438QPaintEngine* QskGraphic::paintEngine() const
439{
440 if ( m_paintEngine == nullptr )
441 m_paintEngine = new QskGraphicPaintEngine();
442
443 return m_paintEngine;
444}
445
446int QskGraphic::metric( PaintDeviceMetric deviceMetric ) const
447{
448 int value = 0;
449
450 switch ( deviceMetric )
451 {
452 case PdmWidth:
453 {
454 value = qCeil( defaultSize().width() );
455 break;
456 }
457 case PdmHeight:
458 {
459 value = qCeil( defaultSize().height() );
460 break;
461 }
462 case PdmNumColors:
463 {
464 value = 0xffffffff;
465 break;
466 }
467 case PdmDepth:
468 {
469 value = 32;
470 break;
471 }
472 case PdmPhysicalDpiX:
473 case PdmPhysicalDpiY:
474 case PdmDpiY:
475 case PdmDpiX:
476 {
477 value = 72;
478 break;
479 }
480 case PdmWidthMM:
481 {
482 value = qRound( metric( PdmWidth ) * 25.4 / metric( PdmDpiX ) );
483 break;
484 }
485 case PdmHeightMM:
486 {
487 value = qRound( metric( PdmHeight ) * 25.4 / metric( PdmDpiY ) );
488 break;
489 }
490 case PdmDevicePixelRatio:
491 {
492 value = 1.0;
493 break;
494 }
495 case PdmDevicePixelRatioScaled:
496 {
497 value = metric( PdmDevicePixelRatio ) * devicePixelRatioFScale();
498 break;
499 }
500#if QT_VERSION >= QT_VERSION_CHECK( 6, 8, 0 )
501 case PdmDevicePixelRatioF_EncodedA:
502 case PdmDevicePixelRatioF_EncodedB:
503 {
504#if 0
505 value = QPaintDevice::encodeMetricF( metric, devicePixelRatio() );
506#endif
507 break;
508 }
509#endif
510 }
511
512 return value;
513}
514
515void QskGraphic::reset()
516{
517 m_data->resetCommands();
518
519 m_data->viewBox = { 0.0, 0.0, -1.0, -1.0 };
520
521 delete m_paintEngine;
522 m_paintEngine = nullptr;
523}
524
525bool QskGraphic::isNull() const
526{
527 return m_data->commands.isEmpty();
528}
529
530bool QskGraphic::isEmpty() const
531{
532 return m_data->boundingRect.isEmpty();
533}
534
535QskGraphic::CommandTypes QskGraphic::commandTypes() const
536{
537 return static_cast< CommandTypes >( m_data->commandTypes );
538}
539
540void QskGraphic::setRenderHint( RenderHint hint, bool on )
541{
542 if ( on )
543 m_data->renderHints |= hint;
544 else
545 m_data->renderHints &= ~hint;
546}
547
548bool QskGraphic::testRenderHint( RenderHint hint ) const
549{
550 return m_data->renderHints & hint;
551}
552
553QskGraphic::RenderHints QskGraphic::renderHints() const
554{
555 return static_cast< QskGraphic::RenderHints >( m_data->renderHints );
556}
557
558QRectF QskGraphic::boundingRect() const
559{
560 if ( m_data->boundingRect.width() < 0 )
561 return QRectF();
562
563 return m_data->boundingRect;
564}
565
566QRectF QskGraphic::controlPointRect() const
567{
568 if ( m_data->pointRect.width() < 0 )
569 return QRectF();
570
571 return m_data->pointRect;
572}
573
574QRectF QskGraphic::scaledBoundingRect( qreal sx, qreal sy ) const
575{
576 if ( sx == 1.0 && sy == 1.0 )
577 return m_data->boundingRect;
578
579 const bool scalePens = !( m_data->renderHints & RenderPensUnscaled );
580
581 QTransform transform;
582 transform.scale( sx, sy );
583
584 QRectF rect = transform.mapRect( m_data->pointRect );
585
586 for ( const auto& info : std::as_const( m_data->pathInfos ) )
587 rect |= info.scaledBoundingRect( sx, sy, scalePens );
588
589 return rect;
590}
591
592QSize QskGraphic::sizeMetrics() const
593{
594 const QSizeF sz = defaultSize();
595 return QSize( qCeil( sz.width() ), qCeil( sz.height() ) );
596}
597
598void QskGraphic::setViewBox( const QRectF& rect )
599{
600 m_data->viewBox = rect;
601}
602
603QRectF QskGraphic::viewBox() const
604{
605 return m_data->viewBox;
606}
607
608QSizeF QskGraphic::defaultSize() const
609{
610 if ( !m_data->viewBox.isEmpty() )
611 return m_data->viewBox.size();
612
613 return boundingRect().size();
614}
615
616qreal QskGraphic::heightForWidth( qreal width ) const
617{
618 const auto sz = defaultSize();
619 if ( sz.isEmpty() )
620 return 0;
621
622 return sz.height() * width / sz.width();
623}
624
625qreal QskGraphic::widthForHeight( qreal height ) const
626{
627 const auto sz = defaultSize();
628 if ( sz.isEmpty() )
629 return 0;
630
631 return sz.width() * height / sz.height();
632}
633
634qreal QskGraphic::aspectRatio() const
635{
636 const auto sz = defaultSize();
637 if ( sz.isEmpty() )
638 return 1.0;
639
640 return sz.width() / sz.height();
641}
642
643void QskGraphic::render( QPainter* painter ) const
644{
645 render( painter, QskColorFilter() );
646}
647
648void QskGraphic::render( QPainter* painter,
649 const QskColorFilter& colorFilter, QTransform* initialTransform ) const
650{
651 if ( isNull() )
652 return;
653
654 const int numCommands = m_data->commands.size();
655 const auto commands = m_data->commands.constData();
656
657 const auto transform = painter->transform();
658 const QskGraphic::RenderHints renderHints( m_data->renderHints );
659
660 painter->save();
661
662 for ( int i = 0; i < numCommands; i++ )
663 {
664 qskExecCommand( painter, commands[ i ], colorFilter,
665 renderHints, transform, initialTransform );
666 }
667
668 painter->restore();
669}
670
671void QskGraphic::render( QPainter* painter, const QSizeF& size,
672 Qt::AspectRatioMode aspectRatioMode ) const
673{
674 const QRectF r( 0.0, 0.0, size.width(), size.height() );
675 render( painter, r, aspectRatioMode );
676}
677
678void QskGraphic::render( QPainter* painter, const QRectF& rect,
679 Qt::AspectRatioMode aspectRatioMode ) const
680{
681 render( painter, rect, QskColorFilter(), aspectRatioMode );
682}
683
684void QskGraphic::render( QPainter* painter, const QRectF& rect,
685 const QskColorFilter& colorFilter, Qt::AspectRatioMode aspectRatioMode ) const
686{
687 if ( isEmpty() || rect.isEmpty() )
688 return;
689
690 const bool scalePens = !( m_data->renderHints & RenderPensUnscaled );
691
692 qreal sx = 1.0;
693 qreal sy = 1.0;
694
695 QRectF boundingBox = m_data->viewBox;
696
697 if ( !boundingBox.isEmpty() )
698 {
699 sx = rect.width() / boundingBox.width();
700 sy = rect.height() / boundingBox.height();
701 }
702 else
703 {
704 boundingBox = m_data->boundingRect;
705
706 if ( m_data->pointRect.width() > 0.0 )
707 sx = rect.width() / m_data->pointRect.width();
708
709 if ( m_data->pointRect.height() > 0.0 )
710 sy = rect.height() / m_data->pointRect.height();
711
712 for ( const auto& info : std::as_const( m_data->pathInfos ) )
713 {
714 const qreal ssx = info.scaleFactorX( m_data->pointRect,
715 rect, m_data->boundingRect, scalePens );
716
717 if ( ssx > 0.0 )
718 sx = qMin( sx, ssx );
719
720 const qreal ssy = info.scaleFactorY( m_data->pointRect,
721 rect, m_data->boundingRect, scalePens );
722
723 if ( ssy > 0.0 )
724 sy = qMin( sy, ssy );
725 }
726 }
727
728 if ( aspectRatioMode == Qt::KeepAspectRatio )
729 {
730 sx = sy = qMin( sx, sy );
731 }
732 else if ( aspectRatioMode == Qt::KeepAspectRatioByExpanding )
733 {
734 sx = sy = qMax( sx, sy );
735 }
736
737 QTransform tr;
738
739 {
740 const auto rc = rect.center();
741
742 tr.translate(
743 rc.x() - 0.5 * sx * boundingBox.width(),
744 rc.y() - 0.5 * sy * boundingBox.height() );
745 tr.scale( sx, sy );
746 tr.translate( -boundingBox.x(), -boundingBox.y() );
747 }
748
749 const auto transform = painter->transform();
750
751 painter->setTransform( tr, true );
752
753 if ( !scalePens && transform.isScaling() )
754 {
755 /*
756 We don't want to scale pens according to sx/sy,
757 but we want to apply the initial scaling from the
758 painter transformation.
759 */
760
761 QTransform initialTransform;
762 initialTransform.scale( transform.m11(), transform.m22() );
763
764 render( painter, colorFilter, &initialTransform );
765 }
766 else
767 {
768 render( painter, colorFilter, nullptr );
769 }
770
771 painter->setTransform( transform );
772}
773
774void QskGraphic::render( QPainter* painter,
775 const QPointF& pos, Qt::Alignment alignment ) const
776{
777 QRectF r( pos, defaultSize() );
778
779 if ( alignment & Qt::AlignLeft )
780 {
781 r.moveLeft( pos.x() );
782 }
783 else if ( alignment & Qt::AlignHCenter )
784 {
785 r.moveCenter( QPointF( pos.x(), r.center().y() ) );
786 }
787 else if ( alignment & Qt::AlignRight )
788 {
789 r.moveRight( pos.x() );
790 }
791
792 if ( alignment & Qt::AlignTop )
793 {
794 r.moveTop( pos.y() );
795 }
796 else if ( alignment & Qt::AlignVCenter )
797 {
798 r.moveCenter( QPointF( r.center().x(), pos.y() ) );
799 }
800 else if ( alignment & Qt::AlignBottom )
801 {
802 r.moveBottom( pos.y() );
803 }
804
805 render( painter, r );
806}
807
808QPixmap QskGraphic::toPixmap( qreal devicePixelRatio ) const
809{
810 if ( isNull() )
811 return QPixmap();
812
813 if ( devicePixelRatio <= 0.0 )
814 devicePixelRatio = qskDevicePixelRatio();
815
816 const QSizeF sz = defaultSize();
817
818 const int w = qCeil( sz.width() * devicePixelRatio );
819 const int h = qCeil( sz.height() * devicePixelRatio );
820
821 QPixmap pixmap( w, h );
822 pixmap.setDevicePixelRatio( devicePixelRatio );
823 pixmap.fill( Qt::transparent );
824
825 const QRectF r( 0.0, 0.0, sz.width(), sz.height() );
826
827 QPainter painter( &pixmap );
828 render( &painter, r, Qt::KeepAspectRatio );
829 painter.end();
830
831 return pixmap;
832}
833
834QPixmap QskGraphic::toPixmap( const QSize& size,
835 Qt::AspectRatioMode aspectRatioMode, qreal devicePixelRatio ) const
836{
837 if ( devicePixelRatio <= 0.0 )
838 devicePixelRatio = qskDevicePixelRatio();
839
840 const int w = qCeil( size.width() * devicePixelRatio );
841 const int h = qCeil( size.height() * devicePixelRatio );
842
843 QPixmap pixmap( w, h );
844 pixmap.setDevicePixelRatio( devicePixelRatio );
845 pixmap.fill( Qt::transparent );
846
847 const QRect r( 0, 0, size.width(), size.height() );
848
849 QPainter painter( &pixmap );
850 render( &painter, r, aspectRatioMode );
851 painter.end();
852
853 return pixmap;
854}
855
856QImage QskGraphic::toImage( const QSize& size,
857 Qt::AspectRatioMode aspectRatioMode, qreal devicePixelRatio ) const
858{
859 if ( devicePixelRatio <= 0.0 )
860 devicePixelRatio = qskDevicePixelRatio();
861
862 const int w = qCeil( size.width() * devicePixelRatio );
863 const int h = qCeil( size.height() * devicePixelRatio );
864
865 QImage image( w, h, QImage::Format_ARGB32_Premultiplied );
866 image.setDevicePixelRatio( devicePixelRatio );
867 image.fill( 0 );
868
869 const QRect r( 0, 0, size.width(), size.height() );
870
871 QPainter painter( &image );
872 render( &painter, r, aspectRatioMode );
873 painter.end();
874
875 return image;
876}
877
878QImage QskGraphic::toImage( qreal devicePixelRatio ) const
879{
880 if ( isNull() )
881 return QImage();
882
883 if ( devicePixelRatio <= 0.0 )
884 devicePixelRatio = qskDevicePixelRatio();
885
886 const QSizeF sz = defaultSize();
887
888 const int w = qCeil( sz.width() * devicePixelRatio );
889 const int h = qCeil( sz.height() * devicePixelRatio );
890
891 QImage image( w, h, QImage::Format_ARGB32 );
892 image.setDevicePixelRatio( devicePixelRatio );
893 image.fill( 0 );
894
895 const QRect r( 0, 0, sz.width(), sz.height() );
896
897 QPainter painter( &image );
898 render( &painter, r, Qt::KeepAspectRatio );
899 painter.end();
900
901 return image;
902}
903
904void QskGraphic::drawPath( const QPainterPath& path )
905{
906 const auto painter = paintEngine()->painter();
907 if ( painter == nullptr )
908 return;
909
910 addCommand( QskPainterCommand( path ) );
911 m_data->commandTypes |= QskGraphic::VectorData;
912
913 if ( !path.isEmpty() )
914 {
915 const auto scaledPath = painter->transform().map( path );
916
917 QRectF pointRect = scaledPath.boundingRect();
918 QRectF boundingRect = pointRect;
919
920 if ( painter->pen().style() != Qt::NoPen &&
921 painter->pen().brush().style() != Qt::NoBrush )
922 {
923 boundingRect = qskStrokedPathRect( painter, path );
924 }
925
926 updateControlPointRect( pointRect );
927 updateBoundingRect( boundingRect );
928
929 m_data->pathInfos += QskGraphicPrivate::PathInfo( pointRect,
930 boundingRect, qskHasScalablePen( painter ) );
931 }
932}
933
934void QskGraphic::drawPixmap( const QRectF& rect,
935 const QPixmap& pixmap, const QRectF& subRect )
936{
937 const auto painter = paintEngine()->painter();
938 if ( painter == nullptr )
939 return;
940
941 addCommand( QskPainterCommand( rect, pixmap, subRect ) );
942 m_data->commandTypes |= QskGraphic::RasterData;
943
944 const QRectF r = painter->transform().mapRect( rect );
945 updateControlPointRect( r );
946 updateBoundingRect( r );
947}
948
949void QskGraphic::drawImage( const QRectF& rect, const QImage& image,
950 const QRectF& subRect, Qt::ImageConversionFlags flags )
951{
952 const auto painter = paintEngine()->painter();
953 if ( painter == nullptr )
954 return;
955
956 addCommand( QskPainterCommand( rect, image, subRect, flags ) );
957 m_data->commandTypes |= QskGraphic::RasterData;
958
959 const QRectF r = painter->transform().mapRect( rect );
960
961 updateControlPointRect( r );
962 updateBoundingRect( r );
963}
964
965void QskGraphic::updateState( const QPaintEngineState& state )
966{
967 addCommand( QskPainterCommand( state ) );
968
969 if ( state.state() & QPaintEngine::DirtyTransform )
970 {
971 if ( !( m_data->commandTypes & QskGraphic::Transformation ) )
972 {
973 /*
974 QTransform::isScaling() returns true for all type
975 of transformations beside simple translations
976 even if it is f.e a rotation
977 */
978 if ( state.transform().isScaling() )
979 m_data->commandTypes |= QskGraphic::Transformation;
980 }
981 }
982}
983
984void QskGraphic::addCommand( const QskPainterCommand& command )
985{
986 m_data->addCommand( command );
987}
988
989void QskGraphic::updateBoundingRect( const QRectF& rect )
990{
991 QRectF br = rect;
992
993 if ( const auto painter = paintEngine()->painter() )
994 {
995 if ( painter->hasClipping() )
996 {
997 QRectF cr = painter->clipRegion().boundingRect();
998 cr = painter->transform().mapRect( cr );
999
1000 br &= cr;
1001 }
1002 }
1003
1004 if ( m_data->boundingRect.width() < 0 )
1005 m_data->boundingRect = br;
1006 else
1007 m_data->boundingRect |= br;
1008}
1009
1010void QskGraphic::updateControlPointRect( const QRectF& rect )
1011{
1012 if ( m_data->pointRect.width() < 0.0 )
1013 m_data->pointRect = rect;
1014 else
1015 m_data->pointRect |= rect;
1016}
1017
1018const QVector< QskPainterCommand >& QskGraphic::commands() const
1019{
1020 return m_data->commands;
1021}
1022
1023void QskGraphic::setCommands( const QVector< QskPainterCommand >& commands )
1024{
1025 m_data->resetCommands();
1026
1027 const int numCommands = commands.size();
1028 if ( numCommands <= 0 )
1029 return;
1030
1031 // to calculate a proper bounding rectangle we don't simply copy
1032 // the commands.
1033
1034 const auto cmds = commands.constData();
1035
1036 const QskColorFilter noFilter;
1037 const QTransform noTransform;
1038
1039 QPainter painter( this );
1040
1041 for ( int i = 0; i < numCommands; i++ )
1042 {
1043 qskExecCommand( &painter, cmds[ i ],
1044 noFilter, RenderHints(), noTransform, nullptr );
1045 }
1046
1047 painter.end();
1048}
1049
1050quint64 QskGraphic::modificationId() const
1051{
1052 return m_data->modificationId;
1053}
1054
1055QskHashValue QskGraphic::hash( QskHashValue seed ) const
1056{
1057 auto hash = qHash( m_data->renderHints, seed );
1058 hash = qHashBits( &m_data->viewBox, sizeof( QRectF ), hash );
1059
1060 return qHash( m_data->modificationId, hash );
1061}
1062
1063QskGraphic QskGraphic::fromImage( const QImage& image )
1064{
1065 QskGraphic graphic;
1066
1067 if ( !image.isNull() )
1068 {
1069 QPainter painter( &graphic );
1070 painter.drawImage( 0, 0, image );
1071 painter.end();
1072 }
1073
1074 return graphic;
1075}
1076
1077QskGraphic QskGraphic::fromPixmap( const QPixmap& pixmap )
1078{
1079 QskGraphic graphic;
1080
1081 if ( !pixmap.isNull() )
1082 {
1083 QPainter painter( &graphic );
1084 painter.drawPixmap( 0, 0, pixmap );
1085 painter.end();
1086 }
1087
1088 return graphic;
1089}
1090
1091QskGraphic QskGraphic::fromPixmapAsImage( const QPixmap& pixmap )
1092{
1093 /*
1094 Using QPainter::drawPixmap in the scene graph thread leads
1095 to warnings about "not being safe". This is probably not critical
1096 for Qt/Quick as the main thread is waiting, when updating the
1097 scene graph nodes.
1098
1099 It needs to be checked, what is going on, when
1100 using the X11 paint engine, where QPixmap/QImage are more different.
1101
1102 TODO ...
1103 */
1104
1105 return fromImage( pixmap.toImage() );
1106}
1107
1108QskGraphic QskGraphic::fromGraphic(
1109 const QskGraphic& graphic, const QskColorFilter& colorFilter )
1110{
1111 if ( colorFilter.isIdentity() )
1112 return graphic;
1113
1114 QskGraphic recoloredGraphic;
1115
1116 QPainter painter( &recoloredGraphic );
1117 graphic.render( &painter, colorFilter );
1118 painter.end();
1119
1120 return recoloredGraphic;
1121}
1122
1123#ifndef QT_NO_DEBUG_STREAM
1124
1125#include <qdebug.h>
1126
1127QDebug operator<<( QDebug debug, const QskGraphic& graphic )
1128{
1129 QDebugStateSaver saver( debug );
1130
1131 debug << "Graphic" << '(';
1132 debug << "\n boundingRect:" << graphic.boundingRect();
1133 debug << "\n controlPointsRect:" << graphic.boundingRect();
1134 debug << "\n commandTypes:" << graphic.commandTypes();
1135
1136 for ( const auto& command : graphic.commands() )
1137 {
1138 switch ( command.type() )
1139 {
1140 case QskPainterCommand::Path:
1141 {
1142 const auto& path = *command.path();
1143
1144 debug << "\n Path(" << path.elementCount() << ")";
1145
1146 const char* types[] = { "MoveTo", "LineTo", "CurveTo", "CurveToData" };
1147
1148 for ( int i = 0; i < path.elementCount(); i++ )
1149 {
1150 debug << "\n ";
1151
1152 const auto el = path.elementAt(i);
1153 debug << types[ el.type] << el.x << el.y;
1154 }
1155
1156 break;
1157 }
1158 case QskPainterCommand::Pixmap:
1159 {
1160 const auto& pixmapData = *command.pixmapData();
1161
1162 debug << "\n Pixmap:";
1163 debug << "\n " << pixmapData.pixmap;
1164 debug << "\n Rect:" << pixmapData.rect;
1165 debug << "\n SubRect:" << pixmapData.subRect;
1166 break;
1167 }
1168 case QskPainterCommand::Image:
1169 {
1170 const auto& imageData = *command.imageData();
1171
1172 debug << "\n Image:";
1173 debug << "\n " << imageData.image;
1174 debug << "\n ConversionFlags" << imageData.flags;
1175 debug << "\n Rect:" << imageData.rect;
1176 debug << "\n SubRect:" << imageData.subRect;
1177
1178 break;
1179 }
1180 case QskPainterCommand::State:
1181 {
1182 const auto& stateData = *command.stateData();
1183 const auto flags = stateData.flags;
1184
1185 debug << "\n State:";
1186
1187 const char indent[] = "\n ";
1188
1189 if ( flags & QPaintEngine::DirtyPen )
1190 {
1191 debug << indent << "Pen:" << stateData.pen;
1192 }
1193
1194 if ( flags & QPaintEngine::DirtyBrush )
1195 {
1196 debug << indent << "Brush:" << stateData.brush;
1197 }
1198
1199 if ( flags & QPaintEngine::DirtyBrushOrigin )
1200 {
1201 debug << indent << "BrushOrigin:" << stateData.brushOrigin;
1202 }
1203
1204 if ( flags & QPaintEngine::DirtyFont )
1205 {
1206 debug << indent << "Font:" << stateData.font;
1207 }
1208
1209 if ( flags & QPaintEngine::DirtyBackground )
1210 {
1211 debug << indent << "Background:"
1212 << stateData.backgroundMode
1213 << stateData.backgroundBrush;
1214 }
1215
1216 if ( flags & QPaintEngine::DirtyTransform )
1217 {
1218 debug << indent << "Transform: " << stateData.transform;
1219 }
1220
1221 if ( flags & QPaintEngine::DirtyClipEnabled )
1222 {
1223 debug << indent << "ClipEnabled: " << stateData.isClipEnabled;
1224 }
1225
1226 if ( flags & QPaintEngine::DirtyClipRegion )
1227 {
1228 debug << indent << "ClipRegion: " << stateData.clipOperation;
1229 }
1230
1231 if ( flags & QPaintEngine::DirtyClipPath )
1232 {
1233 debug << indent << "ClipPath:" << stateData.clipOperation;
1234 }
1235
1236 if ( flags & QPaintEngine::DirtyHints )
1237 {
1238 debug << indent << "RenderHints:" << stateData.renderHints;
1239 }
1240
1241 if ( flags & QPaintEngine::DirtyCompositionMode )
1242 {
1243 debug << indent << "CompositionMode:" << stateData.compositionMode;
1244 }
1245
1246 if ( flags & QPaintEngine::DirtyOpacity )
1247 {
1248 debug << indent << "Opacity:" << stateData.opacity;
1249 }
1250
1251 break;
1252 }
1253 default:
1254 break;
1255 }
1256 }
1257
1258 debug << "\n)";
1259
1260 return debug;
1261}
1262
1263#endif
1264
1265#include "moc_QskGraphic.cpp"
A paint device for scalable graphics.
Definition QskGraphic.h:28