30QgsExpressionUtils::TVL QgsExpressionUtils::AND[3][3] =
33 { False, False, False },
37QgsExpressionUtils::TVL QgsExpressionUtils::OR[3][3] =
44QgsExpressionUtils::TVL QgsExpressionUtils::NOT[3] = { True, False,
Unknown };
47QColor QgsExpressionUtils::getColorValue(
const QVariant &value,
QgsExpression *parent,
bool &isQColor )
49 isQColor = value.userType() == QMetaType::Type::QColor;
50 QColor color = isQColor ? value.value<QColor>() : QgsSymbolLayerUtils::decodeColor( value.toString() );
51 if ( ! color.isValid() )
53 parent->setEvalErrorString( isQColor ? QObject::tr(
"Input color is invalid" )
54 : QObject::tr(
"Cannot convert '%1' to color" ).arg( value.toString() ) );
62 if ( value.userType() == qMetaTypeId<QgsGradientColorRamp>() )
63 return value.
value<QgsGradientColorRamp>();
67 parent->
setEvalErrorString( QObject::tr(
"Cannot convert '%1' to gradient ramp" ).arg( value.toString() ) );
69 return QgsGradientColorRamp();
74 return getMapLayerPrivate( value, context, parent );
83#ifdef __clang_analyzer__
89 ml = value.value< QgsMapLayer * >();
93 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
100 const QString identifier = value.toString();
105 const QList< QgsMapLayerStore * > stores = context->
layerStores();
106 for ( QgsMapLayerStore *store : stores )
108 auto findLayerInStoreFunction = [ storePointer = QPointer< QgsMapLayerStore >( store ), &ml, identifier ]
110 if ( QgsMapLayerStore *store = storePointer.data() )
113 ml = store->mapLayer( identifier );
119 ml = store->mapLayersByName( identifier ).value( 0 );
125 if ( QThread::currentThread() == store->thread() )
126 findLayerInStoreFunction();
128 QMetaObject::invokeMethod( store, std::move( findLayerInStoreFunction ), Qt::BlockingQueuedConnection );
135 auto getMapLayerFromProjectInstance = [ &ml, identifier ]
140 ml = project->
mapLayer( identifier );
148 if ( QThread::currentThread() == qApp->thread() )
149 getMapLayerFromProjectInstance();
151 QMetaObject::invokeMethod( qApp, std::move( getMapLayerFromProjectInstance ), Qt::BlockingQueuedConnection );
161 return QgsCoordinateReferenceSystem();
164 const bool isCrs = value.userType() == qMetaTypeId<QgsCoordinateReferenceSystem>();
166 if ( !isCrs && value.toString().isEmpty() )
168 return QgsCoordinateReferenceSystem();
171 QgsCoordinateReferenceSystem crs = isCrs ? value.value<QgsCoordinateReferenceSystem>() : QgsCoordinateReferenceSystem( value.toString() );
174 parent->setEvalErrorString( isCrs ? QObject::tr(
"Input CRS is invalid" )
175 : QObject::tr(
"Cannot convert '%1' to CRS" ).arg( value.toString() ) );
181QTimeZone QgsExpressionUtils::getTimeZoneValue(
const QVariant &value,
QgsExpression *parent )
190 if ( value.userType() == qMetaTypeId< QTimeZone>() )
193 tz = value.value<QTimeZone>();
199 : QObject::tr(
"Cannot convert '%1' to a time zone" ).arg( value.toString() ) );
209#ifndef __clang_analyzer__
215 ml = value.value< QgsMapLayer * >();
219 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
225 auto runFunction = [ layerPointer = QPointer< QgsMapLayer >( ml ), &function, &foundLayer ]
227 if ( QgsMapLayer *layer = layerPointer.data() )
237 if ( QThread::currentThread() == ml->thread() )
240 QMetaObject::invokeMethod( ml, std::move( runFunction ), Qt::BlockingQueuedConnection );
248 auto runFunction = [ value, context, expression, &function, &foundLayer ]
250 if ( QgsMapLayer *layer = getMapLayerPrivate( value, context, expression ) )
273 const QString identifier = value.toString();
276 const QList< QgsMapLayerStore * > stores = context->layerStores();
278 for ( QgsMapLayerStore *store : stores )
280 QPointer< QgsMapLayerStore > storePointer( store );
281 auto findLayerInStoreFunction = [ storePointer = std::move( storePointer ), identifier, function, &foundLayer ]
283 QgsMapLayer *ml =
nullptr;
284 if ( QgsMapLayerStore *store = storePointer.data() )
287 ml = store->mapLayer( identifier );
291 ml = store->mapLayersByName( identifier ).value( 0 );
304 if ( QThread::currentThread() == store->thread() )
305 findLayerInStoreFunction();
307 QMetaObject::invokeMethod( store, std::move( findLayerInStoreFunction ), Qt::BlockingQueuedConnection );
314 auto getMapLayerFromProjectInstance = [ value, identifier, &function, &foundLayer ]
319 QgsMapLayer *ml = project->
mapLayer( identifier );
335 getMapLayerFromProjectInstance();
337 QMetaObject::invokeMethod(
QgsProject::instance(), getMapLayerFromProjectInstance, Qt::BlockingQueuedConnection );
342QVariant QgsExpressionUtils::runMapLayerFunctionThreadSafe(
const QVariant &value,
const QgsExpressionContext *context,
QgsExpression *expression,
const std::function<QVariant(
QgsMapLayer * )> &function,
bool &foundLayer )
347 executeLambdaForMapLayer( value, context, expression, [&res, function]( QgsMapLayer * layer )
350 res = function( layer );
356std::unique_ptr<QgsVectorLayerFeatureSource> QgsExpressionUtils::getFeatureSource(
const QVariant &value,
const QgsExpressionContext *context,
QgsExpression *e,
bool &foundLayer )
358 std::unique_ptr<QgsVectorLayerFeatureSource> featureSource;
360 executeLambdaForMapLayer( value, context, e, [&featureSource]( QgsMapLayer * layer )
362 if ( QgsVectorLayer *vl = qobject_cast< QgsVectorLayer *>( layer ) )
364 featureSource = std::make_unique<QgsVectorLayerFeatureSource>( vl );
368 return featureSource;
373 return qobject_cast<QgsVectorLayer *>( getMapLayerPrivate( value, context, e ) );
381 if ( QgsMapLayer *layer = getMapLayer( value, context, parent ) )
384 res = parts.value( QStringLiteral(
"path" ) ).toString();
389 res = value.toString();
415 for (
int i = 0; i < fields.
count(); i++ )
427 *foundFeatures = hasFeature;
431 const QVariant value = exp.
evaluate( &context );
434 return std::make_tuple(
static_cast<QMetaType::Type
>( value.userType() ), value.userType() );
439 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.