26#include "moc_qgsfeaturepickermodelbase.cpp"
28using namespace Qt::StringLiterals;
31 : QAbstractItemModel(
parent )
33 mReloadTimer.setInterval( 100 );
34 mReloadTimer.setSingleShot(
true );
35 connect( &mReloadTimer, &QTimer::timeout,
this, &QgsFeaturePickerModelBase::scheduledReload );
46 connect( mGatherer, &QgsFeatureExpressionValuesGatherer::finished, mGatherer, &QgsFeatureExpressionValuesGatherer::deleteLater );
63 mExpressionContext = mSourceLayer->createExpressionContext();
75 return mDisplayExpression.expression();
109 return mFilterExpression;
130 if ( mFormFeature == feature )
133 mFormFeature = feature;
143 return mParentFormFeature;
148 if ( mParentFormFeature == feature )
151 mParentFormFeature = feature;
173 return createIndex( row, column,
nullptr );
180 return QModelIndex();
194 if ( !
index.isValid() )
199 case Qt::DisplayRole:
212 const QVariantList values =
mEntries.value(
index.row() ).identifierFields;
213 return values.value( 0 );
219 case Qt::BackgroundRole:
220 case Qt::ForegroundRole:
221 case Qt::DecorationRole:
228 if ( role == Qt::ForegroundRole )
230 return QBrush( QColor( Qt::gray ) );
232 if ( role == Qt::FontRole )
234 QFont font = QFont();
236 font.setBold(
true );
238 font.setItalic(
true );
253 if ( role == Qt::DecorationRole )
255 if ( role == Qt::FontRole )
267void QgsFeaturePickerModelBase::updateCompleter()
271 QgsFeatureExpressionValuesGatherer *gatherer = qobject_cast<QgsFeatureExpressionValuesGatherer *>( sender() );
272 if ( gatherer->wasCanceled() )
278 QVector<QgsFeatureExpressionValuesGatherer::Entry> entries = mGatherer->entries();
286 const bool reloadCurrentFeatureOnly = mGatherer->data().toBool();
287 if ( reloadCurrentFeatureOnly )
289 if ( !entries.isEmpty() )
293 mShouldReloadCurrentFeature =
false;
294 setExtraValueDoesNotExist(
false );
298 setExtraValueDoesNotExist(
true );
301 mKeepCurrentEntry =
true;
302 mShouldReloadCurrentFeature =
false;
304 if ( mFilterValue.isEmpty() )
310 std::sort( entries.begin(), entries.end(), [&](
const QgsFeatureExpressionValuesGatherer::Entry & a,
const QgsFeatureExpressionValuesGatherer::Entry & b )
313 double aNumericOrderValue = a.orderValue.toDouble( &ok );
316 double bNumericOrderValue = b.orderValue.toDouble( &ok );
319 if ( mSortOrder == Qt::DescendingOrder )
321 return aNumericOrderValue > bNumericOrderValue;
325 return aNumericOrderValue < bNumericOrderValue;
329 if ( mSortOrder == Qt::DescendingOrder )
331 return a.orderValue.localeAwareCompare( b.orderValue ) > 0;
335 return a.orderValue.localeAwareCompare( b.orderValue ) < 0;
340 if ( mAllowNull && mSourceLayer )
342 entries.prepend( QgsFeatureExpressionValuesGatherer::nullEntry( mSourceLayer ) );
345 const int newEntriesSize = entries.size();
348 const int nbFixedEntry = ( mKeepCurrentEntry ? 1 : 0 ) + ( mAllowNull ? 1 : 0 );
351 int currentEntryInNewList = -1;
354 for (
int i = 0; i < newEntriesSize; ++i )
359 currentEntryInNewList = i;
360 setExtraValueDoesNotExist(
false );
382 beginRemoveRows( QModelIndex(), firstRow,
mEntries.size() - firstRow );
388 mIsSettingExtraIdentifierValue =
true;
390 mIsSettingExtraIdentifierValue =
false;
392 if ( currentEntryInNewList == -1 )
394 beginInsertRows( QModelIndex(), firstRow, entries.size() + 1 );
401 setExtraIdentifierValueIndex( firstRow == 0 && mAllowNull && !entries.isEmpty() ? 1 : 0, firstRow == 0 );
405 if ( currentEntryInNewList != 0 )
407 beginInsertRows( QModelIndex(), 0, currentEntryInNewList - 1 );
413 mEntries.replace( 0, entries.at( 0 ) );
417 if ( currentEntryInNewList >= nbFixedEntry )
419 emit dataChanged(
index( currentEntryInNewList, 0, QModelIndex() ),
index( currentEntryInNewList, 0, QModelIndex() ) );
422 beginInsertRows( QModelIndex(), currentEntryInNewList + 1, newEntriesSize - currentEntryInNewList - 1 );
423 mEntries += entries.mid( currentEntryInNewList + 1 );
425 setExtraIdentifierValueIndex( currentEntryInNewList );
430 mKeepCurrentEntry =
false;
436 Q_ASSERT( gatherer == mGatherer );
439 emit isLoadingChanged();
443void QgsFeaturePickerModelBase::scheduledReload()
448 bool wasLoading =
false;
456 QgsFeatureRequest request;
458 if ( mShouldReloadCurrentFeature )
464 QString filterClause;
466 if ( mFilterValue.isEmpty() && !mFilterExpression.isEmpty() )
467 filterClause = mFilterExpression;
468 else if ( mFilterExpression.isEmpty() && !mFilterValue.isEmpty() )
469 filterClause = u
"(%1) ILIKE '%%2%'"_s.arg( mDisplayExpression, mFilterValue );
470 else if ( !mFilterExpression.isEmpty() && !mFilterValue.isEmpty() )
471 filterClause = u
"(%1) AND ((%2) ILIKE '%%3%')"_s.arg( mFilterExpression, mDisplayExpression, mFilterValue );
473 if ( !filterClause.isEmpty() )
478 if ( !mFilterExpression.isEmpty() )
492 if ( !attributes.isEmpty() )
495 attributes += lFilterExpression->referencedColumns();
496 attributes += requestedAttributesForStyle();
501 if ( !mFetchGeometry )
503 if ( mFetchLimit > 0 )
507 mGatherer->setData( mShouldReloadCurrentFeature );
508 connect( mGatherer, &QgsFeatureExpressionValuesGatherer::finished,
this, &QgsFeaturePickerModelBase::updateCompleter );
516QSet<QString> QgsFeaturePickerModelBase::requestedAttributesForStyle()
const
518 QSet<QString> requestedAttrs;
520 const auto rowStyles = mSourceLayer->conditionalStyles()->rowStyles();
522 for (
const QgsConditionalStyle &style : rowStyles )
524 const QgsExpression exp( style.rule() );
525 requestedAttrs += exp.referencedColumns();
528 if ( mDisplayExpression.isField() )
530 const QString fieldName = *mDisplayExpression.referencedColumns().constBegin();
531 const auto constFieldStyles = mSourceLayer->conditionalStyles()->fieldStyles( fieldName );
532 for (
const QgsConditionalStyle &style : constFieldStyles )
534 const QgsExpression exp( style.rule() );
535 requestedAttrs += exp.referencedColumns();
539 return requestedAttrs;
543void QgsFeaturePickerModelBase::setExtraIdentifierValueIndex(
int index,
bool force )
553void QgsFeaturePickerModelBase::reloadCurrentFeature()
555 mShouldReloadCurrentFeature =
true;
556 mReloadTimer.start();
562 const QVector<QgsFeatureExpressionValuesGatherer::Entry> entries =
mEntries;
565 for (
const QgsFeatureExpressionValuesGatherer::Entry &entry : entries )
569 setExtraIdentifierValueIndex(
index );
580 if ( !isNull || mAllowNull )
582 beginInsertRows( QModelIndex(), 0, 0 );
586 setExtraValueDoesNotExist(
true );
587 reloadCurrentFeature();
591 mEntries.prepend( QgsFeatureExpressionValuesGatherer::nullEntry( mSourceLayer ) );
592 setExtraValueDoesNotExist(
false );
596 setExtraIdentifierValueIndex( 0,
true );
620 styles += matchingFieldStyles;
623 QgsConditionalStyle style;
625 mEntryStylesMap.insert( fid, style );
650 return mFetchGeometry;
681 return mExtraValueDoesNotExist;
685void QgsFeaturePickerModelBase::setExtraValueDoesNotExist(
bool extraValueDoesNotExist )
701void QgsFeaturePickerModelBase::reload()
703 mReloadTimer.start();
712 if ( mIsSettingExtraIdentifierValue )
715 mIsSettingExtraIdentifierValue =
true;
721 mIsSettingExtraIdentifierValue =
false;
729 return mOrderExpression.expression();
@ NoGeometry
Geometry is not required. It may still be returned if e.g. required for a filter condition.
QgsConditionalStyles rowStyles() const
Returns a list of row styles associated with the layer.
QList< QgsConditionalStyle > fieldStyles(const QString &fieldName) const
Returns the conditional styles set for the field with matching fieldName.
Conditional styling for a rule.
static QgsConditionalStyle compressStyles(const QList< QgsConditionalStyle > &styles)
Compress a list of styles into a single style.
static QList< QgsConditionalStyle > matchingConditionalStyles(const QList< QgsConditionalStyle > &styles, const QVariant &value, QgsExpressionContext &context)
Find and return the matching styles for the value and feature.
QColor backgroundColor() const
The background color for style.
QColor textColor() const
The text color set for style.
QFont font() const
The font for the style.
bool validTextColor() const
Check if the text color is valid for render.
bool isValid() const
isValid Check if this rule is valid.
QPixmap icon() const
The icon set for style generated from the set symbol.
bool validBackgroundColor() const
Check if the background color is valid for render.
static QgsExpressionContextScope * parentFormScope(const QgsFeature &formFeature=QgsFeature(), const QString &formMode=QString())
Creates a new scope which contains functions and variables from the current parent attribute form/tab...
static QgsExpressionContextScope * formScope(const QgsFeature &formFeature=QgsFeature(), const QString &formMode=QString())
Creates a new scope which contains functions and variables from the current attribute form/table form...
static QList< QgsExpressionContextScope * > globalProjectLayerScopes(const QgsMapLayer *layer)
Creates a list of three scopes: global, layer's project and layer.
void appendScope(QgsExpressionContextScope *scope)
Appends a scope to the end of the context.
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the context.
void appendScopes(const QList< QgsExpressionContextScope * > &scopes)
Appends a list of scopes to the end of the context.
Handles parsing and evaluation of expressions (formerly called "search strings").
QSet< QString > referencedColumns() const
Gets list of columns referenced by the expression.
void extraIdentifierValueIndexChanged(int index)
The index at which the extra identifier value is available within the model.
void beginUpdate()
Notification that the model is about to be changed because a job was completed.
virtual QVariant entryIdentifier(const QgsFeatureExpressionValuesGatherer::Entry &entry) const =0
Returns the identifier of the given entry.
void filterValueChanged()
This value will be used to filter the features available from this model.
void setFilterValue(const QString &filterValue)
This value will be used to filter the features available from this model.
void setExtraIdentifierValue(const QVariant &extraIdentifierValue)
Allows specifying one value that does not need to match the filter criteria but will still be availab...
virtual void requestToReloadCurrentFeature(QgsFeatureRequest &request)=0
Update the request to match the current feature to be reloaded.
void setParentFormFeature(const QgsFeature &feature)
Sets a parent attribute form feature to be used with the filter expression.
void parentFormFeatureChanged()
A parent attribute form feature to be used alongside the filter expression.
void filterExpressionChanged()
An additional filter expression to apply, next to the filterValue.
void setFetchLimit(int fetchLimit)
Defines the feature request fetch limit If set to 0, no limit is applied when fetching.
QVariant extraIdentifierValue() const
Allows specifying one value that does not need to match the filter criteria but will still be availab...
void extraValueDoesNotExistChanged(bool found)
Notification whether the model has found a feature tied to the extraIdentifierValue or not.
QgsFeature formFeature() const
Returns an attribute form feature to be used with the filter expression.
virtual QgsFeatureExpressionValuesGatherer * createValuesGatherer(const QgsFeatureRequest &request) const =0
Creates the value gatherer.
void setSortOrder(const Qt::SortOrder sortOrder)
The order direction will be used for sort values in the combobox.
void setFetchGeometry(bool fetchGeometry)
Defines if the geometry will be fetched.
void setExtraIdentifierValueUnguarded(const QVariant &identifierValue)
This will set the identifier value to be set in the model even if it doesn't exist currently in the d...
void extraIdentifierValueChanged()
Allows specifying one value that does not need to match the filter criteria but will still be availab...
void setOrderExpression(const QString &orderExpression)
The order expression will be used for sort values in the combobox.
virtual QSet< QString > requestedAttributes() const
Returns the attributes to be fetched in the request.
int mExtraValueIndex
The current index.
@ IdentifierValues
Used to retrieve the identifierValues (primary keys) of a feature.
@ FeatureId
Used to retrieve the id of a feature.
@ IdentifierValue
Used to retrieve the identifier value (primary key) of a feature.
@ Feature
Used to retrieve the feature, it might be incomplete if the request doesn't fetch all attributes or g...
@ Value
Used to retrieve the displayExpression of a feature.
QModelIndex parent(const QModelIndex &child) const override
void filterJobCompleted()
Indicates that a filter job has been completed and new data may be available.
void setAllowNull(bool allowNull)
Add a NULL entry to the list.
void setDisplayExpression(const QString &displayExpression)
The display expression will be used for.
QVariant mExtraIdentifierValue
The current identifier value.
QgsFeaturePickerModelBase(QObject *parent=nullptr)
Create a new QgsFeaturePickerModelBase, optionally specifying a parent.
virtual QgsFeatureExpressionValuesGatherer::Entry createEntry(const QVariant &identifier) const =0
Creates an entry with just the identifier so the feature can be retrieved in a next iteration.
bool isLoading() const
Indicator if the model is currently performing any feature iteration in the background.
void setFormFeature(const QgsFeature &feature)
Sets an attribute form feature to be used with the filter expression.
virtual bool identifierIsNull(const QVariant &identifier) const =0
Returns true if the entry is null The identifier can be either the feature ID or the list of identifi...
QgsVectorLayer * sourceLayer
QVariant data(const QModelIndex &index, int role) const override
QModelIndex index(int row, int column, const QModelIndex &parent) const override
int extraIdentifierValueIndex
void fetchLimitChanged()
Emitted when the fetching limit for the feature request changes.
void sourceLayerChanged()
The source layer from which features will be fetched.
QgsFeature parentFormFeature() const
Returns a parent attribute form feature to be used with the filter expression.
~QgsFeaturePickerModelBase() override
void setFilterExpression(const QString &filterExpression)
An additional filter expression to apply, next to the filterValue.
void allowNullChanged()
Add a NULL entry to the list.
int rowCount(const QModelIndex &parent) const override
virtual bool compareEntries(const QgsFeatureExpressionValuesGatherer::Entry &a, const QgsFeatureExpressionValuesGatherer::Entry &b) const =0
Returns true if the 2 entries refers to the same feature.
void currentFeatureChanged()
Emitted when the current feature in the model has changed This emitted both when the extra value chan...
void orderExpressionChanged()
An expression for generating values for sorting.
void sortOrderChanged()
The direction used for sorting.
void isLoadingChanged()
Indicator if the model is currently performing any feature iteration in the background.
void formFeatureChanged()
An attribute form feature to be used alongside the filter expression.
QString displayExpression
QVector< QgsFeatureExpressionValuesGatherer::Entry > mEntries
virtual QVariant nullIdentifier() const =0
Returns a null identifier.
void displayExpressionChanged()
The display expression will be used for.
void setSourceLayer(QgsVectorLayer *sourceLayer)
The source layer from which features will be fetched.
bool extraValueDoesNotExist() const
Flag indicating that the extraIdentifierValue does not exist in the data.
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.
QgsExpressionContext * expressionContext()
Returns the expression context used to evaluate filter expressions.
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsExpression * filterExpression() const
Returns the filter expression (if set).
QgsFeatureRequest & setFilterExpression(const QString &expression)
Set the filter expression.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Q_INVOKABLE QVariant attribute(const QString &name) const
Lookup attribute value by attribute name.
Represents a vector layer which manages a vector based dataset.
QgsConditionalLayerStyles * conditionalStyles() const
Returns the conditional styles that are set for this layer.
qint64 QgsFeatureId
64 bit feature ids negative numbers are used for uncommitted/newly added features