26#include <QElapsedTimer>
28#include "moc_qgsvectorlayercache.cpp"
38 connect( mLayer, &QgsVectorLayer::destroyed,
this, &QgsVectorLayerCache::layerDeleted );
49 connectJoinedLayers();
54 qDeleteAll( mCacheIndices );
55 mCacheIndices.clear();
65 return mCache.maxCost();
70 bool shouldCacheGeometry =
cacheGeometry && mLayer->isSpatial();
71 bool mustInvalidate = shouldCacheGeometry && !mCacheGeometry;
72 mCacheGeometry = shouldCacheGeometry;
89 if ( ! mCache.isEmpty() && ! QSet<int>( mCachedAttributes.cbegin(), mCachedAttributes.cend() )
90 .contains( QSet<int>( attributes.cbegin(), attributes.cend( ) ) ) )
92 mCachedAttributes = attributes;
97 return mCachedAttributes;
102 mFullCache = fullCache;
111 .setSubsetOfAttributes( mCachedAttributes )
124 if ( t.elapsed() > 1000 )
143 mCacheIndices.append( cacheIndex );
148 if ( cacheAddedAttributes )
160 bool featureFound =
false;
162 QgsCachedFeature *cachedFeature =
nullptr;
166 cachedFeature = mCache[ featureId ];
171 feature =
QgsFeature( *cachedFeature->feature() );
177 const bool allAttrsFetched { mCachedAttributes.count( ) == mLayer->fields().count() };
178 if ( ! allAttrsFetched )
182 if ( !mCacheGeometry )
186 if ( mLayer->getFeatures( request ).nextFeature( feature ) )
188 cacheFeature( feature, allAttrsFetched );
199 bool featureFound =
false;
201 QgsCachedFeature *cachedFeature =
nullptr;
205 cachedFeature = mCache[ featureId ];
208 if ( cachedFeature && cachedFeature->allAttributesFetched() )
210 feature =
QgsFeature( *cachedFeature->feature() );
214 .setFilterFid( featureId )
216 .nextFeature( feature ) )
218 cacheFeature( feature,
true );
227 bool featureFound =
false;
229 QgsCachedFeature *cachedFeature =
nullptr;
233 cachedFeature = mCache[ featureId ];
236 if ( cachedFeature && cachedFeature->allAttributesFetched() && cachedFeature->geometryFetched() )
238 feature =
QgsFeature( *cachedFeature->feature() );
242 .setFilterFid( featureId ) )
243 .nextFeature( feature ) )
245 cacheFeature( feature,
true,
true );
254 bool removed = mCache.remove( fid );
257 if (
auto unorderedIt = std::find( mCacheUnorderedKeys.begin(), mCacheUnorderedKeys.end(), fid ); unorderedIt != mCacheUnorderedKeys.end() )
259 mCacheUnorderedKeys.erase( unorderedIt );
261 if (
auto orderedIt = std::find( mCacheOrderedKeys.begin(), mCacheOrderedKeys.end(), fid ); orderedIt != mCacheOrderedKeys.end() )
262 mCacheOrderedKeys.erase( orderedIt );
275 return mLayer->crs();
280 return mLayer->wkbType();
285 return mLayer->fields();
290 return mLayer->featureCount();
296 if ( fids.count() <= mCache.size() )
298 for (
const auto &idx : std::as_const( mCacheIndices ) )
300 idx->requestCompleted( featureRequest, fids );
312 const auto constMCacheIndices = mCacheIndices;
315 idx->flushFeature( fid );
319void QgsVectorLayerCache::onAttributeValueChanged(
QgsFeatureId fid,
int field,
const QVariant &value )
325 cachedFeat->mFeature->setAttribute( field, value );
331void QgsVectorLayerCache::onJoinAttributeValueChanged(
QgsFeatureId fid,
int field,
const QVariant &value )
333 const QgsVectorLayer *joinLayer = qobject_cast<const QgsVectorLayer *>( sender() );
335 const auto constVectorJoins = mLayer->vectorJoins();
336 for (
const QgsVectorLayerJoinInfo &info : constVectorJoins )
338 if ( joinLayer == info.joinLayer() )
340 const QgsFeature feature = mLayer->joinBuffer()->targetedFeatureOf( &info, joinLayer->
getFeature( fid ) );
342 const QString fieldName = info.prefixedFieldName( joinLayer->
fields().
field( field ) );
343 const int fieldIndex = mLayer->fields().indexFromName( fieldName );
345 if ( feature.
isValid() && fieldIndex != -1 )
347 onAttributeValueChanged( feature.
id(), fieldIndex, value );
354void QgsVectorLayerCache::featureDeleted(
QgsFeatureId fid )
356 mCache.remove( fid );
358 if (
auto it = mCacheUnorderedKeys.find( fid ); it != mCacheUnorderedKeys.end() )
360 mCacheUnorderedKeys.erase( it );
361 if (
auto orderedIt = std::find( mCacheOrderedKeys.begin(), mCacheOrderedKeys.end(), fid ); orderedIt != mCacheOrderedKeys.end() )
362 mCacheOrderedKeys.erase( orderedIt );
366void QgsVectorLayerCache::onFeatureAdded(
QgsFeatureId fid )
370 if (
cacheSize() <= mLayer->featureCount() )
381void QgsVectorLayerCache::attributeAdded(
int field )
384 mCachedAttributes.append( field );
388void QgsVectorLayerCache::attributeDeleted(
int field )
391 mCachedAttributes.clear();
393 for (
int attr : attrs )
396 mCachedAttributes << attr;
397 else if ( attr > field )
398 mCachedAttributes << attr - 1;
404 QgsCachedFeature *cachedFeat = mCache[ fid ];
408 cachedFeat->mFeature->setGeometry( geom );
412void QgsVectorLayerCache::layerDeleted()
418void QgsVectorLayerCache::invalidate()
420 if ( ! mCache.isEmpty() )
423 mCacheOrderedKeys.clear();
424 mCacheUnorderedKeys.clear();
433 const auto constMCacheIndices = mCacheIndices;
434 for ( QgsAbstractCacheIndex *idx : constMCacheIndices )
436 if ( idx->getCacheIterator( it, featureRequest ) )
447 if ( mCache.contains( featureRequest.
filterFid() ) )
481 bool requiresWriterIt =
true;
489 requiresWriterIt =
false;
494 requiresWriterIt = !canUseCacheForRequest( featureRequest, it );
501 requiresWriterIt =
false;
502 it = mLayer->getFeatures( featureRequest );
505 if ( requiresWriterIt && mLayer->dataProvider() )
511 if ( mCacheGeometry && mLayer->isSpatial() )
517 if ( mCachedAttributes.count( ) != mLayer->fields().count() )
520 QSet<int> attrs( requestSubset.begin(), requestSubset.end() );
521 for (
int attr : std::as_const( mCachedAttributes ) )
522 attrs.insert( attr );
540 return mCache.contains( fid );
545 const QList< QgsFeatureId > keys = mCache.keys();
555 requestedAttributes = mLayer->attributeList();
563 const auto constRequestedAttributes = requestedAttributes;
564 for (
int attr : constRequestedAttributes )
566 if ( !mCachedAttributes.contains( attr ) )
574 && !mCacheGeometry );
577void QgsVectorLayerCache::connectJoinedLayers()
const
579 const auto constVectorJoins = mLayer->
vectorJoins();
588bool QgsVectorLayerCache::QgsCachedFeature::allAttributesFetched()
const
590 return mAllAttributesFetched;
593bool QgsVectorLayerCache::QgsCachedFeature::geometryFetched()
const
595 return mGeometryFetched;
@ Fid
Filter using feature ID.
@ Fids
Filter using feature IDs.
@ Expression
Filter using expression.
@ NoFilter
No filter is applied.
@ 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.
@ NoFlags
No flags are set.
@ NoFilter
No spatial filtering of features.
QFlags< FeatureRequestFlag > FeatureRequestFlags
Flags for controlling feature requests.
WkbType
The WKB type describes the number of dimensions a geometry has.
Abstract base class for cache indices.
Represents a coordinate reference system (CRS).
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 close()
Call to end the iteration.
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.
QgsRectangle filterRect() const
Returns the rectangle from which features will be taken.
Qgis::FeatureRequestFilterType filterType() const
Returns the attribute/ID filter type which is currently set on this request.
Qgis::FeatureRequestFlags flags() const
Returns the flags which affect how features are fetched.
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
Qgis::SpatialFilterType spatialFilterType() const
Returns the spatial filter type which is currently set on this request.
QgsAttributeList subsetOfAttributes() const
Returns the subset of attributes which at least need to be fetched.
const QgsFeatureIds & filterFids() const
Returns the feature IDs that should be fetched.
QgsFeatureId filterFid() const
Returns the feature ID that should be fetched.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
bool isValid() const
Returns the validity of this feature.
Container of fields for a vector layer.
QgsField field(int fieldIdx) const
Returns the field at particular index (must be in range 0..N-1).
A geometry is the spatial representation of a feature.
void dataChanged()
Data of layer changed.
bool contains(const QgsRectangle &rect) const
Returns true when rectangle contains other rectangle.
bool isFidCached(QgsFeatureId fid) const
Check if a certain feature id is cached.
void setFullCache(bool fullCache)
This enables or disables full caching.
void finished()
When filling the cache, this signal gets emitted once the cache is fully initialized.
void featureRemoved(QgsFeatureId fid)
Gets called, whenever a feature has been removed.
void setCacheAddedAttributes(bool cacheAddedAttributes)
If this is enabled, the subset of cached attributes will automatically be extended to also include ne...
friend class QgsCachedFeature
void invalidated()
The cache has been invalidated and cleared.
void setCacheSize(int cacheSize)
Sets the maximum number of features to keep in the cache.
void setCacheSubsetOfAttributes(const QgsAttributeList &attributes)
Set the list (possibly a subset) of attributes to be cached.
void featureAdded(QgsFeatureId fid)
Emitted when a new feature has been added to the layer and this cache.
QgsFields fields() const
Returns the fields associated with features in the cache.
void cachedLayerDeleted()
Is emitted when the cached layer is deleted.
friend class QgsCachedFeatureWriterIterator
void requestCompleted(const QgsFeatureRequest &featureRequest, const QgsFeatureIds &fids)
Gets called, whenever the full list of feature ids for a certain request is known.
void attributeValueChanged(QgsFeatureId fid, int field, const QVariant &value)
Emitted when an attribute is changed.
bool removeCachedFeature(QgsFeatureId fid)
Removes the feature identified by fid from the cache if present.
void progress(int i, bool &cancel)
When filling the cache, this signal gets emitted periodically to notify about the progress and to be ...
QgsVectorLayer * layer()
Returns the layer to which this cache belongs.
long long featureCount() const
Returns the number of features contained in the source, or -1 if the feature count is unknown.
QgsCoordinateReferenceSystem sourceCrs() const
Returns the coordinate reference system for features in the cache.
bool checkInformationCovered(const QgsFeatureRequest &featureRequest)
Checks if the information required to complete the request is cached.
QgsFeatureIds cachedFeatureIds() const
Returns the set of feature IDs for features which are cached.
bool featureAtIdWithAllAttributes(QgsFeatureId featureId, QgsFeature &feature, bool skipCache=false)
Gets the feature at the given feature id with all attributes, if the cached feature already contains ...
bool cacheGeometry() const
Returns true if the cache will fetch and cache feature geometries.
Qgis::WkbType wkbType() const
Returns the geometry type for features in the cache.
int cacheSize()
Returns the maximum number of features this cache will hold.
friend class QgsCachedFeatureIterator
QgsAttributeList cacheSubsetOfAttributes() const
Returns the list (possibly a subset) of cached attributes.
void addCacheIndex(QgsAbstractCacheIndex *cacheIndex)
Adds a QgsAbstractCacheIndex to this cache.
QgsVectorLayerCache(QgsVectorLayer *layer, int cacheSize, QObject *parent=nullptr)
void setCacheGeometry(bool cacheGeometry)
Enable or disable the caching of geometries.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &featureRequest=QgsFeatureRequest())
Query this VectorLayerCache for features.
~QgsVectorLayerCache() override
bool completeFeatureAtId(QgsFeatureId featureId, QgsFeature &feature, bool skipCache=false)
Gets the feature at the given feature id with all attributes and geometry, if the cached feature alre...
bool featureAtId(QgsFeatureId featureId, QgsFeature &feature, bool skipCache=false)
Gets the feature at the given feature id.
Defines left outer join from our vector layer to some other vector layer.
Represents a vector layer which manages a vector based dataset.
void attributeAdded(int idx)
Will be emitted, when a new attribute has been added to this vector layer.
void attributeDeleted(int idx)
Will be emitted, when an attribute has been deleted from this vector layer.
void attributeValueChanged(QgsFeatureId fid, int idx, const QVariant &value)
Emitted whenever an attribute value change is done in the edit buffer.
void updatedFields()
Emitted whenever the fields available from this layer have been changed.
void featureAdded(QgsFeatureId fid)
Emitted when a new feature has been added to the layer.
void featureDeleted(QgsFeatureId fid)
Emitted when a feature has been deleted.
const QList< QgsVectorLayerJoinInfo > vectorJoins() const
void geometryChanged(QgsFeatureId fid, const QgsGeometry &geometry)
Emitted whenever a geometry change is done in the edit buffer.
Q_INVOKABLE QgsFeature getFeature(QgsFeatureId fid) const
Queries the layer for the feature with the given id.
QSet< QgsFeatureId > QgsFeatureIds
qint64 QgsFeatureId
64 bit feature ids negative numbers are used for uncommitted/newly added features
QList< int > QgsAttributeList