35using namespace Qt::StringLiterals;
37#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
38#include <QThreadStorage>
46 const QMutexLocker locker( &layer->mFeatureSourceConstructorMutex );
63 source.
joinSource = std::make_shared< QgsVectorLayerFeatureSource >( joinLayer );
77 if ( request.filterType() == QgsFeatureRequest::FilterFid )
81 if ( L->editBuffer()->addedFeatures().contains( request.filterFid() ) )
82 mAddedFeatures.insert( request.filterFid(), L->editBuffer()->addedFeatures()[ request.filterFid()] );
84 if ( L->editBuffer()->changedGeometries().contains( request.filterFid() ) )
85 mChangedGeometries.insert( request.filterFid(), L->editBuffer()->changedGeometries()[ request.filterFid()] );
87 if ( L->editBuffer()->deletedFeatureIds().contains( request.filterFid() ) )
90 if ( L->editBuffer()->changedAttributeValues().contains( request.filterFid() ) )
91 mChangedAttributeValues.insert( request.filterFid(), L->editBuffer()->changedAttributeValues()[ request.filterFid()] );
93 if ( L->editBuffer()->changedAttributeValues().contains( request.filterFid() ) )
94 mChangedFeaturesRequest.setFilterFids(
QgsFeatureIds() << request.filterFid() );
152 switch (
mRequest.spatialFilterType() )
154 case Qgis::SpatialFilterType::NoFilter:
155 case Qgis::SpatialFilterType::BoundingBox:
158 case Qgis::SpatialFilterType::DistanceWithin:
159 if ( !mRequest.referenceGeometry().isEmpty() )
165 mDistanceWithinGeom = mRequest.referenceGeometry();
166 mDistanceWithinEngine = mRequest.referenceGeometryEngine();
167 mDistanceWithinEngine->prepareGeometry();
168 mDistanceWithin = mRequest.distanceWithin();
173 bool canDelegateLimitToProvider =
true;
176 switch ( updateRequestToSourceCrs( mRequest, mTransform ) )
184 canDelegateLimitToProvider =
false;
189 mFilterRect = mRequest.filterRect();
199 mDelegatedOrderByToProvider = !mSource->mHasEditBuffer;
200 if ( !mRequest.orderBy().isEmpty() )
202 QSet<int> attributeIndexes;
203 const auto usedAttributeIndices = mRequest.orderBy().usedAttributeIndices( mSource->mFields );
204 for ( const int attrIndex : usedAttributeIndices )
206 if ( mSource->mFields.fieldOrigin( attrIndex ) != Qgis::FieldOrigin::Provider )
207 mDelegatedOrderByToProvider = false;
209 attributeIndexes << attrIndex;
214 attributeIndexes += qgis::listToSet( mRequest.subsetOfAttributes() );
215 mRequest.setSubsetOfAttributes( qgis::setToList( attributeIndexes ) );
221 mRequest.expressionContext()->setFields( mSource->mFields );
222 mRequest.filterExpression()->prepare( mRequest.expressionContext() );
224 if ( mRequest.flags() & Qgis::FeatureRequestFlag::SubsetOfAttributes )
227 QSet<int> attributeIndexes = mRequest.filterExpression()->referencedAttributeIndexes( mSource->mFields );
228 attributeIndexes += qgis::listToSet( mRequest.subsetOfAttributes() );
229 mRequest.setSubsetOfAttributes( qgis::setToList( attributeIndexes ) );
235 mHasVirtualAttributes = !mFetchJoinInfo.isEmpty() || !mExpressionFieldInfo.isEmpty();
238 mProviderRequest = mRequest;
242 if ( mRequest.coordinateTransform().isValid() || mRequest.destinationCrs().isValid() )
248 if ( !mDelegatedOrderByToProvider )
253 if ( !canDelegateLimitToProvider )
255 mProviderRequest.setLimit( -1 );
261 QSet<int> providerSubset;
263 const int nPendingFields = mSource->mFields.count();
264 for (
const int attrIndex : subset )
266 if ( attrIndex < 0 || attrIndex >= nPendingFields )
269 providerSubset << mSource->mFields.fieldOriginIndex( attrIndex );
277 if ( !mProviderRequest.orderBy().isEmpty() )
279 const auto usedAttributeIndices = mProviderRequest.orderBy().usedAttributeIndices( mSource->mFields );
280 for (
const int attrIndex : usedAttributeIndices )
282 providerSubset << attrIndex;
286 mProviderRequest.setSubsetOfAttributes( qgis::setToList( providerSubset ) );
291 const bool needsGeom = mProviderRequest.filterExpression()->needsGeometry();
292 const auto constReferencedColumns = mProviderRequest.filterExpression()->referencedColumns();
293 for (
const QString &field : constReferencedColumns )
295 const int idx = source->mFields.lookupField( field );
301 mProviderRequest.disableFilter();
303 mProviderRequest.setLimit( -1 );
314 if ( mSource->mHasEditBuffer )
316 mChangedFeaturesRequest = mProviderRequest;
318 QgsChangedAttributesMap::const_iterator attIt = mSource->mChangedAttributeValues.constBegin();
319 for ( ; attIt != mSource->mChangedAttributeValues.constEnd(); ++attIt )
321 changedIds << attIt.key();
323 mChangedFeaturesRequest.setFilterFids( changedIds );
325 if ( mChangedFeaturesRequest.limit() > 0 )
327 int providerLimit = mProviderRequest.limit();
330 providerLimit += mSource->mDeletedFeatureIds.size();
335 providerLimit += mSource->mChangedAttributeValues.size();
341 providerLimit += mSource->mChangedGeometries.size();
344 mProviderRequest.setLimit( providerLimit );
345 mChangedFeaturesRequest.setLimit( providerLimit );
355 if ( mSource->mProviderFeatureSource )
357 if ( mSource->mHasEditBuffer )
359 mChangedFeaturesIterator = mSource->mProviderFeatureSource->getFeatures( mChangedFeaturesRequest );
363 mProviderIterator = mSource->mProviderFeatureSource->getFeatures( mProviderRequest );
386class QgsThreadStackOverflowGuard
389#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
390 QgsThreadStackOverflowGuard( QThreadStorage<std::deque<QString>> &storage,
const QString &stackFrameInformation,
int maxDepth )
392 QgsThreadStackOverflowGuard( std::deque<QString> &storage,
const QString &stackFrameInformation,
int maxDepth )
394 : mStorage( storage )
395 , mMaxDepth( maxDepth )
397#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
398 if ( !storage.hasLocalData() )
400 storage.setLocalData( std::deque<QString>() );
403 storage.localData().emplace_back( stackFrameInformation );
405 storage.emplace_back( stackFrameInformation );
409 ~QgsThreadStackOverflowGuard()
411#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
412 mStorage.localData().pop_back();
418 bool hasStackOverflow()
const
420#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
421 if ( mStorage.localData().size() > mMaxDepth )
423 if ( mStorage.size() > mMaxDepth )
430 QString topFrames()
const
432 QStringList dumpStack;
433#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
434 const std::deque<QString> &stack = mStorage.localData();
436 const std::deque<QString> &stack = mStorage;
439 const int dumpSize = std::min(
static_cast<int>( stack.size() ), 10 );
440 auto stackIt = stack.begin();
441 for (
int i = 0; i < dumpSize; ++i, stackIt++ )
443 dumpStack += *stackIt;
446 return dumpStack.join(
'\n' );
449 std::size_t depth()
const
451#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
452 return mStorage.localData().size();
454 return mStorage.size();
459#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
460 QThreadStorage<std::deque<QString>> &mStorage;
462 std::deque<QString> &mStorage;
464 std::size_t mMaxDepth;
476#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
477 static QThreadStorage<std::deque<QString>> sStack;
479 static thread_local std::deque<QString> sStack;
482 const QgsThreadStackOverflowGuard guard( sStack,
mSource->id(), 4 );
484 if ( guard.hasStackOverflow() )
495 if ( res && postProcessFeature( f ) )
534 if (
mSource->mProviderFeatureSource )
561 mRequest.expressionContext()->setFeature( f );
562 if ( !
mRequest.filterExpression()->evaluate(
mRequest.expressionContext() ).toBool() )
574 if ( !postProcessFeature( f ) )
620 mInterruptionChecker = interruptionChecker;
647 if ( !postProcessFeature( f ) )
695 mRequest.expressionContext()->setFeature( f );
696 if ( !
mRequest.filterExpression()->evaluate(
mRequest.expressionContext() ).toBool() )
702 if ( postProcessFeature( f ) )
725 const auto changedGeometryIt =
mSource->mChangedGeometries.constFind( f.
id() );
726 if ( changedGeometryIt !=
mSource->mChangedGeometries.constEnd() )
738 mRequest.expressionContext()->setFeature( f );
739 if (
mRequest.filterExpression()->evaluate(
mRequest.expressionContext() ).toBool() && postProcessFeature( f ) )
761 if ( !subsetAttrs || !
mRequest.subsetOfAttributes().isEmpty() )
795 if ( !
mSource->mFields.exists( fieldIdx ) )
801 int sourceLayerIndex;
803 Q_ASSERT( joinInfo );
806 if ( joinSourceIt ==
mSource->mJoinSources.constEnd() )
830 mFetchJoinInfo[joinInfo].attributes.push_back( sourceLayerIndex );
831 mFetchJoinInfo[joinInfo].attributesSourceToDestLayerMap[sourceLayerIndex] = fieldIdx;
837#if !defined( USE_THREAD_LOCAL ) || defined( Q_OS_WIN )
838 static QThreadStorage<std::deque<QString>> sStack;
840 static thread_local std::deque<QString> sStack;
843 const QgsThreadStackOverflowGuard guard( sStack,
mSource->id(), 4 );
845 if ( guard.hasStackOverflow() )
852 const QList<QgsExpressionFieldBuffer::ExpressionField> &exps =
mSource->mExpressionFieldBuffer->expressions();
854 const int oi =
mSource->mFields.fieldOriginIndex( fieldIdx );
855 auto exp = std::make_unique<QgsExpression>( exps[oi].cachedExpression );
860 exp->setGeomCalculator( &da );
864 if ( !mExpressionContext )
865 createExpressionContext();
866 exp->prepare( mExpressionContext.get() );
867 const QSet<int> referencedColumns = exp->referencedAttributeIndexes(
mSource->fields() );
869 QSet<int> requestedAttributes = qgis::listToSet(
mRequest.subsetOfAttributes() );
871 for (
const int dependentFieldIdx : referencedColumns )
875 requestedAttributes += dependentFieldIdx;
878 if ( !mPreparedFields.contains( dependentFieldIdx ) && !mFieldsToPrepare.contains( dependentFieldIdx ) )
879 mFieldsToPrepare << dependentFieldIdx;
884 mRequest.setSubsetOfAttributes( qgis::setToList( requestedAttributes ) );
887 if ( exp->needsGeometry() )
897 mPreparedFields.clear();
898 mFieldsToPrepare.clear();
900 mOrderedJoinInfoList.clear();
902 mExpressionContext.reset();
906 while ( !mFieldsToPrepare.isEmpty() )
908 const int fieldIdx = mFieldsToPrepare.takeFirst();
909 if ( mPreparedFields.contains( fieldIdx ) )
912 mPreparedFields << fieldIdx;
919 createOrderedJoinList();
923void QgsVectorLayerFeatureIterator::createOrderedJoinList()
926 if ( mOrderedJoinInfoList.size() < 2 )
931 QSet<int> resolvedFields;
934 QList< int >::const_iterator prepFieldIt = mPreparedFields.constBegin();
935 for ( ; prepFieldIt != mPreparedFields.constEnd(); ++prepFieldIt )
939 resolvedFields.insert( *prepFieldIt );
946 const int maxIterations = ( mOrderedJoinInfoList.size() + 1 ) * mOrderedJoinInfoList.size() / 2.0;
947 int currentIteration = 0;
949 for (
int i = 0; i < mOrderedJoinInfoList.size() - 1; ++i )
951 if ( !resolvedFields.contains( mOrderedJoinInfoList.at( i ).targetField ) )
953 mOrderedJoinInfoList.append( mOrderedJoinInfoList.at( i ) );
954 mOrderedJoinInfoList.removeAt( i );
959 const int offset = mOrderedJoinInfoList.at( i ).indexOffset;
960 const int joinField = mOrderedJoinInfoList.at( i ).joinField;
962 const QgsAttributeList attributes = mOrderedJoinInfoList.at( i ).attributes;
963 for (
int n = 0; n < attributes.size(); n++ )
965 if ( n != joinField )
967 resolvedFields.insert( joinField < n ? n + offset - 1 : n + offset );
973 if ( currentIteration >= maxIterations )
980bool QgsVectorLayerFeatureIterator::postProcessFeature(
QgsFeature &feature )
982 bool result = checkGeometryValidity( feature );
994bool QgsVectorLayerFeatureIterator::checkGeometryValidity(
const QgsFeature &feature )
996 switch (
mRequest.invalidGeometryCheck() )
1009 if (
mRequest.invalidGeometryCallback() )
1011 mRequest.invalidGeometryCallback()( feature );
1026 if (
mRequest.invalidGeometryCallback() )
1028 mRequest.invalidGeometryCallback()( feature );
1040 switch (
mSource->mFields.fieldOrigin( fieldIdx ) )
1047 if (
mSource->mJoinBuffer->containsJoins() )
1062 QList< FetchJoinInfo >::const_iterator joinIt = mOrderedJoinInfoList.constBegin();
1063 for ( ; joinIt != mOrderedJoinInfoList.constEnd(); ++joinIt )
1065 const QVariant targetFieldValue = f.
attribute( joinIt->targetField );
1066 if ( !targetFieldValue.isValid() )
1069 const QHash< QString, QgsAttributes> &memoryCache = joinIt->joinInfo->cachedAttributes;
1070 if ( memoryCache.isEmpty() )
1071 joinIt->addJoinedAttributesDirect( f, targetFieldValue );
1073 joinIt->addJoinedAttributesCached( f, targetFieldValue );
1081 attr.resize(
mSource->mFields.count() );
1087 QList< int > fetchedVirtualAttributes;
1089 QMap<const QgsVectorLayerJoinInfo *, FetchJoinInfo>::const_iterator joinIt =
mFetchJoinInfo.constBegin();
1096 fetchedVirtualAttributes << joinIt->targetField;
1109 if ( fetchedVirtualAttributes.contains( it.key() ) )
1122 if ( !mExpressionContext )
1123 createExpressionContext();
1125 mExpressionContext->setFeature( f );
1126 QVariant val = exp->
evaluate( mExpressionContext.get() );
1127 ( void )
mSource->mFields.at( attrIndex ).convertCompatible( val );
1138 Q_UNUSED( simplifyMethod )
1144 Q_UNUSED( methodType )
1151 const QHash<QString, QgsAttributes> &memoryCache =
joinInfo->cachedAttributes;
1152 const QHash<QString, QgsAttributes>::const_iterator it = memoryCache.find( joinValue.toString() );
1153 if ( it == memoryCache.constEnd() )
1159 for (
int i = 0; i < featureAttributes.count(); ++i )
1170 if ( joinLayer && ! joinLayer->hasFeatures() )
1177 QString subsetString;
1179 const QString joinFieldName =
joinInfo->joinFieldName();
1181 subsetString.append( u
"\"%1\""_s.arg( joinFieldName ) );
1185 subsetString +=
" IS NULL"_L1;
1189 QString v = joinValue.toString();
1190 switch ( joinValue.userType() )
1192 case QMetaType::Type::Int:
1193 case QMetaType::Type::LongLong:
1194 case QMetaType::Type::Double:
1198 case QMetaType::Type::QString:
1199 v.replace(
'\'',
"''"_L1 );
1200 v.prepend(
'\'' ).append(
'\'' );
1203 subsetString +=
'=' + v;
1206 QList<int> joinedAttributeIndices;
1214 joinedAttributeIndices = qgis::setToList( qgis::listToSet(
attributes ).intersect( qgis::listToSet( subsetIndices.toList() ) ) );
1222 joinedAttributeIndices.removeAll(
joinField );
1240 for (
const int sourceAttrIndex : sourceAttrIndexes )
1247 f.
setAttribute( destAttrIndex, attr.at( sourceAttrIndex ) );
1262 if (
mSource->mDeletedFeatureIds.contains( featureId ) )
1273 for ( QgsFeatureMap::ConstIterator iter =
mSource->mAddedFeatures.constBegin(); iter !=
mSource->mAddedFeatures.constEnd(); ++iter )
1275 if ( iter->id() == featureId )
1288 if (
mSource->mHasEditBuffer )
1307 for (
int idx =
mSource->mDeletedAttributeIds.count() - 1; idx >= 0; --idx )
1309 attrs.remove(
mSource->mDeletedAttributeIds[idx] );
1313 attrs.resize( attrs.count() +
mSource->mAddedAttributes.count() );
1316 if (
mSource->mChangedAttributeValues.contains( f.
id() ) )
1319 for ( QgsAttributeMap::const_iterator it = map.begin(); it != map.end(); ++it )
1320 attrs[it.key()] = it.value();
1327 if (
mSource->mChangedGeometries.contains( f.
id() ) )
1331void QgsVectorLayerFeatureIterator::createExpressionContext()
1333 mExpressionContext = std::make_unique< QgsExpressionContext >();
1340bool QgsVectorLayerFeatureIterator::prepareOrderBy(
const QList<QgsFeatureRequest::OrderByClause> &orderBys )
1342 Q_UNUSED( orderBys )
1343 return mDelegatedOrderByToProvider;
1353 , mSelectedFeatureIds( layer->selectedFeatureIds() )
1354 , mWkbType( layer->
wkbType() )
1355 , mName( layer->name() )
1380 return QgsFeatureIterator(
new QgsVectorLayerSelectedFeatureIterator( mSelectedFeatureIds, req, mSource ) );
1385 return mSource.crs();
1390 return mSource.fields();
1400 return mSelectedFeatureIds.count();
1411 return mLayer->createExpressionContextScope();
1419 return mLayer->hasSpatialIndex();
1431 , mSelectedFeatureIds( selectedFeatureIds )
1439 sourceRequest.setLimit( -1 );
1444bool QgsVectorLayerSelectedFeatureIterator::rewind()
1446 return mIterator.
rewind();
1449bool QgsVectorLayerSelectedFeatureIterator::close()
1451 return mIterator.close();
1454bool QgsVectorLayerSelectedFeatureIterator::fetchFeature(
QgsFeature &f )
1456 while ( mIterator.nextFeature( f ) )
1458 if ( mSelectedFeatureIds.contains( f.
id() ) )
@ Fid
Filter using feature ID.
@ Expression
Filter using expression.
@ NoFilter
No filter is applied.
SpatialIndexPresence
Enumeration of spatial index presence states.
@ Unknown
Spatial index presence cannot be determined, index may or may not exist.
@ SubsetOfAttributes
Fetch only a subset of attributes (setSubsetOfAttributes sets this flag).
@ NoGeometry
Geometry is not required. It may still be returned if e.g. required for a filter condition.
@ Critical
Critical/error message.
@ NoFilter
No spatial filtering of features.
@ Provider
Field originates from the underlying data provider of the vector layer.
@ Edit
Field has been temporarily added in editing mode.
@ Unknown
The field origin has not been specified.
@ Expression
Field is calculated from an expression.
@ Join
Field originates from a joined layer.
@ NoCheck
No invalid geometry checking.
@ AbortOnInvalid
Close iterator on encountering any features with invalid geometry. This requires a slow geometry vali...
@ SkipInvalid
Skip any features with invalid geometry. This requires a slow geometry validity check for every featu...
WkbType
The WKB type describes the number of dimensions a geometry has.
QgsVectorLayerFeatureSource * mSource
QgsAbstractFeatureIteratorFromSource(QgsVectorLayerFeatureSource *source, bool ownSource, const QgsFeatureRequest &request)
Internal feature iterator to be implemented within data providers.
@ Success
Request was successfully updated to the source CRS, or no changes were required.
@ DistanceWithinMustBeCheckedManually
The distance within request cannot be losslessly updated to the source CRS, and callers will need to ...
friend class QgsFeatureIterator
void geometryToDestinationCrs(QgsFeature &feature, const QgsCoordinateTransform &transform) const
Transforms feature's geometry according to the specified coordinate transform.
QgsFeatureRequest mRequest
A copy of the feature request.
bool mClosed
Sets to true, as soon as the iterator is closed.
Represents a coordinate reference system (CRS).
Custom exception class for Coordinate Reference System related exceptions.
A general purpose distance and area calculator, capable of performing ellipsoid based calculations.
void setSourceCrs(const QgsCoordinateReferenceSystem &crs, const QgsCoordinateTransformContext &context)
Sets source spatial reference system crs.
bool setEllipsoid(const QString &ellipsoid)
Sets the ellipsoid by its acronym.
Single scope for storing variables and functions for use within a QgsExpressionContext.
static QgsExpressionContextScope * projectScope(const QgsProject *project)
Creates a new scope which contains variables and functions relating to a QGIS project.
static QgsExpressionContextScope * layerScope(const QgsMapLayer *layer)
Creates a new scope which contains variables and functions relating to a QgsMapLayer.
static QgsExpressionContextScope * globalScope()
Creates a new scope which contains variables and functions relating to the global QGIS context.
Handles parsing and evaluation of expressions (formerly called "search strings").
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.
bool rewind()
Resets the iterator to the starting position.
Represents a list of OrderByClauses, with the most important first and the least important last.
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.
QgsFeedback * feedback() const
Returns the feedback object that can be queried regularly by the iterator to check if it should be ca...
QgsFeatureRequest & setRequestMayBeNested(bool requestMayBeNested)
In case this request may be run nested within another already running iteration on the same connectio...
long long limit() const
Returns the maximum number of features to request, or -1 if no limit set.
QgsFeatureRequest & setFilterFids(const QgsFeatureIds &fids)
Sets the feature IDs that should be fetched.
Qgis::FeatureRequestFilterType filterType() const
Returns the attribute/ID filter type which is currently set on this request.
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsFeatureRequest & setFilterExpression(const QString &expression)
Set the filter expression.
const QgsFeatureIds & filterFids() const
Returns the feature IDs that should be fetched.
QgsFeatureRequest & setFilterFid(QgsFeatureId fid)
Sets the feature ID that should be fetched.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Q_INVOKABLE bool setAttribute(int field, const QVariant &attr)
Sets an attribute's value by field index.
void setAttributes(const QgsAttributes &attrs)
Sets the feature's attributes.
void setFields(const QgsFields &fields, bool initAttributes=false)
Assigns a field map with the feature to allow attribute access by attribute name.
int attributeCount() const
Returns the number of attributes attached to the feature.
void padAttributes(int count)
Resizes the attributes attached to this feature by appending the specified count of NULL values to th...
void setId(QgsFeatureId id)
Sets the feature id for this feature.
void setValid(bool validity)
Sets the validity of the feature.
bool hasGeometry() const
Returns true if the feature has an associated geometry.
Q_INVOKABLE QVariant attribute(const QString &name) const
Lookup attribute value by attribute name.
void setGeometry(const QgsGeometry &geometry)
Set the feature's geometry.
Base class for feedback objects to be used for cancellation of something running in a worker thread.
Container of fields for a vector layer.
A geometry is the spatial representation of a feature.
const QgsAbstractGeometry * constGet() const
Returns a non-modifiable (const) reference to the underlying abstract geometry primitive.
bool isGeosValid(Qgis::GeometryValidityFlags flags=Qgis::GeometryValidityFlags()) const
Checks validity of the geometry using GEOS.
QgsCoordinateReferenceSystem crs
static void logMessage(const QString &message, const QString &tag=QString(), Qgis::MessageLevel level=Qgis::MessageLevel::Warning, bool notifyUser=true, const char *file=__builtin_FILE(), const char *function=__builtin_FUNCTION(), int line=__builtin_LINE(), Qgis::StringFormat format=Qgis::StringFormat::PlainText)
Adds a message to the log instance (and creates it if necessary).
static QgsProject * instance()
Returns the QgsProject singleton instance.
Contains information about how to simplify geometries fetched from a QgsFeatureIterator.
static bool isNull(const QVariant &variant, bool silenceNullWarnings=false)
Returns true if the specified variant should be considered a NULL value.
Base class for vector data providers.
virtual QgsAbstractFeatureSource * featureSource() const =0
Returns feature source object that can be used for querying provider's data.
virtual QgsTransaction * transaction() const
Returns the transaction this data provider is included in, if any.
QgsFeatureIds deletedFeatureIds() const
Returns a list of deleted feature IDs which are not committed.
QgsChangedAttributesMap changedAttributeValues() const
Returns a map of features with changed attributes values which are not committed.
QgsFeatureMap addedFeatures() const
Returns a map of new features which are not committed.
QgsGeometryMap changedGeometries() const
Returns a map of features with changed geometries which are not committed.
QList< QgsField > addedAttributes() const
Returns a list of added attributes fields which are not committed.
QgsAttributeList deletedAttributeIds() const
Returns a list of deleted attributes fields which are not committed.
void updateChangedAttributes(QgsFeature &f)
Update feature with uncommitted attribute updates.
void prepareExpression(int fieldIdx)
void useAddedFeature(const QgsFeature &src, QgsFeature &f)
QMap< const QgsVectorLayerJoinInfo *, QgsVectorLayerFeatureIterator::FetchJoinInfo > mFetchJoinInfo
Information about joins used in the current select() statement.
void prepareField(int fieldIdx)
QgsFeatureMap::ConstIterator mFetchAddedFeaturesIt
bool mHasVirtualAttributes
QgsFeatureRequest mProviderRequest
QSet< QgsFeatureId > mFetchConsidered
void setInterruptionChecker(QgsFeedback *interruptionChecker) override
Attach an object that can be queried regularly by the iterator to check if it must stopped.
void addVirtualAttributes(QgsFeature &f)
Adds attributes that don't source from the provider but are added inside QGIS Includes.
bool close() override
end of iterating: free the resources / lock
void updateFeatureGeometry(QgsFeature &f)
Update feature with uncommitted geometry updates.
bool nextFeatureFid(QgsFeature &f)
QgsFeatureIterator mChangedFeaturesIterator
bool fetchNextAddedFeature(QgsFeature &f)
void addExpressionAttribute(QgsFeature &f, int attrIndex)
Adds an expression based attribute to a feature.
QMap< int, QgsExpression * > mExpressionFieldInfo
QgsGeometryMap::ConstIterator mFetchChangedGeomIt
bool prepareSimplification(const QgsSimplifyMethod &simplifyMethod) override
Setup the simplification of geometries to fetch using the specified simplify method.
void prepareJoin(int fieldIdx)
void useChangedAttributeFeature(QgsFeatureId fid, const QgsGeometry &geom, QgsFeature &f)
bool fetchFeature(QgsFeature &feature) override
fetch next feature, return true on success
~QgsVectorLayerFeatureIterator() override
std::shared_ptr< QgsGeometryEngine > mDistanceWithinEngine
void addJoinedAttributes(QgsFeature &f)
bool isValid() const override
Returns if this iterator is valid.
bool rewind() override
reset the iterator to the starting position
bool fetchNextChangedGeomFeature(QgsFeature &f)
QgsFeatureIterator mProviderIterator
QgsCoordinateTransform mTransform
bool fetchNextChangedAttributeFeature(QgsFeature &f)
QgsVectorLayerFeatureIterator(QgsVectorLayerFeatureSource *source, bool ownSource, const QgsFeatureRequest &request)
Partial snapshot of vector layer's state (only the members necessary for access to features).
QgsCoordinateReferenceSystem crs() const
Returns the coordinate reference system for features retrieved from this source.
QgsChangedAttributesMap mChangedAttributeValues
QgsCoordinateReferenceSystem mCrs
QList< QgsField > mAddedAttributes
std::unique_ptr< QgsAbstractFeatureSource > mProviderFeatureSource
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) override
Gets an iterator for features matching the specified request.
QgsVectorLayerFeatureSource(const QgsVectorLayer *layer)
Constructor for QgsVectorLayerFeatureSource.
~QgsVectorLayerFeatureSource() override
QgsFeatureIds mDeletedFeatureIds
QString id() const
Returns the layer id of the source layer.
QgsAttributeList mDeletedAttributeIds
QgsExpressionContextScope mLayerScope
QgsFields fields() const
Returns the fields that will be available for features that are retrieved from this source.
std::unique_ptr< QgsVectorLayerJoinBuffer > mJoinBuffer
std::unique_ptr< QgsExpressionFieldBuffer > mExpressionFieldBuffer
QMap< QString, JoinLayerSource > mJoinSources
Contains prepared join sources by layer ID.
friend class QgsVectorLayerFeatureIterator
QgsFeatureMap mAddedFeatures
QgsGeometryMap mChangedGeometries
bool containsJoins() const
Quick way to test if there is any join at all.
QgsVectorLayerJoinBuffer * clone() const
Create a copy of the join buffer.
void createJoinCaches()
Calls cacheJoinLayer() for all vector joins.
static QVector< int > joinSubsetIndices(QgsVectorLayer *joinLayer, const QStringList &joinFieldsSubset)
Returns a vector of indices for use in join based on field names from the layer.
Defines left outer join from our vector layer to some other vector layer.
QStringList * joinFieldNamesSubset() const
Returns the subset of fields to be used from joined layer.
QString joinFieldName() const
Returns name of the field of joined layer that will be used for join.
QString targetFieldName() const
Returns name of the field of our layer that will be used for join.
QString joinLayerId() const
ID of the joined layer - may be used to resolve reference to the joined layer.
long long featureCount() const override
Returns the number of features contained in the source, or -1 if the feature count is unknown.
Qgis::WkbType wkbType() const override
Returns the geometry type for features returned by this source.
Qgis::SpatialIndexPresence hasSpatialIndex() const override
Returns an enum value representing the presence of a valid spatial index on the source,...
QgsFields fields() const override
Returns the fields associated with features in the source.
QgsCoordinateReferenceSystem sourceCrs() const override
Returns the coordinate reference system for features in the source.
QgsVectorLayerSelectedFeatureSource(QgsVectorLayer *layer)
Constructor for QgsVectorLayerSelectedFeatureSource, for selected features from the specified layer.
QgsExpressionContextScope * createExpressionContextScope() const override
This method needs to be reimplemented in all classes which implement this interface and return an exp...
QString sourceName() const override
Returns a friendly display name for the source.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const override
Returns an iterator for the features in the source.
Represents a vector layer which manages a vector based dataset.
Q_INVOKABLE QgsVectorLayerEditBuffer * editBuffer()
Buffer with uncommitted editing operations. Only valid after editing has been turned on.
QgsVectorDataProvider * dataProvider() final
Returns the layer's data provider, it may be nullptr.
QMap< int, QVariant > QgsAttributeMap
QMap< QgsFeatureId, QgsGeometry > QgsGeometryMap
QMap< QgsFeatureId, QgsAttributeMap > QgsChangedAttributesMap
QSet< QgsFeatureId > QgsFeatureIds
qint64 QgsFeatureId
64 bit feature ids negative numbers are used for uncommitted/newly added features
QList< int > QgsAttributeList
QMap< QgsFeatureId, QgsFeature > QgsFeatureMap
Join information prepared for fast attribute id mapping in QgsVectorLayerJoinBuffer::updateFeatureAtt...
QMap< int, int > attributesSourceToDestLayerMap
Mapping from original attribute index to the joined layer index.
void addJoinedAttributesDirect(QgsFeature &f, const QVariant &joinValue) const
std::shared_ptr< QgsVectorLayerFeatureSource > joinSource
Feature source for join.
QgsFields joinLayerFields
Fields from joined layer.
int targetField
Index of field (of this layer) that drives the join.
const QgsVectorLayerJoinInfo * joinInfo
Canonical source of information about the join.
QgsAttributeList attributes
Attributes to fetch.
int indexOffset
At what position the joined fields start.
void addJoinedAttributesCached(QgsFeature &f, const QVariant &joinValue) const
Contains join layer source information prepared in a thread-safe way, ready for vector layer feature ...
std::shared_ptr< QgsVectorLayerFeatureSource > joinSource
Feature source for join.
QgsFields joinLayerFields
Fields from joined layer.