30using namespace Qt::StringLiterals;
34QgsExpressionUtils::TVL QgsExpressionUtils::AND[3][3] = {
36 { False, False, False },
40QgsExpressionUtils::TVL QgsExpressionUtils::OR[3][3] = {
46QgsExpressionUtils::TVL QgsExpressionUtils::NOT[3] = { True, False,
Unknown };
49QColor QgsExpressionUtils::getColorValue(
const QVariant &value,
QgsExpression *parent,
bool &isQColor )
51 isQColor = value.userType() == QMetaType::Type::QColor;
52 QColor color = isQColor ? value.value<QColor>() : QgsSymbolLayerUtils::decodeColor( value.toString() );
53 if ( !color.isValid() )
55 parent->setEvalErrorString( isQColor ? QObject::tr(
"Input color is invalid" ) : QObject::tr(
"Cannot convert '%1' to color" ).arg( value.toString() ) );
63 if ( value.userType() == qMetaTypeId<QgsGradientColorRamp>() )
64 return value.
value<QgsGradientColorRamp>();
68 parent->
setEvalErrorString( QObject::tr(
"Cannot convert '%1' to gradient ramp" ).arg( value.toString() ) );
70 return QgsGradientColorRamp();
75 return getMapLayerPrivate( value, context, parent );
84#ifdef __clang_analyzer__
90 ml = value.value< QgsMapLayer * >();
94 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
101 const QString identifier = value.toString();
106 const QList< QgsMapLayerStore * > stores = context->
layerStores();
107 for ( QgsMapLayerStore *store : stores )
109 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] {
139 ml = project->
mapLayer( identifier );
147 if ( QThread::currentThread() == qApp->thread() )
148 getMapLayerFromProjectInstance();
150 QMetaObject::invokeMethod( qApp, std::move( getMapLayerFromProjectInstance ), Qt::BlockingQueuedConnection );
160 return QgsCoordinateReferenceSystem();
163 const bool isCrs = value.userType() == qMetaTypeId<QgsCoordinateReferenceSystem>();
165 if ( !isCrs && value.toString().isEmpty() )
167 return QgsCoordinateReferenceSystem();
170 QgsCoordinateReferenceSystem crs = isCrs ? value.value<QgsCoordinateReferenceSystem>() : QgsCoordinateReferenceSystem( value.toString() );
173 parent->setEvalErrorString( isCrs ? QObject::tr(
"Input CRS is invalid" ) : QObject::tr(
"Cannot convert '%1' to CRS" ).arg( value.toString() ) );
179QTimeZone QgsExpressionUtils::getTimeZoneValue(
const QVariant &value,
QgsExpression *parent )
188 if ( value.userType() == qMetaTypeId< QTimeZone>() )
191 tz = value.value<QTimeZone>();
196 parent->
setEvalErrorString( isTz ? QObject::tr(
"Input time zone is invalid" ) : QObject::tr(
"Cannot convert '%1' to a time zone" ).arg( value.toString() ) );
206#ifndef __clang_analyzer__
212 ml = value.value< QgsMapLayer * >();
216 qWarning(
"Raw map layer pointer stored in expression evaluation, switch to QgsWeakMapLayerPointer instead" );
222 auto runFunction = [layerPointer = QPointer< QgsMapLayer >( ml ), &function, &foundLayer] {
223 if ( QgsMapLayer *layer = layerPointer.data() )
233 if ( QThread::currentThread() == ml->thread() )
236 QMetaObject::invokeMethod( ml, std::move( runFunction ), Qt::BlockingQueuedConnection );
244 auto runFunction = [value, context, expression, &function, &foundLayer] {
245 if ( QgsMapLayer *layer = getMapLayerPrivate( value, context, expression ) )
268 const QString identifier = value.toString();
271 const QList< QgsMapLayerStore * > stores = context->layerStores();
273 for ( QgsMapLayerStore *store : stores )
275 QPointer< QgsMapLayerStore > storePointer( store );
276 auto findLayerInStoreFunction = [storePointer = std::move( storePointer ), identifier, function, &foundLayer] {
277 QgsMapLayer *ml =
nullptr;
278 if ( QgsMapLayerStore *store = storePointer.data() )
281 ml = store->mapLayer( identifier );
285 ml = store->mapLayersByName( identifier ).value( 0 );
298 if ( QThread::currentThread() == store->thread() )
299 findLayerInStoreFunction();
301 QMetaObject::invokeMethod( store, std::move( findLayerInStoreFunction ), Qt::BlockingQueuedConnection );
308 auto getMapLayerFromProjectInstance = [value, identifier, &function, &foundLayer] {
312 QgsMapLayer *ml = project->
mapLayer( identifier );
328 getMapLayerFromProjectInstance();
330 QMetaObject::invokeMethod(
QgsProject::instance(), getMapLayerFromProjectInstance, Qt::BlockingQueuedConnection );
335QVariant QgsExpressionUtils::runMapLayerFunctionThreadSafe(
342 executeLambdaForMapLayer(
346 [&res, function]( QgsMapLayer *layer ) {
348 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(
364 [&featureSource]( QgsMapLayer *layer ) {
365 if ( QgsVectorLayer *vl = qobject_cast< QgsVectorLayer *>( layer ) )
367 featureSource = std::make_unique<QgsVectorLayerFeatureSource>( vl );
373 return featureSource;
378 return qobject_cast<QgsVectorLayer *>( getMapLayerPrivate( value, context, e ) );
386 if ( QgsMapLayer *layer = getMapLayer( value, context, parent ) )
389 res = parts.value( u
"path"_s ).toString();
394 res = value.toString();
418 for (
int i = 0; i < fields.
count(); i++ )
430 *foundFeatures = hasFeature;
434 const QVariant value = exp.
evaluate( &context );
437 return std::make_tuple(
static_cast<QMetaType::Type
>( value.userType() ), value.userType() );
442 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.