30using namespace Qt::StringLiterals;
34QgsExpressionUtils::TVL QgsExpressionUtils::AND[3][3] =
37 { False, False, False },
41QgsExpressionUtils::TVL QgsExpressionUtils::OR[3][3] =
48QgsExpressionUtils::TVL QgsExpressionUtils::NOT[3] = { True, False,
Unknown };
51QColor QgsExpressionUtils::getColorValue(
const QVariant &value,
QgsExpression *parent,
bool &isQColor )
53 isQColor = value.userType() == QMetaType::Type::QColor;
54 QColor color = isQColor ? value.value<QColor>() : QgsSymbolLayerUtils::decodeColor( value.toString() );
55 if ( ! color.isValid() )
57 parent->setEvalErrorString( isQColor ? QObject::tr(
"Input color is invalid" )
58 : QObject::tr(
"Cannot convert '%1' to color" ).arg( value.toString() ) );
66 if ( value.userType() == qMetaTypeId<QgsGradientColorRamp>() )
67 return value.
value<QgsGradientColorRamp>();
71 parent->
setEvalErrorString( QObject::tr(
"Cannot convert '%1' to gradient ramp" ).arg( value.toString() ) );
73 return QgsGradientColorRamp();
78 return getMapLayerPrivate( value, context, parent );
87#ifdef __clang_analyzer__
93 ml = value.value< QgsMapLayer * >();
97 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
104 const QString identifier = value.toString();
109 const QList< QgsMapLayerStore * > stores = context->
layerStores();
110 for ( QgsMapLayerStore *store : stores )
112 auto findLayerInStoreFunction = [ storePointer = QPointer< QgsMapLayerStore >( store ), &ml, identifier ]
114 if ( QgsMapLayerStore *store = storePointer.data() )
117 ml = store->mapLayer( identifier );
123 ml = store->mapLayersByName( identifier ).value( 0 );
129 if ( QThread::currentThread() == store->thread() )
130 findLayerInStoreFunction();
132 QMetaObject::invokeMethod( store, std::move( findLayerInStoreFunction ), Qt::BlockingQueuedConnection );
139 auto getMapLayerFromProjectInstance = [ &ml, identifier ]
144 ml = project->
mapLayer( identifier );
152 if ( QThread::currentThread() == qApp->thread() )
153 getMapLayerFromProjectInstance();
155 QMetaObject::invokeMethod( qApp, std::move( getMapLayerFromProjectInstance ), Qt::BlockingQueuedConnection );
165 return QgsCoordinateReferenceSystem();
168 const bool isCrs = value.userType() == qMetaTypeId<QgsCoordinateReferenceSystem>();
170 if ( !isCrs && value.toString().isEmpty() )
172 return QgsCoordinateReferenceSystem();
175 QgsCoordinateReferenceSystem crs = isCrs ? value.value<QgsCoordinateReferenceSystem>() : QgsCoordinateReferenceSystem( value.toString() );
178 parent->setEvalErrorString( isCrs ? QObject::tr(
"Input CRS is invalid" )
179 : QObject::tr(
"Cannot convert '%1' to CRS" ).arg( value.toString() ) );
185QTimeZone QgsExpressionUtils::getTimeZoneValue(
const QVariant &value,
QgsExpression *parent )
194 if ( value.userType() == qMetaTypeId< QTimeZone>() )
197 tz = value.value<QTimeZone>();
203 : QObject::tr(
"Cannot convert '%1' to a time zone" ).arg( value.toString() ) );
213#ifndef __clang_analyzer__
219 ml = value.value< QgsMapLayer * >();
223 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
229 auto runFunction = [ layerPointer = QPointer< QgsMapLayer >( ml ), &function, &foundLayer ]
231 if ( QgsMapLayer *layer = layerPointer.data() )
241 if ( QThread::currentThread() == ml->thread() )
244 QMetaObject::invokeMethod( ml, std::move( runFunction ), Qt::BlockingQueuedConnection );
252 auto runFunction = [ value, context, expression, &function, &foundLayer ]
254 if ( QgsMapLayer *layer = getMapLayerPrivate( value, context, expression ) )
277 const QString identifier = value.toString();
280 const QList< QgsMapLayerStore * > stores = context->layerStores();
282 for ( QgsMapLayerStore *store : stores )
284 QPointer< QgsMapLayerStore > storePointer( store );
285 auto findLayerInStoreFunction = [ storePointer = std::move( storePointer ), identifier, function, &foundLayer ]
287 QgsMapLayer *ml =
nullptr;
288 if ( QgsMapLayerStore *store = storePointer.data() )
291 ml = store->mapLayer( identifier );
295 ml = store->mapLayersByName( identifier ).value( 0 );
308 if ( QThread::currentThread() == store->thread() )
309 findLayerInStoreFunction();
311 QMetaObject::invokeMethod( store, std::move( findLayerInStoreFunction ), Qt::BlockingQueuedConnection );
318 auto getMapLayerFromProjectInstance = [ value, identifier, &function, &foundLayer ]
323 QgsMapLayer *ml = project->
mapLayer( identifier );
339 getMapLayerFromProjectInstance();
341 QMetaObject::invokeMethod(
QgsProject::instance(), getMapLayerFromProjectInstance, Qt::BlockingQueuedConnection );
346QVariant QgsExpressionUtils::runMapLayerFunctionThreadSafe(
const QVariant &value,
const QgsExpressionContext *context,
QgsExpression *expression,
const std::function<QVariant(
QgsMapLayer * )> &function,
bool &foundLayer )
351 executeLambdaForMapLayer( value, context, expression, [&res, function]( QgsMapLayer * layer )
354 res = function( layer );
360std::unique_ptr<QgsVectorLayerFeatureSource> QgsExpressionUtils::getFeatureSource(
const QVariant &value,
const QgsExpressionContext *context,
QgsExpression *e,
bool &foundLayer )
362 std::unique_ptr<QgsVectorLayerFeatureSource> featureSource;
364 executeLambdaForMapLayer( value, context, e, [&featureSource]( QgsMapLayer * layer )
366 if ( QgsVectorLayer *vl = qobject_cast< QgsVectorLayer *>( layer ) )
368 featureSource = std::make_unique<QgsVectorLayerFeatureSource>( vl );
372 return featureSource;
377 return qobject_cast<QgsVectorLayer *>( getMapLayerPrivate( value, context, e ) );
385 if ( QgsMapLayer *layer = getMapLayer( value, context, parent ) )
388 res = parts.value( u
"path"_s ).toString();
393 res = value.toString();
419 for (
int i = 0; i < fields.
count(); i++ )
431 *foundFeatures = hasFeature;
435 const QVariant value = exp.
evaluate( &context );
438 return std::make_tuple(
static_cast<QMetaType::Type
>( value.userType() ), value.userType() );
443 return std::make_tuple(
static_cast<QMetaType::Type
>( value.userType() ), value.userType() );
@ NoGeometry
Geometry is not required. It may still be returned if e.g. required for a filter condition.
@ NoFlags
No flags are set.
@ Expression
Field is calculated from an expression.
Represents a coordinate reference system (CRS).
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
QList< QgsMapLayerStore * > layerStores() const
Returns the list of layer stores associated with the context.
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the context.
static std::tuple< QMetaType::Type, int > determineResultType(const QString &expression, const QgsVectorLayer *layer, const QgsFeatureRequest &request=QgsFeatureRequest(), const QgsExpressionContext &context=QgsExpressionContext(), bool *foundFeatures=nullptr)
Returns a value type and user type for a given expression.
Handles parsing and evaluation of expressions (formerly called "search strings").
void setEvalErrorString(const QString &str)
Sets evaluation error (used internally by evaluation functions).
bool needsGeometry() const
Returns true if the expression uses feature geometry for some computation.
QVariant evaluate()
Evaluate the feature and return the result.
Wrapper for iterator of features from vector data provider or vector layer.
bool nextFeature(QgsFeature &f)
Fetch next feature and stores in f, returns true on success.
Wraps a request for features to a vector layer (or directly its vector data provider).
QgsFeatureRequest & setFlags(Qgis::FeatureRequestFlags flags)
Sets flags that affect how features will be fetched.
QgsFeatureRequest & setLimit(long long limit)
Set the maximum number of features to request.
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsFeatureRequest & setExpressionContext(const QgsExpressionContext &context)
Sets the expression context used to evaluate filter expressions.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Container of fields for a vector layer.
Qgis::FieldOrigin fieldOrigin(int fieldIdx) const
Returns the field's origin (value from an enumeration).
Gradient color ramp, which smoothly interpolates between two colors and also supports optional extra ...
double value(int index) const override
Returns relative value between [0,1] of color at specified index.
Base class for all map layer types.
QString source() const
Returns the source for the layer.
QString providerType() const
Returns the provider type (provider key) for this layer.
static QgsProject * instance()
Returns the QgsProject singleton instance.
Q_INVOKABLE QgsMapLayer * mapLayer(const QString &layerId) const
Retrieve a pointer to a registered layer by layer ID.
Q_INVOKABLE QList< QgsMapLayer * > mapLayersByName(const QString &layerName) const
Retrieve a list of matching registered layers by layer name.
QVariantMap decodeUri(const QString &providerKey, const QString &uri)
Breaks a provider data source URI into its component paths (e.g.
static QgsProviderRegistry * instance(const QString &pluginPath=QString())
Means of accessing canonical single instance.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
Represents a vector layer which manages a vector based dataset.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const final
Queries the layer for features specified in request.
@ Unknown
Unknown/invalid format.
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
#define Q_NOWARN_DEPRECATED_POP
#define Q_NOWARN_DEPRECATED_PUSH
QList< int > QgsAttributeList
QPointer< QgsMapLayer > QgsWeakMapLayerPointer
Weak pointer for QgsMapLayer.