30 mStrokeWidth = strokeWidth;
40 mStrokeColoring = strokeColoring;
45 return mStrokeColoring;
50 mStrokeWidthUnit = strokeWidthUnit;
55 return mStrokeWidthUnit;
60 QPainter *painter = context.
painter();
64 QPointF dir = p2 - p1;
65 double length = sqrt( pow( dir.x(), 2 ) + pow( dir.y(), 2 ) );
66 QPointF diru = dir / length;
67 QPointF orthu = QPointF( -diru.y(), diru.x() );
69 QList<double> breakValues;
70 QList<QColor> breakColors;
71 QList<QLinearGradient> gradients;
73 mStrokeColoring.graduatedColors( valueColor1, valueColor2, breakValues, breakColors, gradients );
76 if ( gradients.isEmpty() && !breakValues.empty() && !breakColors.isEmpty() )
78 Q_ASSERT( breakColors.count() == breakValues.count() );
79 for (
int i = 0; i < breakValues.count(); ++i )
81 const bool widthIsInverted { valueWidth1 > valueWidth2 };
82 const double value = breakValues.at( i );
83 const double width = context.
convertToPainterUnits( mStrokeWidth.strokeWidth( widthIsInverted ? mStrokeWidth.maximumValue() - value : value ), mStrokeWidthUnit );
84 QPen pen( mSelected ? selectedColor : breakColors.at( i ) );
85 pen.setWidthF( width );
86 pen.setCapStyle( Qt::PenCapStyle::RoundCap );
87 painter->setPen( pen );
88 const QPointF point = p1 + dir * ( value - valueColor1 ) / ( valueColor2 - valueColor1 );
89 painter->drawPoint( point );
94 double width1 = mStrokeWidth.strokeWidth( valueWidth1 );
95 double width2 = mStrokeWidth.strokeWidth( valueWidth2 );
97 if ( !std::isnan( width1 ) && !std::isnan( width2 ) )
100 QBrush brush( Qt::SolidPattern );
103 startAngle = ( acos( -orthu.x() ) / M_PI ) * 180;
105 startAngle = 360 - startAngle;
107 bool outOfRange1 = std::isnan( width1 );
108 bool outOfRange2 = std::isnan( width2 );
113 QRectF capBox1( p1.x() - width1 / 2, p1.y() - width1 / 2, width1, width1 );
114 brush.setColor( mSelected ? selectedColor : mStrokeColoring.color( valueColor1 ) );
115 painter->setBrush( brush );
116 pen.setBrush( brush );
117 painter->setPen( pen );
118 painter->drawPie( capBox1, ( startAngle - 1 ) * 16, 182 * 16 );
124 QRectF capBox2( p2.x() - width2 / 2, p2.y() - width2 / 2, width2, width2 );
125 brush.setColor( mSelected ? selectedColor : mStrokeColoring.color( valueColor2 ) );
126 pen.setBrush( brush );
127 painter->setBrush( brush );
128 painter->setPen( pen );
129 painter->drawPie( capBox2, ( startAngle + 179 ) * 16, 182 * 16 );
132 if ( ( gradients.isEmpty() && breakValues.empty() && breakColors.count() == 1 ) || mSelected )
134 double startAdjusting = 0;
136 adjustLine( valueColor1, valueColor1, valueColor2, width1, startAdjusting );
139 double endAdjusting = 0;
141 adjustLine( valueColor2, valueColor1, valueColor2, width2, endAdjusting );
143 QPointF pointStartAdjusted = p1 + dir * startAdjusting;
144 QPointF pointEndAdjusted = p2 - dir * endAdjusting;
147 double semiWidth1 = width1 / 2;
148 double semiWidth2 = width2 / 2;
150 varLine.append( pointStartAdjusted + orthu * semiWidth1 );
151 varLine.append( pointEndAdjusted + orthu * semiWidth2 );
152 varLine.append( pointEndAdjusted - orthu * semiWidth2 );
153 varLine.append( pointStartAdjusted - orthu * semiWidth1 );
155 QBrush brush( Qt::SolidPattern );
156 brush.setColor( mSelected ? selectedColor : breakColors.first() );
157 painter->setBrush( brush );
158 painter->setPen( pen );
161 pen.setBrush( brush );
163 painter->setPen( pen );
165 painter->drawPolygon( varLine );
168 else if ( !gradients.isEmpty() && !breakValues.isEmpty() && !breakColors.isEmpty() )
170 Q_ASSERT( breakColors.count() == breakValues.count() );
171 Q_ASSERT( breakColors.count() == gradients.count() + 1 );
172 double widthColorVariationValueRatio = ( valueWidth2 - valueWidth1 ) / ( valueColor2 - valueColor1 );
174 for (
int i = 0; i < gradients.count(); ++i )
176 double firstValue = breakValues.at( i );
177 double secondValue = breakValues.at( i + 1 );
178 double w1 = mStrokeWidth.strokeWidth( widthColorVariationValueRatio * ( firstValue - valueColor1 ) + valueWidth1 );
179 double w2 = mStrokeWidth.strokeWidth( widthColorVariationValueRatio * ( secondValue - valueColor1 ) + valueWidth1 );
181 if ( std::isnan( w1 ) && std::isnan( w2 ) )
184 double firstAdjusting = 0;
185 if ( std::isnan( w1 ) )
186 adjustLine( firstValue, valueColor1, valueColor2, w1, firstAdjusting );
189 double secondAdjusting = 0;
190 if ( std::isnan( w2 ) )
191 adjustLine( secondValue, valueColor1, valueColor2, w2, secondAdjusting );
196 QPointF pointStart = p1 + dir * ( firstValue - valueColor1 ) / ( valueColor2 - valueColor1 );
197 QPointF pointEnd = p1 + dir * ( secondValue - valueColor1 ) / ( valueColor2 - valueColor1 );
199 QPointF pointStartAdjusted = pointStart + dir * firstAdjusting;
200 QPointF pointEndAdjusted = pointEnd - dir * secondAdjusting;
206 varLine.append( pointStartAdjusted + orthu * sw1 );
207 varLine.append( pointEndAdjusted + orthu * sw2 );
208 varLine.append( pointEndAdjusted - orthu * sw2 );
209 varLine.append( pointStartAdjusted - orthu * sw1 );
211 QLinearGradient gradient = gradients.at( i );
212 gradient.setStart( pointStart );
213 gradient.setFinalStop( pointEnd );
214 QBrush brush( gradient );
215 painter->setBrush( brush );
218 pen.setBrush( brush );
220 painter->setPen( pen );
222 painter->drawPolygon( varLine );
237 if ( value1 > value2 )
239 std::swap( value1, value2 );
240 std::swap( point1, point2 );
256 if ( valueColor1 > valueColor2 )
258 std::swap( valueColor1, valueColor2 );
259 std::swap( valueWidth1, valueWidth2 );
260 std::swap( point1, point2 );
271 mSelected = selected;
274void QgsInterpolatedLineRenderer::adjustLine(
const double value,
const double value1,
const double value2,
double &width,
double &adjusting )
const
278 adjusting = fabs( ( value - mStrokeWidth.
maximumValue() ) / ( value2 - value1 ) );
283 adjusting = fabs( ( value - mStrokeWidth.minimumValue() ) / ( value2 - value1 ) );
284 width = mStrokeWidth.minimumWidth();
290 return mMinimumValue;
296 mNeedUpdateFormula =
true;
301 return mMaximumValue;
307 mNeedUpdateFormula =
true;
312 return mMinimumWidth;
318 mNeedUpdateFormula =
true;
323 return mMaximumWidth;
329 mNeedUpdateFormula =
true;
334 if ( mIsWidthVariable )
336 if ( mNeedUpdateFormula )
337 updateLinearFormula();
339 if ( mUseAbsoluteValue )
340 value = std::fabs( value );
342 if ( value > mMaximumValue )
344 if ( mIgnoreOutOfRange )
345 return std::numeric_limits<double>::quiet_NaN();
347 return mMaximumWidth;
350 if ( value < mMinimumValue )
352 if ( mIgnoreOutOfRange )
353 return std::numeric_limits<double>::quiet_NaN();
355 return mMinimumWidth;
358 return ( value - mMinimumValue ) * mLinearCoef + mMinimumWidth;
366 QDomElement elem = doc.createElement( QStringLiteral(
"mesh-stroke-width" ) );
368 elem.setAttribute( QStringLiteral(
"width-varying" ), mIsWidthVariable ? 1 : 0 );
369 elem.setAttribute( QStringLiteral(
"fixed-width" ), mFixedWidth );
370 elem.setAttribute( QStringLiteral(
"minimum-value" ), mMinimumValue );
371 elem.setAttribute( QStringLiteral(
"maximum-value" ), mMaximumValue );
372 elem.setAttribute( QStringLiteral(
"minimum-width" ), mMinimumWidth );
373 elem.setAttribute( QStringLiteral(
"maximum-width" ), mMaximumWidth );
374 elem.setAttribute( QStringLiteral(
"ignore-out-of-range" ), mIgnoreOutOfRange ? 1 : 0 );
375 elem.setAttribute( QStringLiteral(
"use-absolute-value" ), mUseAbsoluteValue ? 1 : 0 );
382 mIsWidthVariable = elem.attribute( QStringLiteral(
"width-varying" ) ).toInt();
383 mFixedWidth = elem.attribute( QStringLiteral(
"fixed-width" ) ).toDouble();
384 mMinimumValue = elem.attribute( QStringLiteral(
"minimum-value" ) ).toDouble();
385 mMaximumValue = elem.attribute( QStringLiteral(
"maximum-value" ) ).toDouble();
386 mMinimumWidth = elem.attribute( QStringLiteral(
"minimum-width" ) ).toDouble();
387 mMaximumWidth = elem.attribute( QStringLiteral(
"maximum-width" ) ).toDouble();
388 mIgnoreOutOfRange = elem.attribute( QStringLiteral(
"ignore-out-of-range" ) ).toInt();
389 mUseAbsoluteValue = elem.attribute( QStringLiteral(
"use-absolute-value" ) ).toInt();
394 return mUseAbsoluteValue;
409 return mIgnoreOutOfRange;
419 return mIsWidthVariable;
424 mIsWidthVariable = isWidthVarying;
429 mFixedWidth = fixedWidth;
432void QgsInterpolatedLineWidth::updateLinearFormula()
const
435 mLinearCoef = ( mMaximumWidth - mMinimumWidth ) / ( mMaximumValue - mMinimumValue ) ;
438 mNeedUpdateFormula =
false;
443 mColorRampShader.setMinimumValue( std::numeric_limits<double>::quiet_NaN() );
444 mColorRampShader.setMaximumValue( std::numeric_limits<double>::quiet_NaN() );
456 mColorRampShader.setMinimumValue( std::numeric_limits<double>::quiet_NaN() );
457 mColorRampShader.setMaximumValue( std::numeric_limits<double>::quiet_NaN() );
463 if ( ( mColorRampShader.sourceColorRamp() ) )
471 mSingleColor =
color;
476 QgsColorRamp *lSourceColorRamp = mColorRampShader.sourceColorRamp();
477 if ( mColoringMethod ==
ColorRamp && lSourceColorRamp )
479 if ( mColorRampShader.isEmpty() )
480 return lSourceColorRamp->
color( 0 );
483 if ( mColorRampShader.shade( magnitude, &r, &g, &b, &a ) )
484 return QColor( r, g, b, a );
486 return QColor( 0, 0, 0, 0 );
496 return mColoringMethod;
501 return mColorRampShader;
511 QDomElement elem = doc.createElement( QStringLiteral(
"mesh-stroke-color" ) );
514 elem.setAttribute( QStringLiteral(
"coloring-method" ), mColoringMethod );
515 elem.appendChild( mColorRampShader.writeXml( doc ) );
522 QDomElement shaderElem = elem.firstChildElement( QStringLiteral(
"colorrampshader" ) );
523 mColorRampShader.
readXml( shaderElem );
527 elem.attribute( QStringLiteral(
"coloring-method" ) ).toInt() );
537 breakColors.append( mSingleColor );
541 switch ( mColorRampShader.colorRampType() )
544 graduatedColorsInterpolated( value1, value2, breakValues, breakColors, gradients );
547 graduatedColorsDiscrete( value1, value2, breakValues, breakColors, gradients );
550 graduatedColorsExact( value1, value2, breakValues, breakColors, gradients );
561QLinearGradient QgsInterpolatedLineColor::makeSimpleLinearGradient(
const QColor &color1,
const QColor &color2 )
const
563 QLinearGradient gradient;
564 gradient.setColorAt( 0, color1 );
565 gradient.setColorAt( 1, color2 );
570int QgsInterpolatedLineColor::itemColorIndexInf(
double value )
const
572 QList<QgsColorRampShader::ColorRampItem> itemList = mColorRampShader.colorRampItemList();
574 if ( itemList.isEmpty() || itemList.first().value > value )
578 itemList.removeLast();
580 if ( value > itemList.last().value )
581 return itemList.count() - 1;
583 int indSup = itemList.count() - 1;
588 if ( abs( indSup - indInf ) <= 1 )
591 int newInd = ( indInf + indSup ) / 2;
593 if ( std::isnan( itemList.at( newInd ).value ) )
596 if ( itemList.at( newInd ).value <= value )
603void QgsInterpolatedLineColor::graduatedColorsExact(
double value1,
double value2, QList<double> &breakValues, QList<QColor> &breakColors,
const QList<QLinearGradient> &gradients )
const
606 Q_ASSERT( breakValues.isEmpty() );
607 Q_ASSERT( breakColors.isEmpty() );
608 Q_ASSERT( gradients.isEmpty() );
610 const QList<QgsColorRampShader::ColorRampItem> &itemList = mColorRampShader.colorRampItemList();
611 if ( itemList.isEmpty() )
614 int index = itemColorIndexInf( value1 );
615 if ( index < 0 || !
qgsDoubleNear( value1, itemList.at( index ).value ) )
621 breakColors.append( itemList.at( index ).color );
625 while ( index < itemList.count() && itemList.at( index ).value <= value2 )
627 breakValues.append( itemList.at( index ).value );
628 breakColors.append( itemList.at( index ).color );
633void QgsInterpolatedLineColor::graduatedColorsInterpolated(
double value1,
double value2, QList<double> &breakValues, QList<QColor> &breakColors, QList<QLinearGradient> &gradients )
const
636 Q_ASSERT( breakValues.isEmpty() );
637 Q_ASSERT( breakColors.isEmpty() );
638 Q_ASSERT( gradients.isEmpty() );
641 const QList<QgsColorRampShader::ColorRampItem> &itemList = mColorRampShader.colorRampItemList();
642 if ( itemList.empty() )
645 if ( itemList.count() == 1 )
647 breakColors.append( itemList.first().color );
651 if ( value2 <= itemList.first().value )
653 if ( !mColorRampShader.clip() )
654 breakColors.append( itemList.first().color );
658 if ( value1 > itemList.last().value )
660 if ( !mColorRampShader.clip() )
661 breakColors.append( itemList.last().color );
671 if ( mColorRampShader.shade( value1, &r, &g, &b, &a ) )
672 color = QColor( r, g, b, a );
673 breakColors.append(
color );
678 int index = itemColorIndexInf( value1 );
681 QColor
color = itemList.first().color;
682 breakColors.append(
color );
683 if ( mColorRampShader.clip() )
684 breakValues.append( itemList.first().value );
686 breakValues.append( value1 );
693 if ( mColorRampShader.shade( value1, &r, &g, &b, &a ) )
694 color = QColor( r, g, b, a );
695 breakValues.append( value1 );
696 breakColors.append(
color );
701 while ( index < itemList.count() && itemList.at( index ).value < value2 )
703 QColor color1 = breakColors.last();
704 QColor color2 = itemList.at( index ).color;
705 breakValues.append( itemList.at( index ).value );
706 breakColors.append( color2 );
707 gradients.append( makeSimpleLinearGradient( color1, color2 ) );
712 QColor color1 = breakColors.last();
714 if ( value2 < itemList.last().value )
717 if ( mColorRampShader.shade( value2, &r, &g, &b, &a ) )
718 color2 = QColor( r, g, b, a );
719 breakValues.append( value2 );
723 color2 = itemList.last().color;
724 if ( mColorRampShader.clip() )
725 breakValues.append( itemList.last().value );
727 breakValues.append( value2 );
729 breakColors.append( color2 );
730 gradients.append( makeSimpleLinearGradient( color1, color2 ) );
734void QgsInterpolatedLineColor::graduatedColorsDiscrete(
double value1,
double value2, QList<double> &breakValues, QList<QColor> &breakColors, QList<QLinearGradient> &gradients )
const
737 Q_ASSERT( breakValues.isEmpty() );
738 Q_ASSERT( breakColors.isEmpty() );
739 Q_ASSERT( gradients.isEmpty() );
741 const QList<QgsColorRampShader::ColorRampItem> &itemList = mColorRampShader.colorRampItemList();
742 if ( itemList.empty() )
745 if ( itemList.count() == 1 )
747 breakColors.append( itemList.first().color );
751 double lastValue = itemList.at( itemList.count() - 2 ).value;
754 if ( value2 <= itemList.first().value )
756 breakColors.append( itemList.first().color );
760 if ( value1 > lastValue )
762 breakColors.append( itemList.last().color );
767 int index = itemColorIndexInf( value1 );
773 breakColors.append( itemList.at( index + 1 ).color );
779 breakValues.append( value1 );
780 breakColors.append( itemList.first().color );
784 QColor
color = itemList.at( index ).color;
785 breakValues.append( value1 );
786 breakColors.append(
color );
791 while ( index < ( itemList.count() - 1 ) && itemList.at( index ).value < value2 )
793 QColor
color = itemList.at( index ).color;
794 breakValues.append( itemList.at( index ).value );
795 breakColors.append(
color );
796 gradients.append( makeSimpleLinearGradient(
color,
color ) );
801 QColor lastColor = itemList.at( index ).color;
802 breakColors.append( lastColor );
803 breakValues.append( value2 );
804 gradients.append( makeSimpleLinearGradient( lastColor, lastColor ) );
828 std::unique_ptr<QgsInterpolatedLineSymbolLayer> symbolLayer;
829 symbolLayer = std::make_unique<QgsInterpolatedLineSymbolLayer>( );
831 if (
properties.contains( QStringLiteral(
"start_width_expression" ) ) )
833 if (
properties.contains( QStringLiteral(
"end_width_expression" ) ) )
836 if (
properties.contains( QStringLiteral(
"start_color_expression" ) ) )
838 if (
properties.contains( QStringLiteral(
"end_color_expression" ) ) )
841 if (
properties.contains( QStringLiteral(
"line_width" ) ) )
842 symbolLayer->mLineRender.mStrokeWidth.setFixedStrokeWidth(
properties.value( QStringLiteral(
"line_width" ) ).toDouble() ) ;
843 if (
properties.contains( QStringLiteral(
"line_width_unit" ) ) )
845 if (
properties.contains( QStringLiteral(
"width_varying_minimum_value" ) ) )
846 symbolLayer->mLineRender.mStrokeWidth.setMinimumValue(
properties.value( QStringLiteral(
"width_varying_minimum_value" ) ).toDouble() );
847 if (
properties.contains( QStringLiteral(
"width_varying_maximum_value" ) ) )
848 symbolLayer->mLineRender.mStrokeWidth.setMaximumValue(
properties.value( QStringLiteral(
"width_varying_maximum_value" ) ).toDouble() );
849 if (
properties.contains( QStringLiteral(
"width_varying_use_absolute_value" ) ) )
850 symbolLayer->mLineRender.mStrokeWidth.setUseAbsoluteValue(
properties.value( QStringLiteral(
"width_varying_use_absolute_value" ) ).toInt() );
851 if (
properties.contains( QStringLiteral(
"width_varying_minimum_width" ) ) )
852 symbolLayer->mLineRender.mStrokeWidth.setMinimumWidth(
properties.value( QStringLiteral(
"width_varying_minimum_width" ) ).toDouble() );
853 if (
properties.contains( QStringLiteral(
"width_varying_maximum_width" ) ) )
854 symbolLayer->mLineRender.mStrokeWidth.setMaximumWidth(
properties.value( QStringLiteral(
"width_varying_maximum_width" ) ).toDouble() );
855 if (
properties.contains( QStringLiteral(
"width_varying_ignore_out_of_range" ) ) )
856 symbolLayer->mLineRender.mStrokeWidth.setIgnoreOutOfRange(
properties.value( QStringLiteral(
"width_varying_ignore_out_of_range" ) ).toInt() );
857 if (
properties.contains( QStringLiteral(
"width_varying_is_variable_width" ) ) )
858 symbolLayer->mLineRender.mStrokeWidth.setIsVariableWidth(
properties.value( QStringLiteral(
"width_varying_is_variable_width" ) ).toInt() );
860 if (
properties.contains( QStringLiteral(
"single_color" ) ) )
862 if (
properties.contains( QStringLiteral(
"color_ramp_shader" ) ) )
863 symbolLayer->mLineRender.mStrokeColoring.setColor( createColorRampShaderFromProperties(
properties.value( QStringLiteral(
"color_ramp_shader" ) ) ) );
864 if (
properties.contains( QStringLiteral(
"coloring_method" ) ) )
865 symbolLayer->mLineRender.mStrokeColoring.setColoringMethod(
868 return symbolLayer.release();
883 props.insert( QStringLiteral(
"line_width" ), QString::number( mLineRender.mStrokeWidth.fixedStrokeWidth() ) );
885 props.insert( QStringLiteral(
"width_varying_minimum_value" ), mLineRender.mStrokeWidth.minimumValue() );
886 props.insert( QStringLiteral(
"width_varying_maximum_value" ), mLineRender.mStrokeWidth.maximumValue() );
887 props.insert( QStringLiteral(
"width_varying_use_absolute_value" ), mLineRender.mStrokeWidth.useAbsoluteValue() ? 1 : 0 );
888 props.insert( QStringLiteral(
"width_varying_minimum_width" ), mLineRender.mStrokeWidth.minimumWidth() );
889 props.insert( QStringLiteral(
"width_varying_maximum_width" ), mLineRender.mStrokeWidth.maximumWidth() );
890 props.insert( QStringLiteral(
"width_varying_ignore_out_of_range" ), mLineRender.mStrokeWidth.ignoreOutOfRange() ? 1 : 0 );
891 props.insert( QStringLiteral(
"width_varying_is_variable_width" ), mLineRender.mStrokeWidth.isVariableWidth() ? 1 : 0 );
894 props.insert( QStringLiteral(
"coloring_method" ), mLineRender.mStrokeColoring.coloringMethod() );
896 props.insert( QStringLiteral(
"color_ramp_shader" ), colorRampShaderProperties() );
907 double min = std::min( mLineRender.interpolatedLineWidth().minimumValue(), mLineRender.interpolatedColor().colorRampShader().minimumValue() );
908 double max = std::max( mLineRender.interpolatedLineWidth().maximumValue(), mLineRender.interpolatedColor().colorRampShader().maximumValue() );
910 double totalLength = geometry.
length();
914 double variation = ( max - min ) / totalLength;
917 double lengthFromStart = 0;
918 for (
int i = 1; i < points.count(); ++i )
920 QPointF p1 = points.at( i - 1 );
921 QPointF p2 = points.at( i );
923 double v1 = min + variation * lengthFromStart;
924 QPointF vectDist = p2 - p1;
925 lengthFromStart += sqrt( pow( vectDist.x(), 2 ) + pow( vectDist.y(), 2 ) );
926 double v2 = min + variation * lengthFromStart;
927 mLineRender.renderInDeviceCoordinates( v1, v2, v1, v2, p1, p2, context.
renderContext() );
936 switch ( mLineRender.interpolatedColor().coloringMethod() )
939 return mLineRender.interpolatedColor().singleColor();
949 if ( start.isEmpty() )
972 mLineRender.mStrokeWidthUnit = strokeWidthUnit;
979 mLineRender.mStrokeWidth = interpolatedLineWidth;
986 if ( start.isEmpty() )
1009 mLineRender.setInterpolatedColor( interpolatedLineColor );
1014 return mLineRender.interpolatedColor();
1017QVariant QgsInterpolatedLineSymbolLayer::colorRampShaderProperties()
const
1024 props.insert( QStringLiteral(
"color_ramp_shader_type" ),
static_cast< int >( colorRampShader.
colorRampType() ) );
1025 props.insert( QStringLiteral(
"color_ramp_shader_classification_mode" ),
static_cast< int >( colorRampShader.
classificationMode() ) );
1026 QVariantList colorRampItemListVariant;
1028 const QList<QgsColorRampShader::ColorRampItem> colorRampItemList = colorRampShader.
colorRampItemList();
1031 QVariantMap itemVar;
1032 itemVar[QStringLiteral(
"label" )] = item.label;
1034 itemVar[QStringLiteral(
"value" )] = item.value;
1035 colorRampItemListVariant.append( itemVar );
1037 props.insert( QStringLiteral(
"color_ramp_shader_items_list" ), colorRampItemListVariant );
1039 props.insert( QStringLiteral(
"color_ramp_shader_minimum_value" ), colorRampShader.
minimumValue() );
1040 props.insert( QStringLiteral(
"color_ramp_shader_maximum_value" ), colorRampShader.
maximumValue() );
1041 props.insert( QStringLiteral(
"color_ramp_shader_value_out_of_range" ), colorRampShader.
clip() ? 1 : 0 );
1042 props.insert( QStringLiteral(
"color_ramp_shader_label_precision" ), colorRampShader.
labelPrecision() );
1047QgsColorRampShader QgsInterpolatedLineSymbolLayer::createColorRampShaderFromProperties(
const QVariant &properties )
1049 QgsColorRampShader colorRampShader;
1051 if (
properties.userType() != QMetaType::Type::QVariantMap )
1052 return colorRampShader;
1054 QVariantMap shaderVariantMap =
properties.toMap();
1056 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_source" ) ) )
1059 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_type" ) ) )
1061 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_classification_mode" ) ) )
1063 shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_classification_mode" ) ).toInt() ) );
1065 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_items_list" ) ) )
1067 QVariant colorRampItemsVar = shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_items_list" ) );
1068 if ( colorRampItemsVar.userType() == QMetaType::Type::QVariantList )
1070 QVariantList itemVariantList = colorRampItemsVar.toList();
1071 QList<QgsColorRampShader::ColorRampItem> colorRampItemList;
1072 for (
const QVariant &itemVar : std::as_const( itemVariantList ) )
1074 QgsColorRampShader::ColorRampItem item;
1075 if ( itemVar.userType() != QMetaType::Type::QVariantMap )
1077 QVariantMap itemVarMap = itemVar.toMap();
1078 if ( !itemVarMap.contains( QStringLiteral(
"label" ) ) || !itemVarMap.contains( QStringLiteral(
"color" ) ) || !itemVarMap.contains( QStringLiteral(
"value" ) ) )
1081 item.
label = itemVarMap.value( QStringLiteral(
"label" ) ).toString();
1083 item.
value = itemVarMap.value( QStringLiteral(
"value" ) ).toDouble();
1085 colorRampItemList.append( item );
1091 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_minimum_value" ) ) )
1092 colorRampShader.
setMinimumValue( shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_minimum_value" ) ).toDouble() );
1094 colorRampShader.
setMinimumValue( std::numeric_limits<double>::quiet_NaN() );
1096 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_maximum_value" ) ) )
1097 colorRampShader.
setMaximumValue( shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_maximum_value" ) ).toDouble() );
1099 colorRampShader.
setMaximumValue( std::numeric_limits<double>::quiet_NaN() );
1101 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_value_out_of_range" ) ) )
1102 colorRampShader.
setClip( shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_value_out_of_range" ) ).toInt() == 1 );
1103 if ( shaderVariantMap.contains( QStringLiteral(
"color_ramp_shader_label_precision" ) ) )
1104 colorRampShader.
setLabelPrecision( shaderVariantMap.value( QStringLiteral(
"color_ramp_shader_label_precision" ) ).toInt() );
1106 return colorRampShader;
1114 mRenderingFeature =
true;
1120 mRenderingFeature =
false;
1122 if ( mLineParts.empty() )
1125 render( mLineParts, context );
1129void QgsInterpolatedLineSymbolLayer::render(
const QVector< QPolygonF > &parts,
QgsRenderContext &context )
1131 const double totalLength = std::accumulate( parts.begin(), parts.end(), 0.0, [](
double total,
const QPolygonF & part )
1133 return total + QgsSymbolLayerUtils::polylineLength( part );
1139 double startValWidth = 0;
1140 double variationPerMapUnitWidth = 0;
1141 double startValColor = 0;
1142 double variationPerMapUnitColor = 0;
1155 double endValWidth = 0;
1163 variationPerMapUnitWidth = ( endValWidth - startValWidth ) / totalLength;
1175 double endValColor = 0;
1183 variationPerMapUnitColor = ( endValColor - startValColor ) / totalLength;
1186 for (
const QPolygonF &poly : parts )
1188 double lengthFromStart = 0;
1189 for (
int i = 1; i < poly.count(); ++i )
1191 const QPointF p1 = poly.at( i - 1 );
1192 const QPointF p2 = poly.at( i );
1194 const double v1c = startValColor + variationPerMapUnitColor * lengthFromStart;
1195 const double v1w = startValWidth + variationPerMapUnitWidth * lengthFromStart;
1197 const double v2c = startValColor + variationPerMapUnitColor * lengthFromStart;
1198 const double v2w = startValWidth + variationPerMapUnitWidth * lengthFromStart;
1199 mLineRender.renderInDeviceCoordinates( v1c, v2c, v1w, v2w, p1, p2, context );
1206 mLineRender.setSelected( context.
selected() );
1208 if ( points.empty() )
1211 if ( mRenderingFeature )
1215 mLineParts.append( points );
@ DisableFeatureClipping
If present, indicates that features should never be clipped to the map extent during rendering.
@ CanCalculateMaskGeometryPerFeature
If present, indicates that mask geometry can safely be calculated per feature for the symbol layer....
ShaderInterpolationMethod
Color ramp shader interpolation methods.
@ Exact
Assigns the color of the exact matching value in the color ramp item list.
@ Linear
Interpolates the color between two class breaks linearly.
@ Discrete
Assigns the color of the higher class for every pixel between two class breaks.
QFlags< SymbolLayerFlag > SymbolLayerFlags
Symbol layer flags.
ShaderClassificationMethod
Color ramp shader classification methods.
RenderUnit
Rendering size units.
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.
A ramp shader will color a raster pixel based on a list of values ranges in a ramp.
Qgis::ShaderClassificationMethod classificationMode() const
Returns the classification mode.
Qgis::ShaderInterpolationMethod colorRampType() const
Returns the color ramp interpolation method.
void setSourceColorRamp(QgsColorRamp *colorramp)
Set the source color ramp.
void setClassificationMode(Qgis::ShaderClassificationMethod classificationMode)
Sets the classification mode.
QList< QgsColorRampShader::ColorRampItem > colorRampItemList() const
Returns the custom color map.
void setClip(bool clip)
Sets whether the shader should not render values out of range.
bool clip() const
Returns whether the shader will clip values which are out of range.
QgsColorRamp * sourceColorRamp() const
Returns the source color ramp.
void setColorRampType(Qgis::ShaderInterpolationMethod colorRampType)
Sets the color ramp interpolation method.
void setColorRampItemList(const QList< QgsColorRampShader::ColorRampItem > &list)
Sets a custom color map.
void readXml(const QDomElement &elem, const QgsReadWriteContext &context=QgsReadWriteContext())
Reads configuration from the given DOM element.
Abstract base class for color ramps.
virtual QColor color(double value) const =0
Returns the color corresponding to a specified value.
static QColor colorFromString(const QString &string)
Decodes a string into a color value.
static QString colorToString(const QColor &color)
Encodes a color into a string value.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
static double distance2D(double x1, double y1, double x2, double y2)
Returns the 2D distance between (x1, y1) and (x2, y2).
A geometry is the spatial representation of a feature.
QPolygonF asQPolygonF() const
Returns contents of the geometry as a QPolygonF.
double length() const
Returns the planar, 2-dimensional length of geometry.
Defines color interpolation for rendering mesh datasets.
QgsInterpolatedLineColor::ColoringMethod coloringMethod() const
Returns the coloring method used.
QgsColorRampShader colorRampShader() const
Returns the color ramp shader.
void readXml(const QDomElement &elem, const QgsReadWriteContext &context)
Reads configuration from the given DOM element.
QColor color(double magnitude) const
Returns the color corresponding to the magnitude.
void graduatedColors(double value1, double value2, QList< double > &breakValues, QList< QColor > &breakColors, QList< QLinearGradient > &gradients) const
Returns the break values, graduated colors and the associated gradients between two values.
QgsInterpolatedLineColor()
Default constructor.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Writes configuration to a new DOM element.
void setColor(const QgsColorRampShader &colorRampShader)
Sets the color ramp to define the coloring.
QColor singleColor() const
Returns the single color that is used if SingleColor coloring mode is set.
void setColoringMethod(ColoringMethod coloringMethod)
Sets the coloring method used.
ColoringMethod
Defines how the color is defined.
@ ColorRamp
Render with a color ramp.
@ SingleColor
Render with a single color.
void setInterpolatedColor(const QgsInterpolatedLineColor &strokeColoring)
Sets the stroke color used to render.
void renderInDeviceCoordinates(double valueColor1, double valueColor2, double valueWidth1, double valueWidth2, QPointF point1, QPointF point2, QgsRenderContext &context) const
Renders a line in the context between point1 and point2 in device (painter) coordinates with color th...
Qgis::RenderUnit widthUnit() const
Returns the unit of the stroke width.
void setInterpolatedWidth(const QgsInterpolatedLineWidth &strokeWidth)
Sets the stroke width used to render.
void setSelected(bool selected)
Sets if the rendering must be done as the element is selected.
void render(double value1, double value2, const QgsPointXY &point1, const QgsPointXY &point2, QgsRenderContext &context) const
Renders a line in the context between point1 and point2 with color and width that vary depending on v...
QgsInterpolatedLineWidth interpolatedLineWidth() const
Returns the stroke width used to render.
QgsInterpolatedLineColor interpolatedColor() const
Returns the stroke color used to render.
void setWidthUnit(Qgis::RenderUnit strokeWidthUnit)
Sets the unit of the stroke width.
Q_DECL_DEPRECATED QString endValueExpressionForWidth() const
Returns the expression related to the end extremity value for width.
QColor color() const override
Returns the "representative" color of the symbol layer.
Q_DECL_DEPRECATED QString endValueExpressionForColor() const
Returns the expression related to the end extremity value for width for color.
Q_DECL_DEPRECATED void setExpressionsStringForColor(const QString &start, const QString &end)
Sets the expressions (as string) that define the extremety values af the line feature for color.
Qgis::RenderUnit widthUnit() const
Returns the width unit.
Qgis::SymbolLayerFlags flags() const override
Returns flags which control the symbol layer's behavior.
QgsInterpolatedLineSymbolLayer * clone() const override
Shall be reimplemented by subclasses to create a deep copy of the instance.
bool canCauseArtifactsBetweenAdjacentTiles() const override
Returns true if the symbol layer rendering can cause visible artifacts across a single feature when t...
void renderPolyline(const QPolygonF &points, QgsSymbolRenderContext &context) override
Renders the line symbol layer along the line joining points, using the given render context.
QgsInterpolatedLineColor interpolatedColor() const
Returns the interpolated color used to render the colors of lines, see QgsInterpolatedLineColor.
Q_DECL_DEPRECATED QString startValueExpressionForColor() const
Returns the epression related to the start extremity value for width for color.
void startRender(QgsSymbolRenderContext &context) override
Called before a set of rendering operations commences on the supplied render context.
QgsInterpolatedLineSymbolLayer()
void startFeatureRender(const QgsFeature &feature, QgsRenderContext &context) override
Called before the layer will be rendered for a particular feature.
QString layerType() const override
Returns a string that represents this layer type.
Q_DECL_DEPRECATED void setExpressionsStringForWidth(const QString &start, const QString &end)
Sets the expressions (as string) that define the extremety values af the line feature for width.
void setInterpolatedWidth(const QgsInterpolatedLineWidth &interpolatedLineWidth)
Sets the interpolated width used to render the width of lines, see QgsInterpolatedLineWidth.
void setInterpolatedColor(const QgsInterpolatedLineColor &interpolatedLineColor)
Sets the interpolated color used to render the colors of lines, see QgsInterpolatedLineColor.
Q_DECL_DEPRECATED QString startValueExpressionForWidth() const
Returns the epression related to the start extremity value for width.
void drawPreviewIcon(QgsSymbolRenderContext &context, QSize size) override
QVariantMap properties() const override
Should be reimplemented by subclasses to return a string map that contains the configuration informat...
QgsInterpolatedLineWidth interpolatedWidth() const
Returns the interpolated width used to render the width of lines, see QgsInterpolatedLineWidth.
void stopFeatureRender(const QgsFeature &feature, QgsRenderContext &context) override
Called after the layer has been rendered for a particular feature.
void setWidthUnit(Qgis::RenderUnit strokeWidthUnit)
Sets the width unit.
static QgsSymbolLayer * create(const QVariantMap &properties)
Creates the symbol layer.
void stopRender(QgsSymbolRenderContext &context) override
Called after a set of rendering operations has finished on the supplied render context.
Represents a width that can vary depending on values.
void setFixedStrokeWidth(double fixedWidth)
Sets the fixed width.
double strokeWidth(double value) const
Returns the variable width depending on value, if not varying returns the fixed width.
void setUseAbsoluteValue(bool useAbsoluteValue)
Sets whether absolute value are used as input.
double minimumValue() const
Returns the minimum value used to defined the variable width.
void readXml(const QDomElement &elem, const QgsReadWriteContext &context)
Reads configuration from the given DOM element.
void setIgnoreOutOfRange(bool ignoreOutOfRange)
Sets whether the variable width ignores out of range value.
void setMaximumValue(double maximumValue)
Sets the maximum value used to defined the variable width.
bool useAbsoluteValue() const
Returns whether absolute value are used as input.
void setIsVariableWidth(bool isVariableWidth)
Returns whether the width is variable.
void setMinimumValue(double minimumValue)
Sets the minimum value used to defined the variable width.
double maximumWidth() const
Returns the maximum width used to defined the variable width.
void setMaximumWidth(double maximumWidth)
Sets the maximum width used to defined the variable width.
double maximumValue() const
Returns the maximum value used to defined the variable width.
void setMinimumWidth(double minimumWidth)
Sets the minimum width used to defined the variable width.
QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Writes configuration to a new DOM element.
bool ignoreOutOfRange() const
Returns whether the variable width ignores out of range value.
double minimumWidth() const
Returns the minimum width used to defined the variable width.
double fixedStrokeWidth() const
Returns the fixed width.
bool isVariableWidth() const
Returns whether the width is variable.
QgsGeometry geometry() const
Returns the geometry for the patch shape.
QgsLineSymbolLayer(const QgsLineSymbolLayer &other)=delete
Perform transforms between map coordinates and device coordinates.
QgsPointXY transform(const QgsPointXY &p) const
Transforms a point p from map (world) coordinates to device coordinates.
QPointF toQPointF() const
Converts a point to a QPointF.
bool isActive(int key) const final
Returns true if the collection contains an active property with the specified key.
A store for object properties.
static QgsProperty fromExpression(const QString &expression, bool isActive=true)
Returns a new ExpressionBasedProperty created from the specified expression.
double maximumValue() const
Returns the minimum value for the raster shader.
void setLabelPrecision(int labelPrecision)
Sets label precision to labelPrecision.
int labelPrecision() const
Returns label precision.
virtual void setMaximumValue(double value)
Sets the maximum value for the raster shader.
virtual void setMinimumValue(double value)
Sets the minimum value for the raster shader.
double minimumValue() const
Returns the maximum value for the raster shader.
A container for the context for various read/write operations on objects.
Contains information about the context of a rendering operation.
double convertToPainterUnits(double size, Qgis::RenderUnit unit, const QgsMapUnitScale &scale=QgsMapUnitScale(), Qgis::RenderSubcomponentProperty property=Qgis::RenderSubcomponentProperty::Generic) const
Converts a size from the specified units to painter units (pixels).
QPainter * painter()
Returns the destination QPainter for the render operation.
void setPainterFlagsUsingContext(QPainter *painter=nullptr) const
Sets relevant flags on a destination painter, using the flags and settings currently defined for the ...
QgsExpressionContext & expressionContext()
Gets the expression context.
const QgsMapToPixel & mapToPixel() const
Returns the context's map to pixel transform, which transforms between map coordinates and device coo...
QColor selectionColor() const
Returns the color to use when rendering selected features.
Scoped object for saving and restoring a QPainter object's state.
QgsLegendPatchShape defaultPatch(Qgis::SymbolType type, QSizeF size) const
Returns the default legend patch shape for the given symbol type.
static QgsStyle * defaultStyle(bool initialize=true)
Returns the default application-wide style.
static QVariant colorRampToVariant(const QString &name, QgsColorRamp *ramp)
Saves a color ramp to a QVariantMap, wrapped in a QVariant.
static std::unique_ptr< QgsColorRamp > loadColorRamp(QDomElement &element)
Creates a color ramp from the settings encoded in an XML element.
void copyDataDefinedProperties(QgsSymbolLayer *destLayer) const
Copies all data defined properties of this layer to another symbol layer.
@ LineEndColorValue
End line color for interpolated line renderer.
@ LineStartColorValue
Start line color for interpolated line renderer.
@ LineEndWidthValue
End line width for interpolated line renderer.
@ LineStartWidthValue
Start line width for interpolated line renderer.
void copyPaintEffect(QgsSymbolLayer *destLayer) const
Copies paint effect of this layer to another symbol layer.
virtual Qgis::SymbolLayerFlags flags() const
Returns flags which control the symbol layer's behavior.
QgsPropertyCollection mDataDefinedProperties
QgsSymbolLayer(const QgsSymbolLayer &other)
Encapsulates the context in which a symbol is being rendered.
bool selected() const
Returns true if symbols should be rendered using the selected symbol coloring and style.
const QgsLegendPatchShape * patchShape() const
Returns the symbol patch shape, to use if rendering symbol preview icons.
QgsRenderContext & renderContext()
Returns a reference to the context's render context.
static Q_INVOKABLE Qgis::RenderUnit decodeRenderUnit(const QString &string, bool *ok=nullptr)
Decodes a render unit from a string.
static Q_INVOKABLE QString encodeUnit(Qgis::DistanceUnit unit)
Encodes a distance unit to a string.
#define BUILTIN_UNREACHABLE
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference).