33#include <QSvgRenderer>
36#include <QDomDocument>
45static constexpr int MAX_FONT_CHARACTER_SIZE_IN_PIXELS = 500;
47static void _fixQPictureDPI( QPainter *p )
53 p->scale(
static_cast< double >(
qt_defaultDpiX() ) / p->device()->logicalDpiX(),
54 static_cast< double >(
qt_defaultDpiY() ) / p->device()->logicalDpiY() );
67 QList< Qgis::MarkerShape > shapes;
182 QTransform transform;
185 if ( !hasDataDefinedSize )
195 const double half = scaledSize / 2.0;
196 transform.scale( half, half );
202 transform.rotate(
mAngle );
229 bool hasDataDefinedSize =
false;
230 const double scaledSize =
calculateSize( context, hasDataDefinedSize );
232 bool hasDataDefinedRotation =
false;
238 bool createdNewPath =
false;
256 createdNewPath =
true;
265 QTransform transform;
268 transform.translate( point.x() +
offset.x(), point.y() +
offset.y() );
271 if ( hasDataDefinedSize || createdNewPath )
280 const double half = s / 2.0;
281 transform.scale( half, half );
286 transform.rotate(
angle );
295 polygon = transform.map(
mPolygon );
299 path = transform.map(
mPath );
301 draw( context, symbol, polygon, path );
306 bool hasDataDefinedSize =
false;
307 double scaledSize =
calculateSize( context, hasDataDefinedSize );
309 bool hasDataDefinedRotation =
false;
316 QTransform transform;
319 transform.translate( point.x() +
offset.x(), point.y() +
offset.y() );
322 transform.rotate(
angle );
324 return transform.mapRect( QRectF( -scaledSize / 2.0,
334 const QString cleaned = name.toLower().trimmed();
336 if ( cleaned == QLatin1String(
"square" ) || cleaned == QLatin1String(
"rectangle" ) )
338 else if ( cleaned == QLatin1String(
"trapezoid" ) )
340 else if ( cleaned == QLatin1String(
"parallelogram_right" ) )
342 else if ( cleaned == QLatin1String(
"parallelogram_left" ) )
344 else if ( cleaned == QLatin1String(
"square_with_corners" ) )
346 else if ( cleaned == QLatin1String(
"rounded_square" ) )
348 else if ( cleaned == QLatin1String(
"diamond" ) )
350 else if ( cleaned == QLatin1String(
"shield" ) )
352 else if ( cleaned == QLatin1String(
"pentagon" ) )
354 else if ( cleaned == QLatin1String(
"hexagon" ) )
356 else if ( cleaned == QLatin1String(
"octagon" ) )
358 else if ( cleaned == QLatin1String(
"decagon" ) )
360 else if ( cleaned == QLatin1String(
"triangle" ) )
362 else if ( cleaned == QLatin1String(
"equilateral_triangle" ) )
364 else if ( cleaned == QLatin1String(
"star_diamond" ) )
366 else if ( cleaned == QLatin1String(
"star" ) || cleaned == QLatin1String(
"regular_star" ) )
368 else if ( cleaned == QLatin1String(
"heart" ) )
370 else if ( cleaned == QLatin1String(
"arrow" ) )
372 else if ( cleaned == QLatin1String(
"circle" ) )
374 else if ( cleaned == QLatin1String(
"cross" ) )
376 else if ( cleaned == QLatin1String(
"cross_fill" ) )
378 else if ( cleaned == QLatin1String(
"cross2" ) || cleaned == QLatin1String(
"x" ) )
380 else if ( cleaned == QLatin1String(
"line" ) )
382 else if ( cleaned == QLatin1String(
"arrowhead" ) )
384 else if ( cleaned == QLatin1String(
"filled_arrowhead" ) )
386 else if ( cleaned == QLatin1String(
"semi_circle" ) )
388 else if ( cleaned == QLatin1String(
"third_circle" ) )
390 else if ( cleaned == QLatin1String(
"quarter_circle" ) )
392 else if ( cleaned == QLatin1String(
"quarter_square" ) )
394 else if ( cleaned == QLatin1String(
"half_square" ) )
396 else if ( cleaned == QLatin1String(
"diagonal_half_square" ) )
398 else if ( cleaned == QLatin1String(
"right_half_triangle" ) )
400 else if ( cleaned == QLatin1String(
"left_half_triangle" ) )
402 else if ( cleaned == QLatin1String(
"asterisk_fill" ) )
404 else if ( cleaned == QLatin1String(
"half_arc" ) )
406 else if ( cleaned == QLatin1String(
"third_arc" ) )
408 else if ( cleaned == QLatin1String(
"quarter_arc" ) )
421 return QStringLiteral(
"square" );
423 return QStringLiteral(
"quarter_square" );
425 return QStringLiteral(
"half_square" );
427 return QStringLiteral(
"diagonal_half_square" );
429 return QStringLiteral(
"parallelogram_right" );
431 return QStringLiteral(
"parallelogram_left" );
433 return QStringLiteral(
"trapezoid" );
435 return QStringLiteral(
"shield" );
437 return QStringLiteral(
"diamond" );
439 return QStringLiteral(
"pentagon" );
441 return QStringLiteral(
"hexagon" );
443 return QStringLiteral(
"octagon" );
445 return QStringLiteral(
"decagon" );
447 return QStringLiteral(
"square_with_corners" );
449 return QStringLiteral(
"rounded_square" );
451 return QStringLiteral(
"triangle" );
453 return QStringLiteral(
"equilateral_triangle" );
455 return QStringLiteral(
"left_half_triangle" );
457 return QStringLiteral(
"right_half_triangle" );
459 return QStringLiteral(
"star_diamond" );
461 return QStringLiteral(
"star" );
463 return QStringLiteral(
"heart" );
465 return QStringLiteral(
"arrow" );
467 return QStringLiteral(
"filled_arrowhead" );
469 return QStringLiteral(
"cross_fill" );
471 return QStringLiteral(
"circle" );
473 return QStringLiteral(
"cross" );
475 return QStringLiteral(
"cross2" );
477 return QStringLiteral(
"line" );
479 return QStringLiteral(
"arrowhead" );
481 return QStringLiteral(
"semi_circle" );
483 return QStringLiteral(
"third_circle" );
485 return QStringLiteral(
"quarter_circle" );
487 return QStringLiteral(
"asterisk_fill" );
489 return QStringLiteral(
"half_arc" );
491 return QStringLiteral(
"third_arc" );
493 return QStringLiteral(
"quarter_arc" );
510 polygon = QPolygonF( QRectF( QPointF( -1, -1 ), QPointF( 1, 1 ) ) );
515 static constexpr double VERTEX_OFFSET_FROM_ORIGIN = 0.6072;
517 polygon << QPointF( - VERTEX_OFFSET_FROM_ORIGIN, 1 )
518 << QPointF( VERTEX_OFFSET_FROM_ORIGIN, 1 )
519 << QPointF( 1, VERTEX_OFFSET_FROM_ORIGIN )
520 << QPointF( 1, -VERTEX_OFFSET_FROM_ORIGIN )
521 << QPointF( VERTEX_OFFSET_FROM_ORIGIN, -1 )
522 << QPointF( -VERTEX_OFFSET_FROM_ORIGIN, -1 )
523 << QPointF( -1, -VERTEX_OFFSET_FROM_ORIGIN )
524 << QPointF( -1, VERTEX_OFFSET_FROM_ORIGIN )
525 << QPointF( -VERTEX_OFFSET_FROM_ORIGIN, 1 );
530 polygon = QPolygonF( QRectF( QPointF( -1, -1 ), QPointF( 0, 0 ) ) );
534 polygon = QPolygonF( QRectF( QPointF( -1, -1 ), QPointF( 0, 1 ) ) );
538 polygon << QPointF( -1, -1 ) << QPointF( 1, 1 ) << QPointF( -1, 1 ) << QPointF( -1, -1 );
542 polygon << QPointF( 0.5, -0.5 )
544 << QPointF( -1, 0.5 )
545 << QPointF( -0.5, -0.5 )
546 << QPointF( 0.5, -0.5 );
550 polygon << QPointF( 0.5, 0.5 )
551 << QPointF( 1, -0.5 )
552 << QPointF( -0.5, -0.5 )
553 << QPointF( -1, 0.5 )
554 << QPointF( 0.5, 0.5 );
558 polygon << QPointF( 1, 0.5 )
559 << QPointF( 0.5, -0.5 )
560 << QPointF( -1, -0.5 )
561 << QPointF( -0.5, 0.5 )
562 << QPointF( 1, 0.5 );
566 polygon << QPointF( -1, 0 ) << QPointF( 0, 1 )
567 << QPointF( 1, 0 ) << QPointF( 0, -1 ) << QPointF( -1, 0 );
571 polygon << QPointF( 1, 0.5 )
574 << QPointF( -1, 0.5 )
576 << QPointF( 1, 0.5 );
586 polygon << QPointF( -0.9511, -0.3090 )
587 << QPointF( -0.5878, 0.8090 )
588 << QPointF( 0.5878, 0.8090 )
589 << QPointF( 0.9511, -0.3090 )
591 << QPointF( -0.9511, -0.3090 );
602 polygon << QPointF( -0.8660, -0.5 )
603 << QPointF( -0.8660, 0.5 )
605 << QPointF( 0.8660, 0.5 )
606 << QPointF( 0.8660, -0.5 )
608 << QPointF( -0.8660, -0.5 );
613 static constexpr double VERTEX_OFFSET_FROM_ORIGIN = 1.0 / ( 1 + M_SQRT2 );
615 polygon << QPointF( - VERTEX_OFFSET_FROM_ORIGIN, 1 )
616 << QPointF( VERTEX_OFFSET_FROM_ORIGIN, 1 )
617 << QPointF( 1, VERTEX_OFFSET_FROM_ORIGIN )
618 << QPointF( 1, -VERTEX_OFFSET_FROM_ORIGIN )
619 << QPointF( VERTEX_OFFSET_FROM_ORIGIN, -1 )
620 << QPointF( -VERTEX_OFFSET_FROM_ORIGIN, -1 )
621 << QPointF( -1, -VERTEX_OFFSET_FROM_ORIGIN )
622 << QPointF( -1, VERTEX_OFFSET_FROM_ORIGIN )
623 << QPointF( -VERTEX_OFFSET_FROM_ORIGIN, 1 );
630 polygon << QPointF( 0.587785252, 0.809016994 )
631 << QPointF( 0.951056516, 0.309016994 )
632 << QPointF( 0.951056516, -0.309016994 )
633 << QPointF( 0.587785252, -0.809016994 )
635 << QPointF( -0.587785252, -0.809016994 )
636 << QPointF( -0.951056516, -0.309016994 )
637 << QPointF( -0.951056516, 0.309016994 )
638 << QPointF( -0.587785252, 0.809016994 )
640 << QPointF( 0.587785252, 0.809016994 );
645 polygon << QPointF( -1, 1 ) << QPointF( 1, 1 ) << QPointF( 0, -1 ) << QPointF( -1, 1 );
653 polygon << QPointF( -0.8660, 0.5 )
654 << QPointF( 0.8660, 0.5 )
656 << QPointF( -0.8660, 0.5 );
660 polygon << QPointF( 0, 1 ) << QPointF( 1, 1 ) << QPointF( 0, -1 ) << QPointF( 0, 1 );
664 polygon << QPointF( -1, 1 ) << QPointF( 0, 1 ) << QPointF( 0, -1 ) << QPointF( -1, 1 );
669 const double inner_r = std::cos(
DEG2RAD( 72.0 ) ) / std::cos(
DEG2RAD( 36.0 ) );
671 polygon << QPointF( inner_r * std::sin(
DEG2RAD( 315.0 ) ), - inner_r * std::cos(
DEG2RAD( 315.0 ) ) )
672 << QPointF( std::sin(
DEG2RAD( 270 ) ), - std::cos(
DEG2RAD( 270 ) ) )
673 << QPointF( inner_r * std::sin(
DEG2RAD( 225.0 ) ), - inner_r * std::cos(
DEG2RAD( 225.0 ) ) )
674 << QPointF( std::sin(
DEG2RAD( 180 ) ), - std::cos(
DEG2RAD( 180 ) ) )
675 << QPointF( inner_r * std::sin(
DEG2RAD( 135.0 ) ), - inner_r * std::cos(
DEG2RAD( 135.0 ) ) )
676 << QPointF( std::sin(
DEG2RAD( 90 ) ), - std::cos(
DEG2RAD( 90 ) ) )
677 << QPointF( inner_r * std::sin(
DEG2RAD( 45.0 ) ), - inner_r * std::cos(
DEG2RAD( 45.0 ) ) )
678 << QPointF( std::sin(
DEG2RAD( 0 ) ), - std::cos(
DEG2RAD( 0 ) ) );
684 const double inner_r = std::cos(
DEG2RAD( 72.0 ) ) / std::cos(
DEG2RAD( 36.0 ) );
686 polygon << QPointF( inner_r * std::sin(
DEG2RAD( 324.0 ) ), - inner_r * std::cos(
DEG2RAD( 324.0 ) ) )
687 << QPointF( std::sin(
DEG2RAD( 288.0 ) ), - std::cos(
DEG2RAD( 288 ) ) )
688 << QPointF( inner_r * std::sin(
DEG2RAD( 252.0 ) ), - inner_r * std::cos(
DEG2RAD( 252.0 ) ) )
689 << QPointF( std::sin(
DEG2RAD( 216.0 ) ), - std::cos(
DEG2RAD( 216.0 ) ) )
690 << QPointF( 0, inner_r )
691 << QPointF( std::sin(
DEG2RAD( 144.0 ) ), - std::cos(
DEG2RAD( 144.0 ) ) )
692 << QPointF( inner_r * std::sin(
DEG2RAD( 108.0 ) ), - inner_r * std::cos(
DEG2RAD( 108.0 ) ) )
693 << QPointF( std::sin(
DEG2RAD( 72.0 ) ), - std::cos(
DEG2RAD( 72.0 ) ) )
694 << QPointF( inner_r * std::sin(
DEG2RAD( 36.0 ) ), - inner_r * std::cos(
DEG2RAD( 36.0 ) ) )
696 << QPointF( inner_r * std::sin(
DEG2RAD( 324.0 ) ), - inner_r * std::cos(
DEG2RAD( 324.0 ) ) );
701 polygon << QPointF( 0, -1 )
702 << QPointF( 0.5, -0.5 )
703 << QPointF( 0.25, -0.5 )
704 << QPointF( 0.25, 1 )
705 << QPointF( -0.25, 1 )
706 << QPointF( -0.25, -0.5 )
707 << QPointF( -0.5, -0.5 )
712 polygon << QPointF( 0, 0 ) << QPointF( -1, 1 ) << QPointF( -1, -1 ) << QPointF( 0, 0 );
716 polygon << QPointF( -1, -0.2 )
717 << QPointF( -1, -0.2 )
718 << QPointF( -1, 0.2 )
719 << QPointF( -0.2, 0.2 )
720 << QPointF( -0.2, 1 )
722 << QPointF( 0.2, 0.2 )
724 << QPointF( 1, -0.2 )
725 << QPointF( 0.2, -0.2 )
726 << QPointF( 0.2, -1 )
727 << QPointF( -0.2, -1 )
728 << QPointF( -0.2, -0.2 )
729 << QPointF( -1, -0.2 );
734 static constexpr double THICKNESS = 0.3;
735 static constexpr double HALF_THICKNESS = THICKNESS / 2.0;
736 static constexpr double INTERSECTION_POINT = THICKNESS / M_SQRT2;
737 static constexpr double DIAGONAL1 = M_SQRT1_2 - INTERSECTION_POINT * 0.5;
738 static constexpr double DIAGONAL2 = M_SQRT1_2 + INTERSECTION_POINT * 0.5;
740 polygon << QPointF( -HALF_THICKNESS, -1 )
741 << QPointF( HALF_THICKNESS, -1 )
742 << QPointF( HALF_THICKNESS, -HALF_THICKNESS - INTERSECTION_POINT )
743 << QPointF( DIAGONAL1, -DIAGONAL2 )
744 << QPointF( DIAGONAL2, -DIAGONAL1 )
745 << QPointF( HALF_THICKNESS + INTERSECTION_POINT, -HALF_THICKNESS )
746 << QPointF( 1, -HALF_THICKNESS )
747 << QPointF( 1, HALF_THICKNESS )
748 << QPointF( HALF_THICKNESS + INTERSECTION_POINT, HALF_THICKNESS )
749 << QPointF( DIAGONAL2, DIAGONAL1 )
750 << QPointF( DIAGONAL1, DIAGONAL2 )
751 << QPointF( HALF_THICKNESS, HALF_THICKNESS + INTERSECTION_POINT )
752 << QPointF( HALF_THICKNESS, 1 )
753 << QPointF( -HALF_THICKNESS, 1 )
754 << QPointF( -HALF_THICKNESS, HALF_THICKNESS + INTERSECTION_POINT )
755 << QPointF( -DIAGONAL1, DIAGONAL2 )
756 << QPointF( -DIAGONAL2, DIAGONAL1 )
757 << QPointF( -HALF_THICKNESS - INTERSECTION_POINT, HALF_THICKNESS )
758 << QPointF( -1, HALF_THICKNESS )
759 << QPointF( -1, -HALF_THICKNESS )
760 << QPointF( -HALF_THICKNESS - INTERSECTION_POINT, -HALF_THICKNESS )
761 << QPointF( -DIAGONAL2, -DIAGONAL1 )
762 << QPointF( -DIAGONAL1, -DIAGONAL2 )
763 << QPointF( -HALF_THICKNESS, -HALF_THICKNESS - INTERSECTION_POINT )
764 << QPointF( -HALF_THICKNESS, -1 );
789 mPath = QPainterPath();
795 mPath.addEllipse( QRectF( -1, -1, 2, 2 ) );
799 mPath.moveTo( -1, -1 );
800 mPath.addRoundedRect( -1, -1, 2, 2, 0.25, 0.25 );
804 mPath.arcTo( -1, -1, 2, 2, 0, 180 );
805 mPath.lineTo( 0, 0 );
809 mPath.arcTo( -1, -1, 2, 2, 90, 120 );
810 mPath.lineTo( 0, 0 );
814 mPath.arcTo( -1, -1, 2, 2, 90, 90 );
815 mPath.lineTo( 0, 0 );
819 mPath.moveTo( 1, 0 );
820 mPath.arcTo( -1, -1, 2, 2, 0, 180 );
824 mPath.moveTo( 0, -1 );
825 mPath.arcTo( -1, -1, 2, 2, 90, 120 );
829 mPath.moveTo( 0, -1 );
830 mPath.arcTo( -1, -1, 2, 2, 90, 90 );
834 mPath.moveTo( -1, 0 );
835 mPath.lineTo( 1, 0 );
836 mPath.moveTo( 0, -1 );
837 mPath.lineTo( 0, 1 );
841 mPath.moveTo( -1, -1 );
842 mPath.lineTo( 1, 1 );
843 mPath.moveTo( 1, -1 );
844 mPath.lineTo( -1, 1 );
848 mPath.moveTo( 0, -1 );
849 mPath.lineTo( 0, 1 );
853 mPath.moveTo( -1, -1 );
854 mPath.lineTo( 0, 0 );
855 mPath.lineTo( -1, 1 );
859 mPath.moveTo( 0, 0.75 );
860 mPath.arcTo( 0, -1, 1, 1, -45, 210 );
861 mPath.arcTo( -1, -1, 1, 1, 15, 210 );
862 mPath.lineTo( 0, 0.75 );
896 double scaledSize =
mSize;
900 if ( hasDataDefinedSize )
907 if ( hasDataDefinedSize && ok )
912 scaledSize = std::sqrt( scaledSize );
927 markerOffset( context, scaledSize, scaledSize, offsetX, offsetY );
928 offset = QPointF( offsetX, offsetY );
930 hasDataDefinedRotation =
false;
943 hasDataDefinedRotation =
true;
948 if ( hasDataDefinedRotation )
977 , mStrokeColor( strokeColor )
978 , mPenJoinStyle( penJoinStyle )
995 if ( props.contains( QStringLiteral(
"name" ) ) )
999 if ( props.contains( QStringLiteral(
"color" ) ) )
1001 if ( props.contains( QStringLiteral(
"color_border" ) ) )
1006 else if ( props.contains( QStringLiteral(
"outline_color" ) ) )
1010 else if ( props.contains( QStringLiteral(
"line_color" ) ) )
1014 if ( props.contains( QStringLiteral(
"joinstyle" ) ) )
1018 if ( props.contains( QStringLiteral(
"size" ) ) )
1019 size = props[QStringLiteral(
"size" )].toDouble();
1020 if ( props.contains( QStringLiteral(
"angle" ) ) )
1021 angle = props[QStringLiteral(
"angle" )].toDouble();
1022 if ( props.contains( QStringLiteral(
"scale_method" ) ) )
1026 if ( props.contains( QStringLiteral(
"offset" ) ) )
1028 if ( props.contains( QStringLiteral(
"offset_unit" ) ) )
1030 if ( props.contains( QStringLiteral(
"offset_map_unit_scale" ) ) )
1032 if ( props.contains( QStringLiteral(
"size_unit" ) ) )
1034 if ( props.contains( QStringLiteral(
"size_map_unit_scale" ) ) )
1037 if ( props.contains( QStringLiteral(
"outline_style" ) ) )
1041 else if ( props.contains( QStringLiteral(
"line_style" ) ) )
1045 if ( props.contains( QStringLiteral(
"outline_width" ) ) )
1047 m->
setStrokeWidth( props[QStringLiteral(
"outline_width" )].toDouble() );
1049 else if ( props.contains( QStringLiteral(
"line_width" ) ) )
1051 m->
setStrokeWidth( props[QStringLiteral(
"line_width" )].toDouble() );
1053 if ( props.contains( QStringLiteral(
"outline_width_unit" ) ) )
1057 if ( props.contains( QStringLiteral(
"line_width_unit" ) ) )
1061 if ( props.contains( QStringLiteral(
"outline_width_map_unit_scale" ) ) )
1066 if ( props.contains( QStringLiteral(
"horizontal_anchor_point" ) ) )
1070 if ( props.contains( QStringLiteral(
"vertical_anchor_point" ) ) )
1075 if ( props.contains( QStringLiteral(
"cap_style" ) ) )
1088 return QStringLiteral(
"SimpleMarker" );
1095 QColor brushColor =
mColor;
1098 brushColor.setAlphaF(
mColor.alphaF() * context.
opacity() );
1101 mBrush = QBrush( brushColor );
1102 mPen = QPen( penColor );
1112 selBrushColor.setAlphaF( context.
opacity() );
1113 selPenColor.setAlphaF( context.
opacity() );
1116 mSelPen = QPen( selPenColor );
1132 mCachedOpacity = context.
opacity();
1138 mSelPen.setColor( selBrushColor );
1171 scaledSize = ( std::abs( std::sin(
mAngle * M_PI / 180 ) ) + std::abs( std::cos(
mAngle * M_PI / 180 ) ) ) * scaledSize;
1174 const double pw =
static_cast< int >( std::round( ( (
qgsDoubleNear(
mPen.widthF(), 0.0 ) ? 1 :
mPen.widthF() * 4 ) + 1 ) ) ) / 2 * 2;
1175 const int imageSize = (
static_cast< int >( scaledSize ) + pw ) / 2 * 2 + 1;
1176 const double center = imageSize / 2.0;
1182 mCache = QImage( QSize( imageSize * deviceRatio,
1183 imageSize * deviceRatio ), QImage::Format_ARGB32_Premultiplied );
1193 p.setRenderHint( QPainter::Antialiasing );
1194 p.setBrush( needsBrush ?
mBrush : Qt::NoBrush );
1196 p.translate( QPointF( center, center ) );
1204 mSelCache = QImage( QSize( imageSize, imageSize ), QImage::Format_ARGB32_Premultiplied );
1208 p.setRenderHint( QPainter::Antialiasing );
1209 p.setBrush( needsBrush ?
mSelBrush : Qt::NoBrush );
1211 p.translate( QPointF( center, center ) );
1222 p.setRenderHint( QPainter::Antialiasing );
1223 p.fillRect( 0, 0, imageSize, imageSize, selColor );
1224 p.setBrush( needsBrush ?
mBrush : Qt::NoBrush );
1226 p.translate( QPointF( center, center ) );
1245 QColor brushColor =
mColor;
1246 brushColor.setAlphaF( brushColor.alphaF() * context.
opacity() );
1247 mBrush.setColor( brushColor );
1250 penColor.setAlphaF( penColor.alphaF() * context.
opacity() );
1251 mPen.setColor( penColor );
1260 c.setAlphaF(
c.alphaF() * context.
opacity() );
1270 c.setAlphaF(
c.alphaF() * context.
opacity() );
1323 p->setBrush( Qt::NoBrush );
1327 if ( !polygon.isEmpty() )
1328 p->drawPolygon( polygon );
1330 p->drawPath( path );
1348 const double s = img.width() / img.devicePixelRatioF();
1350 bool hasDataDefinedSize =
false;
1351 const double scaledSize =
calculateSize( context, hasDataDefinedSize );
1353 bool hasDataDefinedRotation =
false;
1358 p->drawImage( QRectF( point.x() - s / 2.0 +
offset.x(),
1359 point.y() - s / 2.0 +
offset.y(),
1374 map[QStringLiteral(
"size" )] = QString::number(
mSize );
1377 map[QStringLiteral(
"angle" )] = QString::number(
mAngle );
1383 map[QStringLiteral(
"outline_width" )] = QString::number(
mStrokeWidth );
1416 QDomElement graphicElem = doc.createElement( QStringLiteral(
"se:Graphic" ) );
1417 element.appendChild( graphicElem );
1433 const double angle = props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toDouble( &ok );
1436 angleFunc = QStringLiteral(
"%1 + %2" ).arg( props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toString() ).arg(
mAngle );
1453 Q_UNUSED( mmScaleFactor )
1454 Q_UNUSED( mapUnitScaleFactor )
1456 QString ogrType =
"3";
1457 if ( mName ==
"square" )
1461 else if ( mName ==
"triangle" )
1465 else if ( mName ==
"star" )
1469 else if ( mName ==
"circle" )
1473 else if ( mName ==
"cross" )
1477 else if ( mName ==
"x" || mName ==
"cross2" )
1481 else if ( mName ==
"line" )
1487 ogrString.append(
"SYMBOL(" );
1488 ogrString.append(
"id:" );
1489 ogrString.append(
'\"' );
1490 ogrString.append(
"ogr-sym-" );
1491 ogrString.append( ogrType );
1492 ogrString.append(
'\"' );
1493 ogrString.append(
",c:" );
1494 ogrString.append(
mColor.name() );
1495 ogrString.append(
",o:" );
1497 ogrString.append( QString(
",s:%1mm" ).arg(
mSize ) );
1498 ogrString.append(
')' );
1503 ogrString.append(
"PEN(" );
1504 ogrString.append(
"c:" );
1505 ogrString.append(
mColor.name() );
1506 ogrString.append(
",w:" );
1507 ogrString.append( QString::number(
mSize ) );
1508 ogrString.append(
"mm" );
1509 ogrString.append(
")" );
1517 QDomElement graphicElem = element.firstChildElement( QStringLiteral(
"Graphic" ) );
1518 if ( graphicElem.isNull() )
1521 QString name = QStringLiteral(
"square" );
1534 const double d = angleFunc.toDouble( &ok );
1544 double scaleFactor = 1.0;
1545 const QString uom = element.attribute( QStringLiteral(
"uom" ) );
1572 p->drawPath(
mPath );
1585 if ( hasDataDefinedSize )
1604 size *= mmMapUnitScaleFactor;
1611 const double halfSize =
size / 2.0;
1628 QColor pc =
mPen.color();
1629 QColor bc =
mBrush.color();
1649 QPointF off( offsetX, offsetY );
1678 t.translate( shift.x() + off.x(), shift.y() - off.y() );
1686 t.scale( halfSize, -halfSize );
1688 polygon = t.map( polygon );
1691 p.reserve( polygon.size() );
1692 for (
int i = 0; i < polygon.size(); i++ )
1697 if (
mBrush.style() != Qt::NoBrush )
1699 if (
mPen.style() != Qt::NoPen )
1704 shift += QPointF( off.x(), -off.y() );
1705 if (
mBrush.style() != Qt::NoBrush )
1707 if (
mPen.style() != Qt::NoPen )
1712 const QPointF pt1 = t.map( QPointF( 0, -halfSize ) );
1713 const QPointF pt2 = t.map( QPointF( 0, halfSize ) );
1715 if (
mPen.style() != Qt::NoPen )
1720 if (
mPen.style() != Qt::NoPen )
1722 const QPointF pt1 = t.map( QPointF( -halfSize, 0 ) );
1723 const QPointF pt2 = t.map( QPointF( halfSize, 0 ) );
1724 const QPointF pt3 = t.map( QPointF( 0, -halfSize ) );
1725 const QPointF pt4 = t.map( QPointF( 0, halfSize ) );
1733 if (
mPen.style() != Qt::NoPen )
1735 const QPointF pt1 = t.map( QPointF( -halfSize, -halfSize ) );
1736 const QPointF pt2 = t.map( QPointF( halfSize, halfSize ) );
1737 const QPointF pt3 = t.map( QPointF( halfSize, -halfSize ) );
1738 const QPointF pt4 = t.map( QPointF( -halfSize, halfSize ) );
1746 if (
mPen.style() != Qt::NoPen )
1748 const QPointF pt1 = t.map( QPointF( -halfSize, halfSize ) );
1749 const QPointF pt2 = t.map( QPointF( 0, 0 ) );
1750 const QPointF pt3 = t.map( QPointF( -halfSize, -halfSize ) );
1836 symbolBounds.adjust( -penWidth / 2.0, -penWidth / 2.0,
1837 penWidth / 2.0, penWidth / 2.0 );
1839 return symbolBounds;
1888 if ( props.contains( QStringLiteral(
"name" ) ) )
1889 name = props[QStringLiteral(
"name" )].toString();
1890 if ( props.contains( QStringLiteral(
"size" ) ) )
1891 size = props[QStringLiteral(
"size" )].toDouble();
1892 if ( props.contains( QStringLiteral(
"angle" ) ) )
1893 angle = props[QStringLiteral(
"angle" )].toDouble();
1894 if ( props.contains( QStringLiteral(
"scale_method" ) ) )
1898 if ( props.contains( QStringLiteral(
"offset" ) ) )
1900 if ( props.contains( QStringLiteral(
"offset_unit" ) ) )
1902 if ( props.contains( QStringLiteral(
"offset_map_unit_scale" ) ) )
1904 if ( props.contains( QStringLiteral(
"size_unit" ) ) )
1906 if ( props.contains( QStringLiteral(
"size_map_unit_scale" ) ) )
1908 if ( props.contains( QStringLiteral(
"horizontal_anchor_point" ) ) )
1912 if ( props.contains( QStringLiteral(
"vertical_anchor_point" ) ) )
1926 return QStringLiteral(
"FilledMarker" );
1951 map[QStringLiteral(
"size" )] = QString::number(
mSize );
1954 map[QStringLiteral(
"angle" )] = QString::number(
mAngle );
2010 attr.unite( mFill->usedAttributes( context ) );
2018 if ( mFill && mFill->hasDataDefinedProperties() )
2027 mFill->setColor(
c );
2032 return mFill ? mFill->color() :
mColor;
2039 || ( mFill && mFill->usesMapUnits() );
2046 mFill->setOutputUnit( unit );
2060 const double prevOpacity = mFill->opacity();
2061 mFill->setOpacity( mFill->opacity() * context.
opacity() );
2065 p->setBrush( Qt::red );
2069 p->setBrush( Qt::NoBrush );
2071 p->setPen( Qt::black );
2077 if ( !polygon.isEmpty() )
2079 mFill->renderPolygon( polygon,
nullptr, context.
feature(), context.
renderContext(), -1, useSelectedColor );
2083 const QPolygonF poly = path.toFillPolygon();
2084 mFill->renderPolygon( poly,
nullptr, context.
feature(), context.
renderContext(), -1, useSelectedColor );
2089 mFill->setOpacity( prevOpacity );
2104 mColor = QColor( 35, 35, 35 );
2118 if ( props.contains( QStringLiteral(
"name" ) ) )
2119 name = props[QStringLiteral(
"name" )].toString();
2120 if ( props.contains( QStringLiteral(
"size" ) ) )
2121 size = props[QStringLiteral(
"size" )].toDouble();
2122 if ( props.contains( QStringLiteral(
"angle" ) ) )
2123 angle = props[QStringLiteral(
"angle" )].toDouble();
2124 if ( props.contains( QStringLiteral(
"scale_method" ) ) )
2129 if ( props.contains( QStringLiteral(
"size_unit" ) ) )
2131 if ( props.contains( QStringLiteral(
"size_map_unit_scale" ) ) )
2133 if ( props.contains( QStringLiteral(
"fixedAspectRatio" ) ) )
2135 if ( props.contains( QStringLiteral(
"offset" ) ) )
2137 if ( props.contains( QStringLiteral(
"offset_unit" ) ) )
2139 if ( props.contains( QStringLiteral(
"offset_map_unit_scale" ) ) )
2141 if ( props.contains( QStringLiteral(
"fill" ) ) )
2146 else if ( props.contains( QStringLiteral(
"color" ) ) )
2150 if ( props.contains( QStringLiteral(
"outline" ) ) )
2155 else if ( props.contains( QStringLiteral(
"outline_color" ) ) )
2159 else if ( props.contains( QStringLiteral(
"line_color" ) ) )
2164 if ( props.contains( QStringLiteral(
"outline-width" ) ) )
2167 m->
setStrokeWidth( props[QStringLiteral(
"outline-width" )].toDouble() );
2169 else if ( props.contains( QStringLiteral(
"outline_width" ) ) )
2171 m->
setStrokeWidth( props[QStringLiteral(
"outline_width" )].toDouble() );
2173 else if ( props.contains( QStringLiteral(
"line_width" ) ) )
2175 m->
setStrokeWidth( props[QStringLiteral(
"line_width" )].toDouble() );
2178 if ( props.contains( QStringLiteral(
"outline_width_unit" ) ) )
2182 else if ( props.contains( QStringLiteral(
"line_width_unit" ) ) )
2186 if ( props.contains( QStringLiteral(
"outline_width_map_unit_scale" ) ) )
2189 if ( props.contains( QStringLiteral(
"horizontal_anchor_point" ) ) )
2193 if ( props.contains( QStringLiteral(
"vertical_anchor_point" ) ) )
2202 if ( props.contains( QStringLiteral(
"parameters" ) ) )
2204 const QVariantMap
parameters = props[QStringLiteral(
"parameters" )].toMap();
2213 const QVariantMap::iterator it =
properties.find( QStringLiteral(
"name" ) );
2232 QColor defaultFillColor, defaultStrokeColor;
2234 bool hasFillOpacityParam =
false, hasStrokeParam =
false, hasStrokeWidthParam =
false, hasStrokeOpacityParam =
false;
2235 bool hasDefaultFillColor =
false, hasDefaultFillOpacity =
false, hasDefaultStrokeColor =
false, hasDefaultStrokeWidth =
false, hasDefaultStrokeOpacity =
false;
2237 hasFillOpacityParam, hasDefaultFillOpacity, fillOpacity,
2238 hasStrokeParam, hasDefaultStrokeColor, defaultStrokeColor,
2239 hasStrokeWidthParam, hasDefaultStrokeWidth,
strokeWidth,
2240 hasStrokeOpacityParam, hasDefaultStrokeOpacity, strokeOpacity );
2242 const double newFillOpacity = hasFillOpacityParam ?
fillColor().alphaF() : 1.0;
2243 const double newStrokeOpacity = hasStrokeOpacityParam ?
strokeColor().alphaF() : 1.0;
2245 if ( hasDefaultFillColor )
2247 defaultFillColor.setAlphaF( newFillOpacity );
2250 if ( hasDefaultFillOpacity )
2253 c.setAlphaF( fillOpacity );
2256 if ( hasDefaultStrokeColor )
2258 defaultStrokeColor.setAlphaF( newStrokeOpacity );
2261 if ( hasDefaultStrokeWidth )
2265 if ( hasDefaultStrokeOpacity )
2268 c.setAlphaF( strokeOpacity );
2282 const double widthScaleFactor = 3.465;
2285 mDefaultAspectRatio = svgViewbox.isValid() ? svgViewbox.height() / svgViewbox.width() : 0.0;
2293 if ( aPreservedAspectRatio && !par )
2297 else if ( !aPreservedAspectRatio && par )
2312 return QStringLiteral(
"SvgMarker" );
2332 bool hasDataDefinedSize =
false;
2333 const double scaledWidth = calculateSize( context, hasDataDefinedSize );
2338 if (
static_cast< int >( width ) < 1 || 10000.0 < width )
2345 bool hasDataDefinedAspectRatio =
false;
2346 const double aspectRatio =
calculateAspectRatio( context, scaledWidth, hasDataDefinedAspectRatio );
2390 scaledHeight = svgViewbox.isValid() ? scaledWidth * svgViewbox.height() / svgViewbox.width() : scaledWidth;
2394 QPointF outputOffset;
2396 calculateOffsetAndRotation( context, scaledWidth, scaledHeight, outputOffset,
angle );
2398 p->translate( point + outputOffset );
2404 bool fitsInCache =
true;
2405 bool usePict =
true;
2412 if ( fitsInCache && img.width() > 1 )
2416 if ( useSelectedColor )
2424 QImage transparentImage = img.copy();
2426 if ( devicePixelRatio == 1 )
2428 p->drawImage( -transparentImage.width() / 2.0, -transparentImage.height() / 2.0, transparentImage );
2432 p->drawImage( QRectF( -transparentImage.width() / 2.0 / devicePixelRatio, -transparentImage.height() / 2.0 / devicePixelRatio,
2433 transparentImage.width() / devicePixelRatio, transparentImage.height() / devicePixelRatio
2434 ), transparentImage );
2439 if ( devicePixelRatio == 1 )
2441 p->drawImage( -img.width() / 2.0, -img.height() / 2.0, img );
2445 p->drawImage( QRectF( -img.width() / 2.0 / devicePixelRatio, -img.height() / 2.0 / devicePixelRatio,
2446 img.width() / devicePixelRatio, img.height() / devicePixelRatio ), img );
2452 if ( usePict || !fitsInCache )
2454 p->setOpacity( context.
opacity() );
2458 if ( pct.width() > 1 )
2461 _fixQPictureDPI( p );
2462 p->drawPicture( 0, 0, pct );
2470double QgsSvgMarkerSymbolLayer::calculateSize(
QgsSymbolRenderContext &context,
bool &hasDataDefinedSize )
const
2472 double scaledSize =
mSize;
2476 if ( hasDataDefinedSize )
2484 if ( hasDataDefinedSize )
2491 if ( hasDataDefinedSize && ok )
2496 scaledSize = std::sqrt( scaledSize );
2509 if ( !hasDataDefinedAspectRatio )
2519 const double defaultHeight =
mSize * scaledAspectRatio;
2520 scaledAspectRatio = defaultHeight / scaledSize;
2523 double scaledHeight = scaledSize * scaledAspectRatio;
2530 if ( hasDataDefinedAspectRatio && ok )
2535 scaledHeight = sqrt( scaledHeight );
2542 scaledAspectRatio = scaledHeight / scaledSize;
2544 return scaledAspectRatio;
2547void QgsSvgMarkerSymbolLayer::calculateOffsetAndRotation(
QgsSymbolRenderContext &context,
double scaledWidth,
double scaledHeight, QPointF &offset,
double &angle )
const
2552 markerOffset( context, scaledWidth, scaledHeight, offsetX, offsetY );
2553 offset = QPointF( offsetX, offsetY );
2563 if ( hasDataDefinedRotation )
2589 map[QStringLiteral(
"name" )] =
mPath;
2590 map[QStringLiteral(
"size" )] = QString::number(
mSize );
2593 map[QStringLiteral(
"fixedAspectRatio" )] = QString::number(
mFixedAspectRatio );
2594 map[QStringLiteral(
"angle" )] = QString::number(
mAngle );
2601 map[QStringLiteral(
"outline_width" )] = QString::number(
mStrokeWidth );
2676 QDomElement graphicElem = doc.createElement( QStringLiteral(
"se:Graphic" ) );
2677 element.appendChild( graphicElem );
2687 const double angle = props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toDouble( &ok );
2690 angleFunc = QStringLiteral(
"%1 + %2" ).arg( props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toString() ).arg(
mAngle );
2708 QDomElement graphicElem = element.firstChildElement( QStringLiteral(
"Graphic" ) );
2709 if ( graphicElem.isNull() )
2712 QString
path, mimeType;
2720 double scaleFactor = 1.0;
2721 const QString uom = element.attribute( QStringLiteral(
"uom" ) );
2725 if ( mimeType != QLatin1String(
"image/svg+xml" ) )
2733 const double d = angleFunc.toDouble( &ok );
2742 QString realPath {
path };
2743 QUrl svgUrl {
path };
2746 QUrlQuery queryString;
2748 if ( svgUrl.hasQuery() && svgUrl.hasFragment() )
2750 const QString queryPart {
path.mid(
path.indexOf(
'?' ) + 1 ) };
2751 queryString.setQuery( queryPart );
2755 if ( svgUrl.scheme().isEmpty() || svgUrl.isLocalFile() )
2757 svgUrl.setQuery( QString() );
2758 realPath = svgUrl.path();
2763 QMap<QString, QgsProperty> params;
2767 if ( queryString.hasQueryItem( QStringLiteral(
"fill" ) ) )
2769 const QColor
fillColor { queryString.queryItemValue( QStringLiteral(
"fill" ) ) };
2773 if ( queryString.hasQueryItem( QStringLiteral(
"fill-opacity" ) ) )
2775 const double alpha { queryString.queryItemValue( QStringLiteral(
"fill-opacity" ) ).toDouble( &ok ) };
2782 if ( queryString.hasQueryItem( QStringLiteral(
"outline" ) ) )
2784 const QColor
strokeColor { queryString.queryItemValue( QStringLiteral(
"outline" ) ) };
2788 if ( queryString.hasQueryItem( QStringLiteral(
"outline-opacity" ) ) )
2790 const double alpha { queryString.queryItemValue( QStringLiteral(
"outline-opacity" ) ).toDouble( &ok ) };
2797 if ( queryString.hasQueryItem( QStringLiteral(
"outline-width" ) ) )
2799 const int width { queryString.queryItemValue( QStringLiteral(
"outline-width" ) ).toInt( &ok )};
2806 if ( ! params.isEmpty() )
2825 if ( hasDataDefinedSize )
2831 if ( hasDataDefinedSize && ok )
2845 size *= mmMapUnitScaleFactor;
2859 const double offsetX =
offset.x();
2860 const double offsetY =
offset.y();
2862 QPointF outputOffset( offsetX, offsetY );
2912 QSvgRenderer r( svgContent );
2919 QSizeF outSize( r.defaultSize() );
2920 outSize.scale(
size,
size, Qt::KeepAspectRatio );
2926 p.translate( r.defaultSize().width() / 2.0, r.defaultSize().height() / 2.0 );
2928 p.translate( -r.defaultSize().width() / 2.0, -r.defaultSize().height() / 2.0 );
2930 pd.
setShift( shift + QPointF( outputOffset.x(), -outputOffset.y() ) );
2931 pd.
setOutputSize( QRectF( -outSize.width() / 2.0, -outSize.height() / 2.0, outSize.width(), outSize.height() ) );
2940 bool hasDataDefinedSize =
false;
2941 double scaledWidth = calculateSize( context, hasDataDefinedSize );
2943 bool hasDataDefinedAspectRatio =
false;
2944 const double aspectRatio =
calculateAspectRatio( context, scaledWidth, hasDataDefinedAspectRatio );
2951 if (
static_cast< int >( scaledWidth ) < 1 || 10000.0 < scaledWidth )
2956 QPointF outputOffset;
2958 calculateOffsetAndRotation( context, scaledWidth, scaledHeight, outputOffset,
angle );
2997 scaledHeight = svgViewbox.isValid() ? scaledWidth * svgViewbox.height() / svgViewbox.width() : scaledWidth;
3001 QTransform transform;
3003 transform.translate( point.x() + outputOffset.x(), point.y() + outputOffset.y() );
3006 transform.rotate(
angle );
3011 QRectF symbolBounds = transform.mapRect( QRectF( -scaledWidth / 2.0,
3012 -scaledHeight / 2.0,
3020 return symbolBounds;
3044 if ( props.contains( QStringLiteral(
"imageFile" ) ) )
3045 path = props[QStringLiteral(
"imageFile" )].toString();
3046 if ( props.contains( QStringLiteral(
"size" ) ) )
3047 size = props[QStringLiteral(
"size" )].toDouble();
3048 if ( props.contains( QStringLiteral(
"angle" ) ) )
3049 angle = props[QStringLiteral(
"angle" )].toDouble();
3050 if ( props.contains( QStringLiteral(
"scale_method" ) ) )
3053 std::unique_ptr< QgsRasterMarkerSymbolLayer > m = std::make_unique< QgsRasterMarkerSymbolLayer >(
path,
size,
angle,
scaleMethod );
3054 m->setCommonProperties( props );
3060 if (
properties.contains( QStringLiteral(
"alpha" ) ) )
3065 if (
properties.contains( QStringLiteral(
"size_unit" ) ) )
3067 if (
properties.contains( QStringLiteral(
"size_map_unit_scale" ) ) )
3069 if (
properties.contains( QStringLiteral(
"fixedAspectRatio" ) ) )
3072 if (
properties.contains( QStringLiteral(
"offset" ) ) )
3074 if (
properties.contains( QStringLiteral(
"offset_unit" ) ) )
3076 if (
properties.contains( QStringLiteral(
"offset_map_unit_scale" ) ) )
3079 if (
properties.contains( QStringLiteral(
"horizontal_anchor_point" ) ) )
3083 if (
properties.contains( QStringLiteral(
"vertical_anchor_point" ) ) )
3094 const QVariantMap::iterator it =
properties.find( QStringLiteral(
"name" ) );
3095 if ( it !=
properties.end() && it.value().type() == QVariant::String )
3113 if ( aPreservedAspectRatio && !par )
3117 else if ( !aPreservedAspectRatio && par )
3136 return QStringLiteral(
"RasterMarker" );
3152 if (
path.isEmpty() )
3156 double height = 0.0;
3158 bool hasDataDefinedSize =
false;
3159 const double scaledSize = calculateSize( context, hasDataDefinedSize );
3161 bool hasDataDefinedAspectRatio =
false;
3162 const double aspectRatio =
calculateAspectRatio( context, scaledSize, hasDataDefinedAspectRatio );
3164 QPointF outputOffset;
3171 if (
size.isEmpty() )
3174 width = ( scaledSize *
static_cast< double >(
size.width() ) ) / 100.0;
3175 height = ( scaledSize *
static_cast< double >(
size.height() ) ) / 100.0;
3178 if (
static_cast< int >( width ) < 1 || 10000.0 < width ||
static_cast< int >( height ) < 1 || 10000.0 < height )
3181 calculateOffsetAndRotation( context, width, height, outputOffset,
angle );
3191 if ( !
size.isNull() &&
size.isValid() &&
size.width() > 0 )
3193 height = width * (
static_cast< double >(
size.height() ) /
static_cast< double >(
size.width() ) );
3198 if (
static_cast< int >( width ) < 1 || 10000.0 < width )
3201 calculateOffsetAndRotation( context, scaledSize, scaledSize * ( height / width ), outputOffset,
angle );
3205 p->translate( point + outputOffset );
3220 if ( !img.isNull() )
3223 if ( useSelectedColor )
3228 p->drawImage( -img.width() / 2.0, -img.height() / 2.0, img );
3234 bool cached =
false;
3238double QgsRasterMarkerSymbolLayer::calculateSize(
QgsSymbolRenderContext &context,
bool &hasDataDefinedSize )
const
3240 double scaledSize =
mSize;
3244 if ( hasDataDefinedSize )
3252 if ( hasDataDefinedSize )
3259 if ( hasDataDefinedSize && ok )
3264 scaledSize = std::sqrt( scaledSize );
3277 if ( !hasDataDefinedAspectRatio )
3287 const double defaultHeight =
mSize * scaledAspectRatio;
3288 scaledAspectRatio = defaultHeight / scaledSize;
3291 double scaledHeight = scaledSize * scaledAspectRatio;
3298 if ( hasDataDefinedAspectRatio && ok )
3303 scaledHeight = sqrt( scaledHeight );
3310 scaledAspectRatio = scaledHeight / scaledSize;
3312 return scaledAspectRatio;
3315void QgsRasterMarkerSymbolLayer::calculateOffsetAndRotation(
QgsSymbolRenderContext &context,
double scaledWidth,
double scaledHeight, QPointF &offset,
double &angle )
const
3320 markerOffset( context, scaledWidth, scaledHeight, offsetX, offsetY );
3321 offset = QPointF( offsetX, offsetY );
3331 if ( hasDataDefinedRotation )
3352 map[QStringLiteral(
"imageFile" )] =
mPath;
3353 map[QStringLiteral(
"size" )] = QString::number(
mSize );
3356 map[QStringLiteral(
"fixedAspectRatio" )] = QString::number(
mFixedAspectRatio );
3357 map[QStringLiteral(
"angle" )] = QString::number(
mAngle );
3358 map[QStringLiteral(
"alpha" )] = QString::number(
mOpacity );
3370 std::unique_ptr< QgsRasterMarkerSymbolLayer > m = std::make_unique< QgsRasterMarkerSymbolLayer >(
mPath,
mSize,
mAngle );
3414 bool hasDataDefinedSize =
false;
3415 const double scaledSize = calculateSize( context, hasDataDefinedSize );
3417 bool hasDataDefinedAspectRatio =
false;
3418 const double aspectRatio =
calculateAspectRatio( context, scaledSize, hasDataDefinedAspectRatio );
3422 if (
static_cast< int >( scaledSize ) < 1 || 10000.0 < scaledSize )
3427 QPointF outputOffset;
3429 calculateOffsetAndRotation( context, scaledSize, scaledSize * ( height / width ), outputOffset,
angle );
3431 QTransform transform;
3434 transform.translate( point.x() + outputOffset.x(), point.y() + outputOffset.y() );
3437 transform.rotate(
angle );
3439 QRectF symbolBounds = transform.mapRect( QRectF( -width / 2.0,
3444 return symbolBounds;
3456 mOrigSize = pointSize;
3476 if ( props.contains( QStringLiteral(
"font" ) ) )
3477 fontFamily = props[QStringLiteral(
"font" )].toString();
3478 if ( props.contains( QStringLiteral(
"chr" ) ) && props[QStringLiteral(
"chr" )].toString().length() > 0 )
3479 string = props[QStringLiteral(
"chr" )].toString();
3480 if ( props.contains( QStringLiteral(
"size" ) ) )
3481 pointSize = props[QStringLiteral(
"size" )].toDouble();
3482 if ( props.contains( QStringLiteral(
"color" ) ) )
3484 if ( props.contains( QStringLiteral(
"angle" ) ) )
3485 angle = props[QStringLiteral(
"angle" )].toDouble();
3489 if ( props.contains( QStringLiteral(
"font_style" ) ) )
3490 m->
setFontStyle( props[QStringLiteral(
"font_style" )].toString() );
3491 if ( props.contains( QStringLiteral(
"outline_color" ) ) )
3493 if ( props.contains( QStringLiteral(
"outline_width" ) ) )
3494 m->
setStrokeWidth( props[QStringLiteral(
"outline_width" )].toDouble() );
3495 if ( props.contains( QStringLiteral(
"offset" ) ) )
3497 if ( props.contains( QStringLiteral(
"offset_unit" ) ) )
3499 if ( props.contains( QStringLiteral(
"offset_map_unit_scale" ) ) )
3501 if ( props.contains( QStringLiteral(
"size_unit" ) ) )
3503 if ( props.contains( QStringLiteral(
"size_map_unit_scale" ) ) )
3505 if ( props.contains( QStringLiteral(
"outline_width_unit" ) ) )
3507 if ( props.contains( QStringLiteral(
"outline_width_map_unit_scale" ) ) )
3509 if ( props.contains( QStringLiteral(
"joinstyle" ) ) )
3511 if ( props.contains( QStringLiteral(
"horizontal_anchor_point" ) ) )
3513 if ( props.contains( QStringLiteral(
"vertical_anchor_point" ) ) )
3523 return QStringLiteral(
"FontMarker" );
3528 QColor brushColor =
mColor;
3529 QColor penColor = mStrokeColor;
3531 brushColor.setAlphaF(
mColor.alphaF() * context.
opacity() );
3532 penColor.setAlphaF( mStrokeColor.alphaF() * context.
opacity() );
3534 mBrush = QBrush( brushColor );
3535 mPen = QPen( penColor );
3536 mPen.setJoinStyle( mPenJoinStyle );
3540 if ( !mFontStyle.isEmpty() )
3548 if ( mNonZeroFontSize && sizePixels > MAX_FONT_CHARACTER_SIZE_IN_PIXELS )
3553 mFontSizeScale = sizePixels / MAX_FONT_CHARACTER_SIZE_IN_PIXELS;
3554 sizePixels = MAX_FONT_CHARACTER_SIZE_IN_PIXELS;
3557 mFontSizeScale = 1.0;
3561 mFont.setPixelSize( std::max( 2,
static_cast< int >( std::round( sizePixels ) ) ) );
3562 mFontMetrics.reset(
new QFontMetrics( mFont ) );
3563 mChrWidth = mFontMetrics->horizontalAdvance( mString );
3564 mChrOffset = QPointF( mChrWidth / 2.0, -mFontMetrics->ascent() / 2.0 );
3571 if ( mUseCachedPath )
3573 QPointF chrOffset = mChrOffset;
3575 const QString charToRender = characterToRender( context, chrOffset, chrWidth );
3576 mCachedPath = QPainterPath();
3577 mCachedPath.addText( -chrOffset.x(), -chrOffset.y(), mFont, charToRender );
3586QString QgsFontMarkerSymbolLayer::characterToRender(
QgsSymbolRenderContext &context, QPointF &charOffset,
double &charWidth )
3588 charOffset = mChrOffset;
3589 QString stringToRender = mString;
3594 if ( stringToRender != mString )
3596 charWidth = mFontMetrics->horizontalAdvance( stringToRender );
3597 charOffset = QPointF( charWidth / 2.0, -mFontMetrics->ascent() / 2.0 );
3600 return stringToRender;
3605 bool &hasDataDefinedRotation,
3607 double &angle )
const
3612 markerOffset( context, scaledSize, scaledSize, offsetX, offsetY );
3613 offset = QPointF( offsetX, offsetY );
3629 if ( hasDataDefinedRotation )
3653 double scaledSize =
mSize;
3657 if ( hasDataDefinedSize )
3663 if ( hasDataDefinedSize && ok )
3668 scaledSize = std::sqrt( scaledSize );
3680 if ( !p || !mNonZeroFontSize )
3683 QTransform transform;
3686 QColor brushColor =
mColor;
3696 brushColor.setAlphaF( brushColor.alphaF() * context.
opacity() );
3698 mBrush.setColor( brushColor );
3700 QColor penColor = mStrokeColor;
3706 penColor.setAlphaF( penColor.alphaF() * context.
opacity() );
3730 p->setBrush( mBrush );
3733 mPen.setColor( penColor );
3734 mPen.setWidthF( penWidth );
3739 p->setPen( Qt::NoPen );
3757 mFontMetrics.reset(
new QFontMetrics( mFont ) );
3760 QPointF chrOffset = mChrOffset;
3762 const QString charToRender = characterToRender( context, chrOffset, chrWidth );
3764 const double sizeToRender = calculateSize( context );
3766 bool hasDataDefinedRotation =
false;
3769 calculateOffsetAndRotation( context, sizeToRender, hasDataDefinedRotation,
offset,
angle );
3771 p->translate( point.x() +
offset.x(), point.y() +
offset.y() );
3774 transform.rotate(
angle );
3778 const double s = sizeToRender / mOrigSize;
3779 transform.scale( s, s );
3783 transform.scale( mFontSizeScale, mFontSizeScale );
3785 if ( mUseCachedPath )
3787 p->drawPath( transform.map( mCachedPath ) );
3792 path.addText( -chrOffset.x(), -chrOffset.y(), mFont, charToRender );
3793 p->drawPath( transform.map( path ) );
3800 props[QStringLiteral(
"font" )] = mFontFamily;
3801 props[QStringLiteral(
"font_style" )] = mFontStyle;
3802 props[QStringLiteral(
"chr" )] = mString;
3803 props[QStringLiteral(
"size" )] = QString::number(
mSize );
3808 props[QStringLiteral(
"outline_width" )] = QString::number( mStrokeWidth );
3812 props[QStringLiteral(
"angle" )] = QString::number(
mAngle );
3845 QDomElement graphicElem = doc.createElement( QStringLiteral(
"se:Graphic" ) );
3846 element.appendChild( graphicElem );
3848 const QString fontPath = QStringLiteral(
"ttf://%1" ).arg( mFontFamily );
3849 int markIndex = !mString.isEmpty() ? mString.at( 0 ).unicode() : 0;
3856 const double angle = props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toDouble( &ok );
3859 angleFunc = QStringLiteral(
"%1 + %2" ).arg( props.value( QStringLiteral(
"angle" ), QStringLiteral(
"0" ) ).toString() ).arg(
mAngle );
3882 mStrokeWidthUnit = unit;
3887 QPointF chrOffset = mChrOffset;
3888 double chrWidth = mChrWidth;
3890 ( void )characterToRender( context, chrOffset, chrWidth );
3892 if ( !mFontMetrics )
3893 mFontMetrics.reset(
new QFontMetrics( mFont ) );
3895 double scaledSize = calculateSize( context );
3898 chrWidth *= scaledSize / mOrigSize;
3900 chrWidth *= mFontSizeScale;
3902 bool hasDataDefinedRotation =
false;
3905 calculateOffsetAndRotation( context, scaledSize, hasDataDefinedRotation,
offset,
angle );
3908 QTransform transform;
3911 transform.translate( point.x() +
offset.x(), point.y() +
offset.y() );
3914 transform.rotate(
angle );
3916 QRectF symbolBounds = transform.mapRect( QRectF( -chrWidth / 2.0,
3920 return symbolBounds;
3927 QDomElement graphicElem = element.firstChildElement( QStringLiteral(
"Graphic" ) );
3928 if ( graphicElem.isNull() )
3931 QString name, format;
3939 if ( !name.startsWith( QLatin1String(
"ttf://" ) ) || format != QLatin1String(
"ttf" ) )
3949 const double d = angleFunc.toDouble( &ok );
3957 double scaleFactor = 1.0;
3958 const QString uom = element.attribute( QStringLiteral(
"uom" ) );
3979 context.
pushMessage( QObject::tr(
"Font “%1” not available on system" ).arg( processedFamily ) );
3985 QMap<QString, QgsProperty>::iterator it =
mParameters.begin();
3997 QMap<QString, QgsProperty>::const_iterator it =
mParameters.constBegin();
4000 attrs.unite( it.value().referencedFields( context.
expressionContext(),
true ) );
4024 if (
properties.contains( QStringLiteral(
"imageFile" ) ) )
4026 if (
properties.contains( QStringLiteral(
"size" ) ) )
4028 if (
properties.contains( QStringLiteral(
"angle" ) ) )
4031 std::unique_ptr< QgsAnimatedMarkerSymbolLayer > m = std::make_unique< QgsAnimatedMarkerSymbolLayer >(
path,
size,
angle );
4032 m->setFrameRate(
properties.value( QStringLiteral(
"frameRate" ), QStringLiteral(
"10" ) ).toDouble() );
4040 return QStringLiteral(
"AnimatedMarker" );
4046 res.insert( QStringLiteral(
"frameRate" ), mFrameRateFps );
4052 std::unique_ptr< QgsAnimatedMarkerSymbolLayer > m = std::make_unique< QgsAnimatedMarkerSymbolLayer >(
mPath,
mSize,
mAngle );
4053 m->setFrameRate( mFrameRateFps );
4062 mPreparedPaths.clear();
4070 mStaticPath =
false;
4076 if ( !mStaticPath && !mPreparedPaths.contains(
path ) )
4079 mPreparedPaths.insert(
path );
4082 const long long mapFrameNumber = context.
currentFrame();
4084 const double markerAnimationDuration = totalFrameCount / mFrameRateFps;
4086 double animationTimeSeconds = 0;
4087 if ( mapFrameNumber >= 0 && context.
frameRate() > 0 )
4090 animationTimeSeconds = mapFrameNumber / context.
frameRate();
4095 animationTimeSeconds = QDateTime::currentMSecsSinceEpoch() / 1000.0;
4098 const double markerAnimationProgressSeconds = std::fmod( animationTimeSeconds, markerAnimationDuration );
4099 const int movieFrame =
static_cast< int >( std::floor( markerAnimationProgressSeconds * mFrameRateFps ) );
4101 bool cached =
false;
@ DynamicRotation
Rotation of symbol may be changed during rendering and symbol should not be cached.
ScaleMethod
Scale methods.
@ ScaleDiameter
Calculate scale by the diameter.
@ ScaleArea
Calculate scale by the area.
MarkerShape
Marker shapes.
@ EquilateralTriangle
Equilateral triangle.
@ SemiCircle
Semi circle (top half)
@ QuarterCircle
Quarter circle (top left quarter)
@ LeftHalfTriangle
Left half of triangle.
@ ArrowHead
Right facing arrow head (unfilled, lines only)
@ ParallelogramRight
Parallelogram that slants right (since QGIS 3.28)
@ AsteriskFill
A filled asterisk shape (since QGIS 3.18)
@ Octagon
Octagon (since QGIS 3.18)
@ HalfArc
A line-only half arc (since QGIS 3.20)
@ QuarterSquare
Quarter square (top left quarter)
@ Cross2
Rotated cross (lines only), 'x' shape.
@ Trapezoid
Trapezoid (since QGIS 3.28)
@ ArrowHeadFilled
Right facing filled arrow head.
@ Shield
A shape consisting of a triangle attached to a rectangle (since QGIS 3.28)
@ HalfSquare
Half square (left half)
@ CrossFill
Solid filled cross.
@ Decagon
Decagon (since QGIS 3.28)
@ RoundedSquare
A square with rounded corners (since QGIS 3.28)
@ RightHalfTriangle
Right half of triangle.
@ ThirdCircle
One third circle (top left third)
@ ThirdArc
A line-only one third arc (since QGIS 3.20)
@ SquareWithCorners
A square with diagonal corners (since QGIS 3.18)
@ QuarterArc
A line-only one quarter arc (since QGIS 3.20)
@ DiamondStar
A 4-sided star (since QGIS 3.28)
@ Cross
Cross (lines only)
@ ParallelogramLeft
Parallelogram that slants left (since QGIS 3.28)
@ Heart
Heart (since QGIS 3.28)
@ DiagonalHalfSquare
Diagonal half square (bottom left half)
RenderUnit
Rendering size units.
@ Percentage
Percentage of another measurement (e.g., canvas size, feature size)
@ Millimeters
Millimeters.
@ Unknown
Mixed or unknown units.
@ MetersInMapUnits
Meters value as Map units.
@ RenderingSubSymbol
Set whenever a sub-symbol of a parent symbol is currently being rendered. Can be used during symbol a...
@ RenderSymbolPreview
The render is for a symbol preview only and map based properties may not be available,...
@ RenderBlocking
Render and load remote sources in the same thread to ensure rendering remote sources (svg and images)...
QColor valueAsColor(int key, const QgsExpressionContext &context, const QColor &defaultColor=QColor(), bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a color.
double valueAsDouble(int key, const QgsExpressionContext &context, double defaultValue=0.0, bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a double.
QString valueAsString(int key, const QgsExpressionContext &context, const QString &defaultString=QString(), bool *ok=nullptr) const
Calculates the current value of the property with the specified key and interprets it as a string.
Animated marker symbol layer class.
QVariantMap properties() const override
Should be reimplemented by subclasses to return a string map that contains the configuration informat...
QgsAnimatedMarkerSymbolLayer * clone() const override
Shall be reimplemented by subclasses to create a deep copy of the instance.
~QgsAnimatedMarkerSymbolLayer() override
void startRender(QgsSymbolRenderContext &context) override
Called before a set of rendering operations commences on the supplied render context.
QImage fetchImage(QgsRenderContext &context, const QString &path, QSize size, bool preserveAspectRatio, double opacity) const override
Fetches the image to render.
QgsAnimatedMarkerSymbolLayer(const QString &path=QString(), double size=DEFAULT_RASTERMARKER_SIZE, double angle=DEFAULT_RASTERMARKER_ANGLE)
Constructor for animated marker symbol layer using the specified source image path.
static QgsSymbolLayer * create(const QVariantMap &properties=QVariantMap())
Creates an animated marker symbol layer from a string map of properties.
QString layerType() const override
Returns a string that represents this layer type.
static QgsImageCache * imageCache()
Returns the application's image cache, used for caching resampled versions of raster images.
static QgsSvgCache * svgCache()
Returns the application's SVG cache, used for caching SVG images and handling parameter replacement w...
static QgsFontManager * fontManager()
Returns the application font manager, which manages available fonts and font installation for the QGI...
Exports QGIS layers to the DXF format.
void writeFilledCircle(const QString &layer, const QColor &color, const QgsPoint &pt, double radius)
Write filled circle (as hatch)
void writeCircle(const QString &layer, const QColor &color, const QgsPoint &pt, double radius, const QString &lineStyleName, double width)
Write circle (as polyline)
static double mapUnitScaleFactor(double scale, Qgis::RenderUnit symbolUnits, Qgis::DistanceUnit mapUnits, double mapUnitsPerPixel=1.0)
Returns scale factor for conversion to map units.
void writeLine(const QgsPoint &pt1, const QgsPoint &pt2, const QString &layer, const QString &lineStyleName, const QColor &color, double width=-1)
Write line (as a polyline)
void writePolygon(const QgsRingSequence &polygon, const QString &layer, const QString &hatchPattern, const QColor &color)
Draw dxf filled polygon (HATCH)
Qgis::DistanceUnit mapUnits() const
Retrieve map units.
double symbologyScale() const
Returns the reference scale for output.
void clipValueToMapUnitScale(double &value, const QgsMapUnitScale &scale, double pixelToMMFactor) const
Clips value to scale minimum/maximum.
void writePolyline(const QgsPointSequence &line, const QString &layer, const QString &lineStyleName, const QColor &color, double width=-1)
Draw dxf primitives (LWPOLYLINE)
A paint device for drawing into dxf files.
void setShift(QPointF shift)
void setLayer(const QString &layer)
void setOutputSize(const QRectF &r)
void setDrawingSize(QSizeF size)
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
bool hasGeometry() const
Returns true if the feature has an associated geometry.
A fill symbol type, for rendering Polygon and MultiPolygon geometries.
static QgsFillSymbol * createSimple(const QVariantMap &properties)
Create a fill symbol with one symbol layer: SimpleFill with specified properties.
Filled marker symbol layer, consisting of a shape which is rendered using a QgsFillSymbol.
QSet< QString > usedAttributes(const QgsRenderContext &context) const override
Returns the set of attributes referenced by the layer.
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
QVariantMap properties() const override
Should be reimplemented by subclasses to return a string map that contains the configuration informat...
QColor color() const override
Returns the "representative" color of the symbol layer.
QgsFilledMarkerSymbolLayer * clone() const override
Shall be reimplemented by subclasses to create a deep copy of the instance.
bool hasDataDefinedProperties() const override
Returns true if the symbol layer (or any of its sub-symbols) contains data defined properties.
void startRender(QgsSymbolRenderContext &context) override
Called before a set of rendering operations commences on the supplied render context.
bool usesMapUnits() const override
Returns true if the symbol layer has any components which use map unit based sizes.
QgsSymbol * subSymbol() override
Returns the symbol's sub symbol, if present.
~QgsFilledMarkerSymbolLayer() override
bool setSubSymbol(QgsSymbol *symbol) override
Sets layer's subsymbol. takes ownership of the passed symbol.
void stopRender(QgsSymbolRenderContext &context) override
Called after a set of rendering operations has finished on the supplied render context.
double estimateMaxBleed(const QgsRenderContext &context) const override
Returns the estimated maximum distance which the layer style will bleed outside the drawn shape when ...
void setColor(const QColor &c) override
Sets the "representative" color for the symbol layer.
QString layerType() const override
Returns a string that represents this layer type.
static QgsSymbolLayer * create(const QVariantMap &properties=QVariantMap())
Creates a new QgsFilledMarkerSymbolLayer.
QgsFilledMarkerSymbolLayer(Qgis::MarkerShape shape=Qgis::MarkerShape::Circle, double size=DEFAULT_SIMPLEMARKER_SIZE, double angle=DEFAULT_SIMPLEMARKER_ANGLE, Qgis::ScaleMethod scaleMethod=DEFAULT_SCALE_METHOD)
Constructor for QgsFilledMarkerSymbolLayer.
QString processFontFamilyName(const QString &name) const
Processes a font family name, applying any matching fontFamilyReplacements() to the name.
void setStrokeWidthUnit(Qgis::RenderUnit unit)
Sets the stroke width unit.
~QgsFontMarkerSymbolLayer() override
void setStrokeColor(const QColor &color) override
Sets the stroke color for the symbol layer.
QgsFontMarkerSymbolLayer(const QString &fontFamily=DEFAULT_FONTMARKER_FONT, QString chr=DEFAULT_FONTMARKER_CHR, double pointSize=DEFAULT_FONTMARKER_SIZE, const QColor &color=DEFAULT_FONTMARKER_COLOR, double angle=DEFAULT_FONTMARKER_ANGLE)
Constructs a font marker symbol layer.
void setStrokeWidthMapUnitScale(const QgsMapUnitScale &scale)
Sets the stroke width map unit scale.
double strokeWidth() const
Returns the marker's stroke width.
QVariantMap properties() const override
Should be reimplemented by subclasses to return a string map that contains the configuration informat...
void renderPoint(QPointF point, QgsSymbolRenderContext &context) override
Renders a marker at the specified point.
void setFontStyle(const QString &style)
Sets the font style for the font which will be used to render the point.
bool usesMapUnits() const override
Returns true if the symbol layer has any components which use map unit based sizes.
QString fontStyle() const
Returns the font style for the associated font which will be used to render the point.
QString fontFamily() const
Returns the font family name for the associated font which will be used to render the point.
QRectF bounds(QPointF point, QgsSymbolRenderContext &context) override
Returns the approximate bounding box of the marker symbol layer, taking into account any data defined...
void writeSldMarker(QDomDocument &doc, QDomElement &element, const QVariantMap &props) const override
Writes the symbol layer definition as a SLD XML element.
void setStrokeWidth(double width)
Set's the marker's stroke width.
static void resolveFonts(const QVariantMap &properties, const QgsReadWriteContext &context)
Resolves fonts from a properties map, raising warnings in the specified context if the required fonts...
void startRender(QgsSymbolRenderContext &context) override
Called before a set of rendering operations commences on the supplied render context.
void setPenJoinStyle(Qt::PenJoinStyle style)
Sets the stroke join style.
static QgsSymbolLayer * createFromSld(QDomElement &element)
Creates a new QgsFontMarkerSymbolLayer from an SLD XML element.
QgsFontMarkerSymbolLayer * clone() const override
Shall be reimplemented by subclasses to create a deep copy of the instance.
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
void stopRender(QgsSymbolRenderContext &context) override
Called after a set of rendering operations has finished on the supplied render context.
QString layerType() const override
Returns a string that represents this layer type.
static QgsSymbolLayer * create(const QVariantMap &properties=QVariantMap())
Creates a new QgsFontMarkerSymbolLayer from a property map (see properties())
static QString translateNamedStyle(const QString &namedStyle)
Returns the localized named style of a font, if such a translation is available.
static QFont createFont(const QString &family, int pointSize=-1, int weight=-1, bool italic=false)
Creates a font with the specified family.
static bool fontFamilyMatchOnSystem(const QString &family, QString *chosen=nullptr, bool *match=nullptr)
Check whether font family is on system.
static bool updateFontViaStyle(QFont &f, const QString &fontstyle, bool fallback=false)
Updates font with named style and retain all font properties.
static void setFontFamily(QFont &font, const QString &family)
Sets the family for a font object.
QSize originalSize(const QString &path, bool blocking=false) const
Returns the original size (in pixels) of the image at the specified path.
int totalFrameCount(const QString &path, bool blocking=false)
Returns the total frame count of the image at the specified path.
QImage pathAsImage(const QString &path, const QSize size, const bool keepAspectRatio, const double opacity, bool &fitsInCache, bool blocking=false, double targetDpi=96, int frameNumber=-1, bool *isMissing=nullptr)
Returns the specified path rendered as an image.
void prepareAnimation(const QString &path)
Prepares for optimized retrieval of frames for the animation at the given path.
static void overlayColor(QImage &image, const QColor &color)
Overlays a color onto an image.
Perform transforms between map coordinates and device coordinates.
double mapUnitsPerPixel() const
Returns the current map units per pixel.
double mapRotation() const
Returns the current map rotation in degrees (clockwise).
Struct for storing maximum and minimum scales for measurements in map units.
Abstract base class for marker symbol layers.
Qgis::RenderUnit mOffsetUnit
Offset units.
QPointF offset() const
Returns the marker's offset, which is the horizontal and vertical displacement which the rendered mar...
double mLineAngle
Line rotation angle (see setLineAngle() for details)
HorizontalAnchorPoint
Symbol horizontal anchor points.
void setOffsetUnit(Qgis::RenderUnit unit)
Sets the units for the symbol's offset.
void setAngle(double angle)
Sets the rotation angle for the marker.
Qgis::ScaleMethod scaleMethod() const
Returns the method to use for scaling the marker's size.
Qgis::RenderUnit outputUnit() const override
Returns the units to use for sizes and widths within the symbol layer.
void setVerticalAnchorPoint(VerticalAnchorPoint v)
Sets the vertical anchor point for positioning the symbol.
QPointF mOffset
Marker offset.
void setHorizontalAnchorPoint(HorizontalAnchorPoint h)
Sets the horizontal anchor point for positioning the symbol.
QgsMapUnitScale mapUnitScale() const override
void setOffset(QPointF offset)
Sets the marker's offset, which is the horizontal and vertical displacement which the rendered marker...
void setSizeMapUnitScale(const QgsMapUnitScale &scale)
Sets the map unit scale for the symbol's size.
double size() const
Returns the symbol size.
QgsMapUnitScale mOffsetMapUnitScale
Offset map unit scale.
HorizontalAnchorPoint mHorizontalAnchorPoint
Horizontal anchor point.
static QPointF _rotatedOffset(QPointF offset, double angle)
Adjusts a marker offset to account for rotation.
Qgis::ScaleMethod mScaleMethod
Marker size scaling method.
QgsMapUnitScale mSizeMapUnitScale
Marker size map unit scale.
Qgis::RenderUnit mSizeUnit
Marker size unit.
void setOutputUnit(Qgis::RenderUnit unit) override
Sets the units to use for sizes and widths within the symbol layer.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units for the symbol's size.
VerticalAnchorPoint
Symbol vertical anchor points.
void markerOffset(QgsSymbolRenderContext &context, double &offsetX, double &offsetY) const
Calculates the required marker offset, including both the symbol offset and any displacement required...
VerticalAnchorPoint mVerticalAnchorPoint
Vertical anchor point.
void setOffsetMapUnitScale(const QgsMapUnitScale &scale)
Sets the map unit scale for the symbol's offset.