30using namespace Qt::StringLiterals;
80 QVariantMap transformerMap = transformer.toMap();
82 mMinValue = transformerMap.value( u
"minValue"_s, 0.0 ).toDouble();
83 mMaxValue = transformerMap.value( u
"maxValue"_s, 1.0 ).toDouble();
86 QVariantMap curve = transformerMap.value( u
"curve"_s ).toMap();
88 if ( !curve.isEmpty() )
99 QVariantMap transformerMap;
101 transformerMap.insert( u
"minValue"_s,
mMinValue );
102 transformerMap.insert( u
"maxValue"_s,
mMaxValue );
108 return transformerMap;
113 baseExpression.clear();
143 , mMinOutput( minOutput )
144 , mMaxOutput( maxOutput )
145 , mNullOutput( nullOutput )
151 auto t = std::make_unique<QgsGenericNumericTransformer>(
mMinValue,
mMaxValue, mMinOutput, mMaxOutput, mNullOutput, mExponent );
161 transformerMap.insert( u
"minOutput"_s, mMinOutput );
162 transformerMap.insert( u
"maxOutput"_s, mMaxOutput );
163 transformerMap.insert( u
"nullOutput"_s, mNullOutput );
164 transformerMap.insert( u
"exponent"_s, mExponent );
166 return transformerMap;
173 QVariantMap transformerMap = transformer.toMap();
175 mMinOutput = transformerMap.value( u
"minOutput"_s, 0.0 ).toDouble();
176 mMaxOutput = transformerMap.value( u
"maxOutput"_s, 1.0 ).toDouble();
177 mNullOutput = transformerMap.value( u
"nullOutput"_s, 0.0 ).toDouble();
178 mExponent = transformerMap.value( u
"exponent"_s, 1.0 ).toDouble();
204 double dblValue = v.toDouble( &ok );
209 return value( dblValue );
219 QString minValueString = QString::number(
mMinValue );
220 QString maxValueString = QString::number(
mMaxValue );
221 QString minOutputString = QString::number( mMinOutput );
222 QString maxOutputString = QString::number( mMaxOutput );
223 QString nullOutputString = QString::number( mNullOutput );
224 QString exponentString = QString::number( mExponent );
227 return u
"coalesce(scale_linear(%1, %2, %3, %4, %5), %6)"_s.arg( baseExpression, minValueString, maxValueString, minOutputString, maxOutputString, nullOutputString );
229 return u
"coalesce(scale_polynomial(%1, %2, %3, %4, %5, %6), %7)"_s.arg( baseExpression, minValueString, maxValueString, minOutputString, maxOutputString, exponentString, nullOutputString );
236 double nullValue = 0.0;
239 baseExpression.clear();
251 QList<QgsExpressionNode *> args = f->
args()->
list();
300 baseExpression = args[0]->dump();
321 auto t = std::make_unique<QgsSizeScaleTransformer>( mType,
mMinValue,
mMaxValue, mMinSize, mMaxSize, mNullSize, mExponent );
331 transformerMap.insert( u
"scaleType"_s,
static_cast< int >( mType ) );
332 transformerMap.insert( u
"minSize"_s, mMinSize );
333 transformerMap.insert( u
"maxSize"_s, mMaxSize );
334 transformerMap.insert( u
"nullSize"_s, mNullSize );
335 transformerMap.insert( u
"exponent"_s, mExponent );
337 return transformerMap;
344 QVariantMap transformerMap = transformer.toMap();
346 mType =
static_cast< ScaleType >( transformerMap.value( u
"scaleType"_s,
Linear ).toInt() );
347 mMinSize = transformerMap.value( u
"minSize"_s, 0.0 ).toDouble();
348 mMaxSize = transformerMap.value( u
"maxSize"_s, 1.0 ).toDouble();
349 mNullSize = transformerMap.value( u
"nullSize"_s, 0.0 ).toDouble();
350 mExponent = transformerMap.value( u
"exponent"_s, 1.0 ).toDouble();
405 double dblValue = value.toDouble( &ok );
410 return size( dblValue );
420 QString minValueString = QString::number(
mMinValue );
421 QString maxValueString = QString::number(
mMaxValue );
422 QString minSizeString = QString::number( mMinSize );
423 QString maxSizeString = QString::number( mMaxSize );
424 QString nullSizeString = QString::number( mNullSize );
425 QString exponentString = QString::number( mExponent );
430 return u
"coalesce(scale_linear(%1, %2, %3, %4, %5), %6)"_s.arg( baseExpression, minValueString, maxValueString, minSizeString, maxSizeString, nullSizeString );
435 return u
"coalesce(scale_polynomial(%1, %2, %3, %4, %5, %6), %7)"_s.arg( baseExpression, minValueString, maxValueString, minSizeString, maxSizeString, exponentString, nullSizeString );
448 baseExpression.clear();
460 QList<QgsExpressionNode *> args = f->
args()->
list();
517 baseExpression = args[0]->dump();
529 , mGradientRamp( ramp )
536 , mGradientRamp( other.mGradientRamp ? other.mGradientRamp->
clone() : nullptr )
537 , mNullColor( other.mNullColor )
538 , mRampName( other.mRampName )
543 if ( &other ==
this )
549 mGradientRamp.reset( other.mGradientRamp ? other.mGradientRamp->clone() :
nullptr );
550 mNullColor = other.mNullColor;
551 mRampName = other.mRampName;
557 auto c = std::make_unique<QgsColorRampTransformer>(
mMinValue,
mMaxValue, mGradientRamp ? mGradientRamp->clone() :
nullptr, mNullColor );
558 c->setRampName( mRampName );
573 transformerMap.insert( u
"rampName"_s, mRampName );
575 return transformerMap;
580 QVariantMap transformerMap = definition.toMap();
584 mGradientRamp.reset(
nullptr );
585 if ( transformerMap.contains( u
"colorramp"_s ) )
591 mRampName = transformerMap.value( u
"rampName"_s ).toString();
603 double dblValue = value.toDouble( &ok );
608 return color( dblValue );
618 if ( !mGradientRamp )
621 QString minValueString = QString::number(
mMinValue );
622 QString maxValueString = QString::number(
mMaxValue );
623 QString nullColorString = mNullColor.name();
625 return u
"coalesce(ramp_color('%1',scale_linear(%2, %3, %4, 0, 1)), '%5')"_s.arg( !mRampName.isEmpty() ? mRampName : u
"custom ramp"_s, baseExpression, minValueString, maxValueString, nullColorString );
632 return mGradientRamp ? mGradientRamp->color( 0 ) : mNullColor;
638 if ( !mGradientRamp )
641 return mGradientRamp->color( scaledVal );
646 return mGradientRamp.get();
651 mGradientRamp.reset( ramp );
661 return a.
x() < b.
x();
667 calcSecondDerivativeArray();
673 std::sort( mControlPoints.begin(), mControlPoints.end(),
sortByX );
674 calcSecondDerivativeArray();
679 delete[] mSecondDerivativeArray;
683 : mControlPoints( other.mControlPoints )
685 if ( other.mSecondDerivativeArray )
687 mSecondDerivativeArray = new double[mControlPoints.count()];
688 memcpy( mSecondDerivativeArray, other.mSecondDerivativeArray, sizeof( double ) * mControlPoints.count() );
694 if (
this != &other )
696 mControlPoints = other.mControlPoints;
697 if ( other.mSecondDerivativeArray )
699 delete[] mSecondDerivativeArray;
700 mSecondDerivativeArray =
new double[mControlPoints.count()];
701 memcpy( mSecondDerivativeArray, other.mSecondDerivativeArray,
sizeof(
double ) * mControlPoints.count() );
709 mControlPoints = points;
710 std::sort( mControlPoints.begin(), mControlPoints.end(),
sortByX );
711 for (
int i = 0; i < mControlPoints.count(); ++i )
713 mControlPoints[i] =
QgsPointXY( std::clamp( mControlPoints.at( i ).x(), 0.0, 1.0 ), std::clamp( mControlPoints.at( i ).y(), 0.0, 1.0 ) );
715 calcSecondDerivativeArray();
721 if ( mControlPoints.contains( point ) )
724 mControlPoints << point;
725 std::sort( mControlPoints.begin(), mControlPoints.end(),
sortByX );
726 calcSecondDerivativeArray();
731 for (
int i = 0; i < mControlPoints.count(); ++i )
735 mControlPoints.removeAt( i );
739 calcSecondDerivativeArray();
748 int n = mControlPoints.count();
750 return std::clamp( x, 0.0, 1.0 );
754 if ( x <= mControlPoints.at( 0 ).x() )
755 return std::clamp( mControlPoints.at( 0 ).y(), 0.0, 1.0 );
756 else if ( x >= mControlPoints.at( n - 1 ).x() )
757 return std::clamp( mControlPoints.at( 1 ).y(), 0.0, 1.0 );
760 double dx = mControlPoints.at( 1 ).x() - mControlPoints.at( 0 ).x();
761 double dy = mControlPoints.at( 1 ).y() - mControlPoints.at( 0 ).y();
762 return std::clamp( ( x - mControlPoints.at( 0 ).x() ) * ( dy / dx ) + mControlPoints.at( 0 ).y(), 0.0, 1.0 );
767 if ( x <= mControlPoints.at( 0 ).x() )
768 return std::clamp( mControlPoints.at( 0 ).y(), 0.0, 1.0 );
769 if ( x >= mControlPoints.at( n - 1 ).x() )
770 return std::clamp( mControlPoints.at( n - 1 ).y(), 0.0, 1.0 );
773 QList<QgsPointXY>::const_iterator pointIt = mControlPoints.constBegin();
778 for (
int i = 0; i < n - 1; ++i )
780 if ( x < nextControlPoint.
x() )
783 double h = nextControlPoint.
x() - currentControlPoint.
x();
784 double t = ( x - currentControlPoint.
x() ) / h;
788 return std::clamp( a * currentControlPoint.
y() + t * nextControlPoint.
y() + ( h * h / 6 ) * ( ( a * a * a - a ) * mSecondDerivativeArray[i] + ( t * t * t - t ) * mSecondDerivativeArray[i + 1] ), 0.0, 1.0 );
792 if ( pointIt == mControlPoints.constEnd() )
795 currentControlPoint = nextControlPoint;
796 nextControlPoint = *pointIt;
800 return std::clamp( x, 0.0, 1.0 );
809 QVector<double> result;
811 int n = mControlPoints.count();
815 const auto constX = x;
816 for (
double i : constX )
823 QList<QgsPointXY>::const_iterator pointIt = mControlPoints.constBegin();
829 double currentX = x.at( xIndex );
831 while ( currentX <= currentControlPoint.
x() )
833 result << std::clamp( currentControlPoint.
y(), 0.0, 1.0 );
835 currentX = x.at( xIndex );
838 for (
int i = 0; i < n - 1; ++i )
840 while ( currentX < nextControlPoint.
x() )
843 double h = nextControlPoint.
x() - currentControlPoint.
x();
845 double t = ( currentX - currentControlPoint.
x() ) / h;
849 result << std::clamp( a * currentControlPoint.
y() + t * nextControlPoint.
y() + ( h * h / 6 ) * ( ( a * a * a - a ) * mSecondDerivativeArray[i] + ( t * t * t - t ) * mSecondDerivativeArray[i + 1] ), 0.0, 1.0 );
851 if ( xIndex == x.count() )
854 currentX = x.at( xIndex );
858 if ( pointIt == mControlPoints.constEnd() )
861 currentControlPoint = nextControlPoint;
862 nextControlPoint = *pointIt;
866 while ( xIndex < x.count() )
868 result << std::clamp( nextControlPoint.
y(), 0.0, 1.0 );
877 QString xString = elem.attribute( u
"x"_s );
878 QString yString = elem.attribute( u
"y"_s );
880 QStringList xVals = xString.split(
',' );
881 QStringList yVals = yString.split(
',' );
882 if ( xVals.count() != yVals.count() )
885 QList< QgsPointXY > newPoints;
887 for (
int i = 0; i < xVals.count(); ++i )
889 double x = xVals.at( i ).toDouble( &ok );
892 double y = yVals.at( i ).toDouble( &ok );
905 const auto constMControlPoints = mControlPoints;
906 for (
const QgsPointXY &p : constMControlPoints )
912 transformElem.setAttribute( u
"x"_s, x.join(
',' ) );
913 transformElem.setAttribute( u
"y"_s,
y.join(
',' ) );
920 QVariantMap transformMap;
924 const auto constMControlPoints = mControlPoints;
925 for (
const QgsPointXY &p : constMControlPoints )
931 transformMap.insert( u
"x"_s, x.join(
',' ) );
932 transformMap.insert( u
"y"_s,
y.join(
',' ) );
939 QVariantMap transformMap = transformer.toMap();
941 QString xString = transformMap.value( u
"x"_s ).toString();
942 QString yString = transformMap.value( u
"y"_s ).toString();
944 QStringList xVals = xString.split(
',' );
945 QStringList yVals = yString.split(
',' );
946 if ( xVals.count() != yVals.count() )
949 QList< QgsPointXY > newPoints;
951 for (
int i = 0; i < xVals.count(); ++i )
953 double x = xVals.at( i ).toDouble( &ok );
956 double y = yVals.at( i ).toDouble( &ok );
969void QgsCurveTransform::calcSecondDerivativeArray()
971 int n = mControlPoints.count();
975 delete[] mSecondDerivativeArray;
977 double *matrix =
new double[n * 3];
978 double *result =
new double[n];
983 QList<QgsPointXY>::const_iterator pointIt = mControlPoints.constBegin();
990 for (
int i = 1; i < n - 1; ++i )
992 matrix[i * 3 + 0] = ( pointI.
x() - pointIm1.
x() ) / 6.0;
993 matrix[i * 3 + 1] = ( pointIp1.
x() - pointIm1.
x() ) / 3.0;
994 matrix[i * 3 + 2] = ( pointIp1.
x() - pointI.
x() ) / 6.0;
995 result[i] = ( pointIp1.
y() - pointI.
y() ) / ( pointIp1.
x() - pointI.
x() ) - ( pointI.
y() - pointIm1.
y() ) / ( pointI.
x() - pointIm1.
x() );
1001 if ( pointIt == mControlPoints.constEnd() )
1004 pointIp1 = *pointIt;
1006 matrix[( n - 1 ) * 3 + 0] = 0;
1007 matrix[( n - 1 ) * 3 + 1] = 1;
1008 matrix[( n - 1 ) * 3 + 2] = 0;
1012 for (
int i = 1; i < n; ++i )
1014 double k = matrix[i * 3 + 0] / matrix[( i - 1 ) * 3 + 1];
1015 matrix[i * 3 + 1] -= k * matrix[( i - 1 ) * 3 + 2];
1016 matrix[i * 3 + 0] = 0;
1017 result[i] -= k * result[i - 1];
1020 for (
int i = n - 2; i >= 0; --i )
1022 double k = matrix[i * 3 + 2] / matrix[( i + 1 ) * 3 + 1];
1023 matrix[i * 3 + 1] -= k * matrix[( i + 1 ) * 3 + 0];
1024 matrix[i * 3 + 2] = 0;
1025 result[i] -= k * result[i + 1];
1029 mSecondDerivativeArray =
new double[n];
1030 for (
int i = 0; i < n; ++i )
1032 mSecondDerivativeArray[i] = result[i] / matrix[( i * 3 ) + 1];
Abstract base class for color ramps.
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.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
An expression node which takes its value from a feature's field.
An expression node for expression functions.
int fnIndex() const
Returns the index of the node's function.
QgsExpressionNode::NodeList * args() const
Returns a list of arguments specified for the function.
QList< QgsExpressionNode * > list()
Gets a list of all the nodes.
Handles parsing and evaluation of expressions (formerly called "search strings").
static const QList< QgsExpressionFunction * > & Functions()
static QString quotedValue(const QVariant &value)
Returns a string representation of a literal value, including appropriate quotations where required.
const QgsExpressionNode * rootNode() const
Returns the root node of the expression.
QVariant evaluate()
Evaluate the feature and return the result.
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.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
As part of the API refactoring and improvements which landed in the Processing API was substantially reworked from the x version This was done in order to allow much of the underlying Processing framework to be ported into c
QString qgsDoubleToString(double a, int precision=17)
Returns a string representation of a double.
bool qgsDoubleNear(double a, double b, double epsilon=4 *std::numeric_limits< double >::epsilon())
Compare two doubles (but allow some difference).