QGIS API Documentation 4.1.0-Master (376402f9aeb)
Loading...
Searching...
No Matches
qgsvectorlayer.cpp
Go to the documentation of this file.
1/***************************************************************************
2 qgsvectorlayer.cpp
3 --------------------
4 begin : Oct 29, 2003
5 copyright : (C) 2003 by Gary E.Sherman
6 email : sherman at mrcc.com
7
8 This class implements a generic means to display vector layers. The features
9 and attributes are read from the data store using a "data provider" plugin.
10 QgsVectorLayer can be used with any data store for which an appropriate
11 plugin is available.
12
13***************************************************************************/
14
15/***************************************************************************
16 * *
17 * This program is free software; you can redistribute it and/or modify *
18 * it under the terms of the GNU General Public License as published by *
19 * the Free Software Foundation; either version 2 of the License, or *
20 * (at your option) any later version. *
21 * *
22 ***************************************************************************/
23
24#include "qgsvectorlayer.h"
25
26#include <limits>
27#include <memory>
28#include <optional>
29
30#include "qgis.h"
31#include "qgsactionmanager.h"
32#include "qgsapplication.h"
33#include "qgsauxiliarystorage.h"
34#include "qgsconditionalstyle.h"
36#include "qgscurve.h"
37#include "qgsdatasourceuri.h"
38#include "qgsdiagramrenderer.h"
43#include "qgsfeature.h"
45#include "qgsfeaturerequest.h"
46#include "qgsfeedback.h"
47#include "qgsfields.h"
48#include "qgsgeometry.h"
49#include "qgsgeometryoptions.h"
51#include "qgslogger.h"
52#include "qgsmaplayerfactory.h"
53#include "qgsmaplayerlegend.h"
55#include "qgsmessagelog.h"
57#include "qgsobjectvisitor.h"
58#include "qgsogcutils.h"
59#include "qgspainting.h"
60#include "qgspallabeling.h"
61#include "qgspoint.h"
62#include "qgspointxy.h"
63#include "qgsprofilerequest.h"
64#include "qgsproject.h"
65#include "qgsproviderregistry.h"
66#include "qgsrectangle.h"
67#include "qgsrelationmanager.h"
68#include "qgsrendercontext.h"
69#include "qgsrenderer.h"
71#include "qgsruntimeprofiler.h"
74#include "qgssettingstree.h"
75#include "qgssldexportcontext.h"
77#include "qgssymbollayer.h"
78#include "qgssymbollayerutils.h"
79#include "qgstaskmanager.h"
80#include "qgsthreadingutils.h"
81#include "qgstransaction.h"
95#include "qgsvectorlayerutils.h"
96#include "qgsweakrelation.h"
97#include "qgsxmlutils.h"
98
99#include <QDir>
100#include <QDomNode>
101#include <QFile>
102#include <QImage>
103#include <QPainter>
104#include <QPainterPath>
105#include <QPolygonF>
106#include <QProgressDialog>
107#include <QRegularExpression>
108#include <QString>
109#include <QStringBuilder>
110#include <QTimer>
111#include <QUndoCommand>
112#include <QUrl>
113#include <QUrlQuery>
114#include <QUuid>
115#include <QVector>
116
117#include "moc_qgsvectorlayer.cpp"
118
119using namespace Qt::StringLiterals;
120
128
129
130#ifdef TESTPROVIDERLIB
131#include <dlfcn.h>
132#endif
133
134typedef bool saveStyle_t(
135 const QString &uri, const QString &qmlStyle, const QString &sldStyle, const QString &styleName, const QString &styleDescription, const QString &uiFileContent, bool useAsDefault, QString &errCause
136);
137
138typedef QString loadStyle_t( const QString &uri, QString &errCause );
139
140typedef int listStyles_t( const QString &uri, QStringList &ids, QStringList &names, QStringList &descriptions, QString &errCause );
141
142typedef QString getStyleById_t( const QString &uri, QString styleID, QString &errCause );
143
144typedef bool deleteStyleById_t( const QString &uri, QString styleID, QString &errCause );
145
146
147QgsVectorLayer::QgsVectorLayer( const QString &vectorLayerPath, const QString &baseName, const QString &providerKey, const QgsVectorLayer::LayerOptions &options )
148 : QgsMapLayer( Qgis::LayerType::Vector, baseName, vectorLayerPath )
149 , mSelectionProperties( new QgsVectorLayerSelectionProperties( this ) )
150 , mTemporalProperties( new QgsVectorLayerTemporalProperties( this ) )
151 , mElevationProperties( new QgsVectorLayerElevationProperties( this ) )
152 , mAuxiliaryLayer( nullptr )
153 , mAuxiliaryLayerKey( QString() )
154 , mReadExtentFromXml( options.readExtentFromXml )
155 , mRefreshRendererTimer( new QTimer( this ) )
156{
158 mLoadAllStoredStyle = options.loadAllStoredStyles;
159
160 if ( options.fallbackCrs.isValid() )
161 setCrs( options.fallbackCrs, false );
162 mWkbType = options.fallbackWkbType;
163
164 setProviderType( providerKey );
165
166 mGeometryOptions = std::make_unique<QgsGeometryOptions>();
167 mActions = new QgsActionManager( this );
168 mConditionalStyles = new QgsConditionalLayerStyles( this );
169 mStoredExpressionManager = new QgsStoredExpressionManager();
170 mStoredExpressionManager->setParent( this );
171
172 mJoinBuffer = new QgsVectorLayerJoinBuffer( this );
173 mJoinBuffer->setParent( this );
174 connect( mJoinBuffer, &QgsVectorLayerJoinBuffer::joinedFieldsChanged, this, &QgsVectorLayer::onJoinedFieldsChanged );
175
176 mExpressionFieldBuffer = new QgsExpressionFieldBuffer();
177 // if we're given a provider type, try to create and bind one to this layer
178 if ( !vectorLayerPath.isEmpty() && !mProviderKey.isEmpty() )
179 {
180 QgsDataProvider::ProviderOptions providerOptions { options.transformContext };
181 Qgis::DataProviderReadFlags providerFlags;
182 if ( options.loadDefaultStyle )
183 {
185 }
186 if ( options.forceReadOnly )
187 {
189 mDataSourceReadOnly = true;
190 }
191 setDataSource( vectorLayerPath, baseName, providerKey, providerOptions, providerFlags );
192 }
193
194 for ( const QgsField &field : std::as_const( mFields ) )
195 {
196 if ( !mAttributeAliasMap.contains( field.name() ) )
197 mAttributeAliasMap.insert( field.name(), QString() );
198 }
199
200 if ( isValid() )
201 {
202 mTemporalProperties->setDefaultsFromDataProviderTemporalCapabilities( mDataProvider->temporalCapabilities() );
203 if ( !mTemporalProperties->isActive() )
204 {
205 // didn't populate temporal properties from provider metadata, so at least try to setup some initially nice
206 // selections
207 mTemporalProperties->guessDefaultsFromFields( mFields );
208 }
209
210 mElevationProperties->setDefaultsFromLayer( this );
211 }
212
213 connect( this, &QgsVectorLayer::selectionChanged, this, [this] { triggerRepaint(); } );
214 connect( QgsProject::instance()->relationManager(), &QgsRelationManager::relationsLoaded, this, &QgsVectorLayer::onRelationsLoaded ); // skip-keyword-check
215
219
220 // Default simplify drawing settings
221 mSimplifyMethod.setSimplifyHints( QgsVectorLayer::settingsSimplifyDrawingHints->valueWithDefaultOverride( mSimplifyMethod.simplifyHints() ) );
222 mSimplifyMethod.setSimplifyAlgorithm( QgsVectorLayer::settingsSimplifyAlgorithm->valueWithDefaultOverride( mSimplifyMethod.simplifyAlgorithm() ) );
223 mSimplifyMethod.setThreshold( QgsVectorLayer::settingsSimplifyDrawingTol->valueWithDefaultOverride( mSimplifyMethod.threshold() ) );
224 mSimplifyMethod.setForceLocalOptimization( QgsVectorLayer::settingsSimplifyLocal->valueWithDefaultOverride( mSimplifyMethod.forceLocalOptimization() ) );
225 mSimplifyMethod.setMaximumScale( QgsVectorLayer::settingsSimplifyMaxScale->valueWithDefaultOverride( mSimplifyMethod.maximumScale() ) );
226
227 connect( mRefreshRendererTimer, &QTimer::timeout, this, [this] { triggerRepaint( true ); } );
228}
229
231{
232 emit willBeDeleted();
233
234 setValid( false );
235
236 delete mDataProvider;
237 delete mEditBuffer;
238 delete mJoinBuffer;
239 delete mExpressionFieldBuffer;
240 delete mLabeling;
241 delete mDiagramLayerSettings;
242 delete mDiagramRenderer;
243
244 delete mActions;
245
246 delete mRenderer;
247 delete mConditionalStyles;
248 delete mStoredExpressionManager;
249
250 if ( mFeatureCounter )
251 mFeatureCounter->cancel();
252
253 qDeleteAll( mRendererGenerators );
254}
255
257{
259
261 // We get the data source string from the provider when
262 // possible because some providers may have changed it
263 // directly (memory provider does that).
264 QString dataSource;
265 if ( mDataProvider )
266 {
267 dataSource = mDataProvider->dataSourceUri();
268 options.transformContext = mDataProvider->transformContext();
269 }
270 else
271 {
272 dataSource = source();
273 }
274 options.forceReadOnly = mDataSourceReadOnly;
275 QgsVectorLayer *layer = new QgsVectorLayer( dataSource, name(), mProviderKey, options );
276 if ( mDataProvider && layer->dataProvider() )
277 {
278 layer->dataProvider()->handlePostCloneOperations( mDataProvider );
279 }
280 QgsMapLayer::clone( layer );
281 layer->mXmlExtent2D = mXmlExtent2D;
282 layer->mLazyExtent2D = mLazyExtent2D;
283 layer->mValidExtent2D = mValidExtent2D;
284 layer->mXmlExtent3D = mXmlExtent3D;
285 layer->mLazyExtent3D = mLazyExtent3D;
286 layer->mValidExtent3D = mValidExtent3D;
287
288 QList<QgsVectorLayerJoinInfo> joins = vectorJoins();
289 const auto constJoins = joins;
290 for ( const QgsVectorLayerJoinInfo &join : constJoins )
291 {
292 // do not copy join information for auxiliary layer
293 if ( !auxiliaryLayer() || ( auxiliaryLayer() && auxiliaryLayer()->id() != join.joinLayerId() ) )
294 layer->addJoin( join );
295 }
296
297 if ( mDataProvider )
298 layer->setProviderEncoding( mDataProvider->encoding() );
299 layer->setSubsetString( subsetString() );
303 layer->setReadOnly( isReadOnly() );
308
309 const auto constActions = actions()->actions();
310 for ( const QgsAction &action : constActions )
311 {
312 layer->actions()->addAction( action );
313 }
314
315 if ( auto *lRenderer = renderer() )
316 {
317 layer->setRenderer( lRenderer->clone() );
318 }
319
320 if ( auto *lLabeling = labeling() )
321 {
322 layer->setLabeling( lLabeling->clone() );
323 }
325
327
328 if ( auto *lDiagramRenderer = diagramRenderer() )
329 {
330 layer->setDiagramRenderer( lDiagramRenderer->clone() );
331 }
332
333 if ( auto *lDiagramLayerSettings = diagramLayerSettings() )
334 {
335 layer->setDiagramLayerSettings( *lDiagramLayerSettings );
336 }
337
338 for ( int i = 0; i < fields().count(); i++ )
339 {
340 layer->setFieldAlias( i, attributeAlias( i ) );
342 layer->setEditorWidgetSetup( i, editorWidgetSetup( i ) );
345
346 QMap< QgsFieldConstraints::Constraint, QgsFieldConstraints::ConstraintStrength> constraints = fieldConstraintsAndStrength( i );
347 auto constraintIt = constraints.constBegin();
348 for ( ; constraintIt != constraints.constEnd(); ++constraintIt )
349 {
350 layer->setFieldConstraint( i, constraintIt.key(), constraintIt.value() );
351 }
352
353 if ( fields().fieldOrigin( i ) == Qgis::FieldOrigin::Expression )
354 {
355 layer->addExpressionField( expressionField( i ), fields().at( i ) );
356 }
357 }
358
360
361 if ( auto *lAuxiliaryLayer = auxiliaryLayer() )
362 layer->setAuxiliaryLayer( lAuxiliaryLayer->clone( layer ) );
363
364 layer->mElevationProperties = mElevationProperties->clone();
365 layer->mElevationProperties->setParent( layer );
366
367 layer->mSelectionProperties = mSelectionProperties->clone();
368 layer->mSelectionProperties->setParent( layer );
369
370 return layer;
371}
372
374{
376
377 if ( mDataProvider )
378 {
379 return mDataProvider->storageType();
380 }
381 return QString();
382}
383
384
386{
388
389 if ( mDataProvider )
390 {
391 return mDataProvider->capabilitiesString();
392 }
393 return QString();
394}
395
397{
399
400 return mDataProvider && mDataProvider->isSqlQuery();
401}
402
404{
406
407 return mDataProvider ? mDataProvider->vectorLayerTypeFlags() : Qgis::VectorLayerTypeFlags();
408}
409
411{
413
414 if ( mDataProvider )
415 {
416 return mDataProvider->dataComment();
417 }
418 return QString();
419}
420
427
429{
431
432 return name();
433}
434
436{
437 // non fatal for now -- the QgsVirtualLayerTask class is not thread safe and calls this
439
440 if ( mDataProvider )
441 {
442 mDataProvider->reloadData();
443 updateFields();
444 }
445}
446
448{
449 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
451
452 return new QgsVectorLayerRenderer( this, rendererContext );
453}
454
455
456void QgsVectorLayer::drawVertexMarker( double x, double y, QPainter &p, Qgis::VertexMarkerType type, int m )
457{
458 switch ( type )
459 {
461 p.setPen( QColor( 50, 100, 120, 200 ) );
462 p.setBrush( QColor( 200, 200, 210, 120 ) );
463 p.drawEllipse( x - m, y - m, m * 2 + 1, m * 2 + 1 );
464 break;
465
467 p.setPen( QColor( 255, 0, 0 ) );
468 p.drawLine( x - m, y + m, x + m, y - m );
469 p.drawLine( x - m, y - m, x + m, y + m );
470 break;
471
473 break;
474 }
475}
476
478{
480
481 mSelectedFeatureIds.insert( fid );
482 mPreviousSelectedFeatureIds.clear();
483
484 emit selectionChanged( QgsFeatureIds() << fid, QgsFeatureIds(), false );
485}
486
487void QgsVectorLayer::select( const QgsFeatureIds &featureIds )
488{
490
491 mSelectedFeatureIds.unite( featureIds );
492 mPreviousSelectedFeatureIds.clear();
493
494 emit selectionChanged( featureIds, QgsFeatureIds(), false );
495}
496
498{
500
501 mSelectedFeatureIds.remove( fid );
502 mPreviousSelectedFeatureIds.clear();
503
504 emit selectionChanged( QgsFeatureIds(), QgsFeatureIds() << fid, false );
505}
506
508{
510
511 mSelectedFeatureIds.subtract( featureIds );
512 mPreviousSelectedFeatureIds.clear();
513
514 emit selectionChanged( QgsFeatureIds(), featureIds, false );
515}
516
518{
520
521 // normalize the rectangle
522 QgsRectangle normalizedRect = rect;
523 normalizedRect.normalize();
524
525 QgsFeatureIds newSelection;
526
529 );
530
531 QgsFeature feat;
532 while ( features.nextFeature( feat ) )
533 {
534 newSelection << feat.id();
535 }
536 features.close();
537
538 selectByIds( newSelection, behavior );
539}
540
541void QgsVectorLayer::selectByExpression( const QString &expression, Qgis::SelectBehavior behavior, QgsExpressionContext *context )
542{
544
545 QgsFeatureIds newSelection;
546
547 std::optional< QgsExpressionContext > defaultContext;
548 if ( !context )
549 {
550 defaultContext.emplace( QgsExpressionContextUtils::globalProjectLayerScopes( this ) );
551 context = &defaultContext.value();
552 }
553 else
554 {
556 }
557
558 QgsExpression exp( expression );
559 exp.prepare( context );
560
562 {
565
566 if ( !exp.needsGeometry() )
568
569 QgsFeatureIterator features = getFeatures( request );
570
571 if ( behavior == Qgis::SelectBehavior::AddToSelection )
572 {
573 newSelection = selectedFeatureIds();
574 }
575 QgsFeature feat;
576 while ( features.nextFeature( feat ) )
577 {
578 newSelection << feat.id();
579 }
580 features.close();
581 }
583 {
584 QgsFeatureIds oldSelection = selectedFeatureIds();
585 QgsFeatureRequest request = QgsFeatureRequest().setFilterFids( oldSelection );
586
587 //refine request
588 if ( !exp.needsGeometry() )
591
592 QgsFeatureIterator features = getFeatures( request );
593 QgsFeature feat;
594 while ( features.nextFeature( feat ) )
595 {
596 context->setFeature( feat );
597 bool matches = exp.evaluate( context ).toBool();
598
599 if ( matches && behavior == Qgis::SelectBehavior::IntersectSelection )
600 {
601 newSelection << feat.id();
602 }
603 else if ( !matches && behavior == Qgis::SelectBehavior::RemoveFromSelection )
604 {
605 newSelection << feat.id();
606 }
607 }
608 }
610 selectByIds( newSelection );
611}
612
613void QgsVectorLayer::selectByIds( const QgsFeatureIds &ids, Qgis::SelectBehavior behavior, bool validateIds )
614{
616
617 // Opt-in validation: filter invalid IDs if requested
618 QgsFeatureIds idsToSelect = ids;
619 if ( validateIds )
620 {
621 idsToSelect = QgsVectorLayerUtils::filterValidFeatureIds( this, ids );
622 }
623
624 QgsFeatureIds newSelection;
625
626 switch ( behavior )
627 {
629 newSelection = idsToSelect;
630 break;
631
633 newSelection = mSelectedFeatureIds + idsToSelect;
634 break;
635
637 newSelection = mSelectedFeatureIds - idsToSelect;
638 break;
639
641 newSelection = mSelectedFeatureIds.intersect( idsToSelect );
642 break;
643 }
644
645 QgsFeatureIds deselectedFeatures = mSelectedFeatureIds - newSelection;
646 mSelectedFeatureIds = newSelection;
647 mPreviousSelectedFeatureIds.clear();
648
649 emit selectionChanged( newSelection, deselectedFeatures, true );
650}
651
652void QgsVectorLayer::modifySelection( const QgsFeatureIds &selectIds, const QgsFeatureIds &deselectIds )
653{
655
656 QgsFeatureIds intersectingIds = selectIds & deselectIds;
657 if ( !intersectingIds.isEmpty() )
658 {
659 QgsDebugMsgLevel( u"Trying to select and deselect the same item at the same time. Unsure what to do. Selecting dubious items."_s, 3 );
660 }
661
662 mSelectedFeatureIds -= deselectIds;
663 mSelectedFeatureIds += selectIds;
664 mPreviousSelectedFeatureIds.clear();
665
666 emit selectionChanged( selectIds, deselectIds - intersectingIds, false );
667}
668
670{
672
674 ids.subtract( mSelectedFeatureIds );
675 selectByIds( ids );
676}
677
684
686{
688
689 // normalize the rectangle
690 QgsRectangle normalizedRect = rect;
691 normalizedRect.normalize();
692
694
695 QgsFeatureIds selectIds;
696 QgsFeatureIds deselectIds;
697
698 QgsFeature fet;
699 while ( fit.nextFeature( fet ) )
700 {
701 if ( mSelectedFeatureIds.contains( fet.id() ) )
702 {
703 deselectIds << fet.id();
704 }
705 else
706 {
707 selectIds << fet.id();
708 }
709 }
710
711 modifySelection( selectIds, deselectIds );
712}
713
715{
717
718 if ( mSelectedFeatureIds.isEmpty() )
719 return;
720
721 const QgsFeatureIds previous = mSelectedFeatureIds;
723 mPreviousSelectedFeatureIds = previous;
724}
725
727{
729
730 if ( mPreviousSelectedFeatureIds.isEmpty() || !mSelectedFeatureIds.empty() )
731 return;
732
733 selectByIds( mPreviousSelectedFeatureIds );
734}
735
737{
738 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
740
741 return mDataProvider;
742}
743
745{
746 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
748
749 return mDataProvider;
750}
751
753{
754 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
756
757 return mSelectionProperties;
758}
759
766
773
775{
777
778 QgsProfileRequest modifiedRequest( request );
779 modifiedRequest.expressionContext().appendScope( createExpressionContextScope() );
780 return new QgsVectorLayerProfileGenerator( this, modifiedRequest );
781}
782
783void QgsVectorLayer::setProviderEncoding( const QString &encoding )
784{
786
787 if ( isValid() && mDataProvider && mDataProvider->encoding() != encoding )
788 {
789 mDataProvider->setEncoding( encoding );
790 updateFields();
791 }
792}
793
795{
797
798 delete mDiagramRenderer;
799 mDiagramRenderer = r;
800 emit rendererChanged();
801 emit styleChanged();
802}
803
805{
806 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
808
809 return QgsWkbTypes::geometryType( mWkbType );
810}
811
813{
815
816 return mWkbType;
817}
818
820{
822
823 if ( !isValid() || !isSpatial() || mSelectedFeatureIds.isEmpty() || !mDataProvider ) //no selected features
824 {
825 return QgsRectangle( 0, 0, 0, 0 );
826 }
827
828 QgsRectangle r, retval;
829 retval.setNull();
830
831 QgsFeature fet;
832 if ( mDataProvider->capabilities() & Qgis::VectorProviderCapability::SelectAtId )
833 {
834 QgsFeatureIterator fit = getFeatures( QgsFeatureRequest().setFilterFids( mSelectedFeatureIds ).setNoAttributes() );
835
836 while ( fit.nextFeature( fet ) )
837 {
838 if ( !fet.hasGeometry() )
839 continue;
840 r = fet.geometry().boundingBox();
841 retval.combineExtentWith( r );
842 }
843 }
844 else
845 {
846 QgsFeatureIterator fit = getFeatures( QgsFeatureRequest().setNoAttributes() );
847
848 while ( fit.nextFeature( fet ) )
849 {
850 if ( mSelectedFeatureIds.contains( fet.id() ) )
851 {
852 if ( fet.hasGeometry() )
853 {
854 r = fet.geometry().boundingBox();
855 retval.combineExtentWith( r );
856 }
857 }
858 }
859 }
860
861 if ( retval.width() == 0.0 || retval.height() == 0.0 )
862 {
863 // If all of the features are at the one point, buffer the
864 // rectangle a bit. If they are all at zero, do something a bit
865 // more crude.
866
867 if ( retval.xMinimum() == 0.0 && retval.xMaximum() == 0.0 && retval.yMinimum() == 0.0 && retval.yMaximum() == 0.0 )
868 {
869 retval.set( -1.0, -1.0, 1.0, 1.0 );
870 }
871 }
872
873 return retval;
874}
875
877{
878 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
880
881 return mLabelsEnabled && static_cast< bool >( mLabeling );
882}
883
885{
887
888 mLabelsEnabled = enabled;
889}
890
892{
893 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
895
896 if ( !mDiagramRenderer || !mDiagramLayerSettings )
897 return false;
898
899 QList<QgsDiagramSettings> settingList = mDiagramRenderer->diagramSettings();
900 if ( !settingList.isEmpty() )
901 {
902 return settingList.at( 0 ).enabled;
903 }
904 return false;
905}
906
907long long QgsVectorLayer::featureCount( const QString &legendKey ) const
908{
910
911 if ( !mSymbolFeatureCounted )
912 return -1;
913
914 return mSymbolFeatureCountMap.value( legendKey, -1 );
915}
916
917QgsFeatureIds QgsVectorLayer::symbolFeatureIds( const QString &legendKey ) const
918{
920
921 if ( !mSymbolFeatureCounted )
922 return QgsFeatureIds();
923
924 return mSymbolFeatureIdMap.value( legendKey, QgsFeatureIds() );
925}
927{
929
930 if ( ( mSymbolFeatureCounted || mFeatureCounter ) && !( storeSymbolFids && mSymbolFeatureIdMap.isEmpty() ) )
931 return mFeatureCounter;
932
933 mSymbolFeatureCountMap.clear();
934 mSymbolFeatureIdMap.clear();
935
936 if ( !isValid() )
937 {
938 QgsDebugMsgLevel( u"invoked with invalid layer"_s, 3 );
939 return mFeatureCounter;
940 }
941 if ( !mDataProvider )
942 {
943 QgsDebugMsgLevel( u"invoked with null mDataProvider"_s, 3 );
944 return mFeatureCounter;
945 }
946 if ( !mRenderer )
947 {
948 QgsDebugMsgLevel( u"invoked with null mRenderer"_s, 3 );
949 return mFeatureCounter;
950 }
951
952 if ( !mFeatureCounter || ( storeSymbolFids && mSymbolFeatureIdMap.isEmpty() ) )
953 {
954 mFeatureCounter = new QgsVectorLayerFeatureCounter( this, QgsExpressionContext(), storeSymbolFids );
955 connect( mFeatureCounter, &QgsTask::taskCompleted, this, &QgsVectorLayer::onFeatureCounterCompleted, Qt::UniqueConnection );
956 connect( mFeatureCounter, &QgsTask::taskTerminated, this, &QgsVectorLayer::onFeatureCounterTerminated, Qt::UniqueConnection );
957 QgsApplication::taskManager()->addTask( mFeatureCounter );
958 }
959
960 return mFeatureCounter;
961}
962
964{
966
967 // do not update extent by default when trust project option is activated
968 if ( force || !mReadExtentFromXml || ( mReadExtentFromXml && mXmlExtent2D.isNull() && mXmlExtent3D.isNull() ) )
969 {
970 mValidExtent2D = false;
971 mValidExtent3D = false;
972 }
973}
974
976{
978
980 mValidExtent2D = true;
981}
982
984{
986
988 mValidExtent3D = true;
989}
990
991void QgsVectorLayer::updateDefaultValues( QgsFeatureId fid, QgsFeature feature, QgsExpressionContext *context )
992{
994
995 if ( !mDefaultValueOnUpdateFields.isEmpty() )
996 {
997 if ( !feature.isValid() )
998 feature = getFeature( fid );
999
1000 int size = mFields.size();
1001 for ( int idx : std::as_const( mDefaultValueOnUpdateFields ) )
1002 {
1003 if ( idx < 0 || idx >= size )
1004 continue;
1005 feature.setAttribute( idx, defaultValue( idx, feature, context ) );
1006 updateFeature( feature, true );
1007 }
1008 }
1009}
1010
1012{
1014
1015 QgsRectangle rect;
1016 rect.setNull();
1017
1018 if ( !isSpatial() )
1019 return rect;
1020
1021 // Don't do lazy extent if the layer is currently in edit mode
1022 if ( mLazyExtent2D && isEditable() )
1023 mLazyExtent2D = false;
1024
1025 if ( mDataProvider && mDataProvider->isValid() && ( mDataProvider->flags() & Qgis::DataProviderFlag::FastExtent2D ) )
1026 {
1027 // Provider has a trivial 2D extent calculation => always get extent from provider.
1028 // Things are nice and simple this way, e.g. we can always trust that this extent is
1029 // accurate and up to date.
1030 updateExtent( mDataProvider->extent() );
1031 mValidExtent2D = true;
1032 mLazyExtent2D = false;
1033 }
1034 else
1035 {
1036 if ( !mValidExtent2D && mLazyExtent2D && mReadExtentFromXml && !mXmlExtent2D.isNull() )
1037 {
1038 updateExtent( mXmlExtent2D );
1039 mValidExtent2D = true;
1040 mLazyExtent2D = false;
1041 }
1042
1043 if ( !mValidExtent2D && mLazyExtent2D && mDataProvider && mDataProvider->isValid() )
1044 {
1045 // store the extent
1046 updateExtent( mDataProvider->extent() );
1047 mValidExtent2D = true;
1048 mLazyExtent2D = false;
1049
1050 // show the extent
1051 QgsDebugMsgLevel( u"2D Extent of layer: %1"_s.arg( mExtent2D.toString() ), 3 );
1052 }
1053 }
1054
1055 if ( mValidExtent2D )
1056 return QgsMapLayer::extent();
1057
1058 if ( !isValid() || !mDataProvider )
1059 {
1060 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider"_s, 3 );
1061 return rect;
1062 }
1063
1064 if ( !mEditBuffer
1065 || ( !mDataProvider->transaction() && ( mEditBuffer->deletedFeatureIds().isEmpty() && mEditBuffer->changedGeometries().isEmpty() ) )
1066 || QgsDataSourceUri( mDataProvider->dataSourceUri() ).useEstimatedMetadata() )
1067 {
1068 mDataProvider->updateExtents();
1069
1070 // get the extent of the layer from the provider
1071 // but only when there are some features already
1072 if ( mDataProvider->featureCount() != 0 )
1073 {
1074 const QgsRectangle r = mDataProvider->extent();
1075 rect.combineExtentWith( r );
1076 }
1077
1078 if ( mEditBuffer && !mDataProvider->transaction() )
1079 {
1080 const auto addedFeatures = mEditBuffer->addedFeatures();
1081 for ( QgsFeatureMap::const_iterator it = addedFeatures.constBegin(); it != addedFeatures.constEnd(); ++it )
1082 {
1083 if ( it->hasGeometry() )
1084 {
1085 const QgsRectangle r = it->geometry().boundingBox();
1086 rect.combineExtentWith( r );
1087 }
1088 }
1089 }
1090 }
1091 else
1092 {
1093 QgsFeatureIterator fit = getFeatures( QgsFeatureRequest().setNoAttributes() );
1094
1095 QgsFeature fet;
1096 while ( fit.nextFeature( fet ) )
1097 {
1098 if ( fet.hasGeometry() && fet.geometry().type() != Qgis::GeometryType::Unknown )
1099 {
1100 const QgsRectangle bb = fet.geometry().boundingBox();
1101 rect.combineExtentWith( bb );
1102 }
1103 }
1104 }
1105
1106 if ( rect.xMinimum() > rect.xMaximum() && rect.yMinimum() > rect.yMaximum() )
1107 {
1108 // special case when there are no features in provider nor any added
1109 rect = QgsRectangle(); // use rectangle with zero coordinates
1110 }
1111
1112 updateExtent( rect );
1113 mValidExtent2D = true;
1114
1115 // Send this (hopefully) up the chain to the map canvas
1116 emit recalculateExtents();
1117
1118 return rect;
1119}
1120
1122{
1124
1125 // if data is 2D, redirect to 2D extend computation, and save it as 2D extent (in 3D bbox)
1126 if ( mDataProvider && mDataProvider->elevationProperties() && !mDataProvider->elevationProperties()->containsElevationData() )
1127 {
1128 return QgsBox3D( extent() );
1129 }
1130
1132 extent.setNull();
1133
1134 if ( !isSpatial() )
1135 return extent;
1136
1137 if ( mDataProvider && mDataProvider->isValid() && ( mDataProvider->flags() & Qgis::DataProviderFlag::FastExtent3D ) )
1138 {
1139 // Provider has a trivial 3D extent calculation => always get extent from provider.
1140 // Things are nice and simple this way, e.g. we can always trust that this extent is
1141 // accurate and up to date.
1142 updateExtent( mDataProvider->extent3D() );
1143 mValidExtent3D = true;
1144 mLazyExtent3D = false;
1145 }
1146 else
1147 {
1148 if ( !mValidExtent3D && mLazyExtent3D && mReadExtentFromXml && !mXmlExtent3D.isNull() )
1149 {
1150 updateExtent( mXmlExtent3D );
1151 mValidExtent3D = true;
1152 mLazyExtent3D = false;
1153 }
1154
1155 if ( !mValidExtent3D && mLazyExtent3D && mDataProvider && mDataProvider->isValid() )
1156 {
1157 // store the extent
1158 updateExtent( mDataProvider->extent3D() );
1159 mValidExtent3D = true;
1160 mLazyExtent3D = false;
1161
1162 // show the extent
1163 QgsDebugMsgLevel( u"3D Extent of layer: %1"_s.arg( mExtent3D.toString() ), 3 );
1164 }
1165 }
1166
1167 if ( mValidExtent3D )
1168 return QgsMapLayer::extent3D();
1169
1170 if ( !isValid() || !mDataProvider )
1171 {
1172 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider"_s, 3 );
1173 return extent;
1174 }
1175
1176 if ( !mEditBuffer
1177 || ( !mDataProvider->transaction() && ( mEditBuffer->deletedFeatureIds().isEmpty() && mEditBuffer->changedGeometries().isEmpty() ) )
1178 || QgsDataSourceUri( mDataProvider->dataSourceUri() ).useEstimatedMetadata() )
1179 {
1180 mDataProvider->updateExtents();
1181
1182 // get the extent of the layer from the provider
1183 // but only when there are some features already
1184 if ( mDataProvider->featureCount() != 0 )
1185 {
1186 const QgsBox3D ext = mDataProvider->extent3D();
1187 extent.combineWith( ext );
1188 }
1189
1190 if ( mEditBuffer && !mDataProvider->transaction() )
1191 {
1192 const auto addedFeatures = mEditBuffer->addedFeatures();
1193 for ( QgsFeatureMap::const_iterator it = addedFeatures.constBegin(); it != addedFeatures.constEnd(); ++it )
1194 {
1195 if ( it->hasGeometry() )
1196 {
1197 const QgsBox3D bbox = it->geometry().boundingBox3D();
1198 extent.combineWith( bbox );
1199 }
1200 }
1201 }
1202 }
1203 else
1204 {
1205 QgsFeatureIterator fit = getFeatures( QgsFeatureRequest().setNoAttributes() );
1206
1207 QgsFeature fet;
1208 while ( fit.nextFeature( fet ) )
1209 {
1210 if ( fet.hasGeometry() && fet.geometry().type() != Qgis::GeometryType::Unknown )
1211 {
1212 const QgsBox3D bb = fet.geometry().boundingBox3D();
1213 extent.combineWith( bb );
1214 }
1215 }
1216 }
1217
1218 if ( extent.xMinimum() > extent.xMaximum() && extent.yMinimum() > extent.yMaximum() && extent.zMinimum() > extent.zMaximum() )
1219 {
1220 // special case when there are no features in provider nor any added
1221 extent = QgsBox3D(); // use rectangle with zero coordinates
1222 }
1223
1224 updateExtent( extent );
1225 mValidExtent3D = true;
1226
1227 // Send this (hopefully) up the chain to the map canvas
1228 emit recalculateExtents();
1229
1230 return extent;
1231}
1232
1239
1246
1248{
1250
1251 if ( !isValid() || !mDataProvider )
1252 {
1253 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider"_s, 3 );
1254 return customProperty( u"storedSubsetString"_s ).toString();
1255 }
1256 return mDataProvider->subsetString();
1257}
1258
1259bool QgsVectorLayer::setSubsetString( const QString &subset )
1260{
1262
1263 if ( !isValid() || !mDataProvider )
1264 {
1265 QgsDebugMsgLevel( u"invoked with invalid layer or null mDataProvider or while editing"_s, 3 );
1266 setCustomProperty( u"storedSubsetString"_s, subset );
1267 return false;
1268 }
1269 else if ( mEditBuffer )
1270 {
1271 QgsDebugMsgLevel( u"invoked while editing"_s, 3 );
1272 return false;
1273 }
1274
1275 if ( subset == mDataProvider->subsetString() )
1276 return true;
1277
1278 bool res = mDataProvider->setSubsetString( subset );
1279
1280 // get the updated data source string from the provider
1281 mDataSource = mDataProvider->dataSourceUri();
1282 updateExtents();
1283 updateFields();
1284
1285 if ( res )
1286 {
1287 emit subsetStringChanged();
1289 }
1290
1291 return res;
1292}
1293
1295{
1296 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
1298
1299 if ( isValid()
1300 && mDataProvider
1301 && !mEditBuffer
1303 && ( mSimplifyMethod.simplifyHints() & simplifyHint )
1304 && renderContext.useRenderingOptimization() )
1305 {
1306 double maximumSimplificationScale = mSimplifyMethod.maximumScale();
1307
1308 // check maximum scale at which generalisation should be carried out
1309 return !( maximumSimplificationScale > 1 && renderContext.rendererScale() <= maximumSimplificationScale );
1310 }
1311 return false;
1312}
1313
1315{
1317
1318 return mConditionalStyles;
1319}
1320
1322{
1323 // non fatal for now -- the aggregate expression functions are not thread safe and call this
1325
1326 if ( !isValid() || !mDataProvider )
1327 return QgsFeatureIterator();
1328
1329 return QgsFeatureIterator( new QgsVectorLayerFeatureIterator( new QgsVectorLayerFeatureSource( this ), true, request ) );
1330}
1331
1333{
1335
1336 QgsFeature feature;
1338 if ( feature.isValid() )
1339 return feature.geometry();
1340 else
1341 return QgsGeometry();
1342}
1343
1345{
1347
1348 if ( !isValid() || !mEditBuffer || !mDataProvider )
1349 return false;
1350
1351
1352 if ( mGeometryOptions->isActive() )
1353 {
1354 QgsGeometry geom = feature.geometry();
1355 mGeometryOptions->apply( geom );
1356 feature.setGeometry( geom );
1357 }
1358
1359 bool success = mEditBuffer->addFeature( feature );
1360
1361 if ( success && mJoinBuffer->containsJoins() )
1362 {
1363 success = mJoinBuffer->addFeature( feature );
1364 }
1365
1366 return success;
1367}
1368
1369bool QgsVectorLayer::updateFeature( QgsFeature &updatedFeature, bool skipDefaultValues )
1370{
1372
1373 if ( !mEditBuffer || !mDataProvider )
1374 {
1375 return false;
1376 }
1377
1378 QgsFeature currentFeature = getFeature( updatedFeature.id() );
1379 if ( currentFeature.isValid() )
1380 {
1381 bool hasChanged = false;
1382 bool hasError = false;
1383
1384 if ( ( updatedFeature.hasGeometry() || currentFeature.hasGeometry() ) && !updatedFeature.geometry().isExactlyEqual( currentFeature.geometry() ) )
1385 {
1386 QgsGeometry geometry = updatedFeature.geometry();
1387 if ( changeGeometry( updatedFeature.id(), geometry, true ) )
1388 {
1389 hasChanged = true;
1390 updatedFeature.setGeometry( geometry );
1391 }
1392 else
1393 {
1394 QgsDebugMsgLevel( u"geometry of feature %1 could not be changed."_s.arg( updatedFeature.id() ), 3 );
1395 }
1396 }
1397
1398 QgsAttributes fa = updatedFeature.attributes();
1399 QgsAttributes ca = currentFeature.attributes();
1400
1401 for ( int attr = 0; attr < fa.count(); ++attr )
1402 {
1403 if ( !qgsVariantEqual( fa.at( attr ), ca.at( attr ) ) )
1404 {
1405 if ( changeAttributeValue( updatedFeature.id(), attr, fa.at( attr ), ca.at( attr ), true ) )
1406 {
1407 hasChanged = true;
1408 }
1409 else
1410 {
1411 QgsDebugMsgLevel( u"attribute %1 of feature %2 could not be changed."_s.arg( attr ).arg( updatedFeature.id() ), 3 );
1412 hasError = true;
1413 }
1414 }
1415 }
1416 if ( hasChanged && !mDefaultValueOnUpdateFields.isEmpty() && !skipDefaultValues )
1417 updateDefaultValues( updatedFeature.id(), updatedFeature );
1418
1419 return !hasError;
1420 }
1421 else
1422 {
1423 QgsDebugMsgLevel( u"feature %1 could not be retrieved"_s.arg( updatedFeature.id() ), 3 );
1424 return false;
1425 }
1426}
1427
1428
1429bool QgsVectorLayer::insertVertex( double x, double y, QgsFeatureId atFeatureId, int beforeVertex )
1430{
1432
1433 if ( !isValid() || !mEditBuffer || !mDataProvider )
1434 return false;
1435
1436 QgsVectorLayerEditUtils utils( this );
1437 bool result = utils.insertVertex( x, y, atFeatureId, beforeVertex );
1438 if ( result )
1439 updateExtents();
1440 return result;
1441}
1442
1443
1444bool QgsVectorLayer::insertVertex( const QgsPoint &point, QgsFeatureId atFeatureId, int beforeVertex )
1445{
1447
1448 if ( !isValid() || !mEditBuffer || !mDataProvider )
1449 return false;
1450
1451 QgsVectorLayerEditUtils utils( this );
1452 bool result = utils.insertVertex( point, atFeatureId, beforeVertex );
1453 if ( result )
1454 updateExtents();
1455 return result;
1456}
1457
1458
1459bool QgsVectorLayer::moveVertex( double x, double y, QgsFeatureId atFeatureId, int atVertex )
1460{
1462
1463 if ( !isValid() || !mEditBuffer || !mDataProvider )
1464 return false;
1465
1466 QgsVectorLayerEditUtils utils( this );
1467 bool result = utils.moveVertex( x, y, atFeatureId, atVertex );
1468
1469 if ( result )
1470 updateExtents();
1471 return result;
1472}
1473
1474bool QgsVectorLayer::moveVertex( const QgsPoint &p, QgsFeatureId atFeatureId, int atVertex )
1475{
1477
1478 if ( !isValid() || !mEditBuffer || !mDataProvider )
1479 return false;
1480
1481 QgsVectorLayerEditUtils utils( this );
1482 bool result = utils.moveVertex( p, atFeatureId, atVertex );
1483
1484 if ( result )
1485 updateExtents();
1486 return result;
1487}
1488
1490{
1492
1493 if ( !isValid() || !mEditBuffer || !mDataProvider )
1495
1496 QgsVectorLayerEditUtils utils( this );
1497 Qgis::VectorEditResult result = utils.deleteVertex( featureId, vertex );
1498
1499 if ( result == Qgis::VectorEditResult::Success )
1500 updateExtents();
1501 return result;
1502}
1503
1504
1506{
1508
1509 if ( !isValid() || !mDataProvider || !( mDataProvider->capabilities() & Qgis::VectorProviderCapability::DeleteFeatures ) )
1510 {
1511 return false;
1512 }
1513
1514 if ( !isEditable() )
1515 {
1516 return false;
1517 }
1518
1519 int deleted = 0;
1520 int count = mSelectedFeatureIds.size();
1521 // Make a copy since deleteFeature modifies mSelectedFeatureIds
1522 QgsFeatureIds selectedFeatures( mSelectedFeatureIds );
1523 for ( QgsFeatureId fid : std::as_const( selectedFeatures ) )
1524 {
1525 deleted += deleteFeature( fid, context ); // removes from selection
1526 }
1527
1529 updateExtents();
1530
1531 if ( deletedCount )
1532 {
1533 *deletedCount = deleted;
1534 }
1535
1536 return deleted == count;
1537}
1538
1539static const QgsPointSequence vectorPointXY2pointSequence( const QVector<QgsPointXY> &points )
1540{
1541 QgsPointSequence pts;
1542 pts.reserve( points.size() );
1543 QVector<QgsPointXY>::const_iterator it = points.constBegin();
1544 while ( it != points.constEnd() )
1545 {
1546 pts.append( QgsPoint( *it ) );
1547 ++it;
1548 }
1549 return pts;
1550}
1551Qgis::GeometryOperationResult QgsVectorLayer::addRing( const QVector<QgsPointXY> &ring, QgsFeatureId *featureId )
1552{
1554
1555 return addRing( vectorPointXY2pointSequence( ring ), featureId );
1556}
1557
1559{
1561
1562 if ( !isValid() || !mEditBuffer || !mDataProvider )
1564
1565 QgsVectorLayerEditUtils utils( this );
1567
1568 //first try with selected features
1569 if ( !mSelectedFeatureIds.isEmpty() )
1570 {
1571 result = utils.addRing( ring, mSelectedFeatureIds, featureId );
1572 }
1573
1575 {
1576 //try with all intersecting features
1577 result = utils.addRing( ring, QgsFeatureIds(), featureId );
1578 }
1579
1580 return result;
1581}
1582
1584{
1586
1587 if ( !isValid() || !mEditBuffer || !mDataProvider )
1588 {
1589 delete ring;
1591 }
1592
1593 if ( !ring )
1594 {
1596 }
1597
1598 if ( !ring->isClosed() )
1599 {
1600 delete ring;
1602 }
1603
1604 QgsVectorLayerEditUtils utils( this );
1606
1607 //first try with selected features
1608 if ( !mSelectedFeatureIds.isEmpty() )
1609 {
1610 result = utils.addRing( static_cast< QgsCurve * >( ring->clone() ), mSelectedFeatureIds, featureId );
1611 }
1612
1614 {
1615 //try with all intersecting features
1616 result = utils.addRing( static_cast< QgsCurve * >( ring->clone() ), QgsFeatureIds(), featureId );
1617 }
1618
1619 delete ring;
1620 return result;
1621}
1622
1624{
1626
1627 QgsPointSequence pts;
1628 pts.reserve( points.size() );
1629 for ( QList<QgsPointXY>::const_iterator it = points.constBegin(); it != points.constEnd(); ++it )
1630 {
1631 pts.append( QgsPoint( *it ) );
1632 }
1633 return addPart( pts );
1634}
1635
1637{
1639
1640 if ( !isValid() || !mEditBuffer || !mDataProvider )
1642
1643 //number of selected features must be 1
1644
1645 if ( mSelectedFeatureIds.empty() )
1646 {
1647 QgsDebugMsgLevel( u"Number of selected features <1"_s, 3 );
1649 }
1650 else if ( mSelectedFeatureIds.size() > 1 )
1651 {
1652 QgsDebugMsgLevel( u"Number of selected features >1"_s, 3 );
1654 }
1655
1656 QgsVectorLayerEditUtils utils( this );
1657 Qgis::GeometryOperationResult result = utils.addPart( points, *mSelectedFeatureIds.constBegin() );
1658
1660 updateExtents();
1661 return result;
1662}
1663
1665{
1667
1668 if ( !isValid() || !mEditBuffer || !mDataProvider )
1670
1671 //number of selected features must be 1
1672
1673 if ( mSelectedFeatureIds.empty() )
1674 {
1675 QgsDebugMsgLevel( u"Number of selected features <1"_s, 3 );
1677 }
1678 else if ( mSelectedFeatureIds.size() > 1 )
1679 {
1680 QgsDebugMsgLevel( u"Number of selected features >1"_s, 3 );
1682 }
1683
1684 QgsVectorLayerEditUtils utils( this );
1685 Qgis::GeometryOperationResult result = utils.addPart( ring, *mSelectedFeatureIds.constBegin() );
1686
1688 updateExtents();
1689 return result;
1690}
1691
1692// TODO QGIS 5.0 -- this should return Qgis::GeometryOperationResult, not int
1693int QgsVectorLayer::translateFeature( QgsFeatureId featureId, double dx, double dy )
1694{
1696
1697 if ( !isValid() || !mEditBuffer || !mDataProvider )
1698 return static_cast< int >( Qgis::GeometryOperationResult::LayerNotEditable );
1699
1700 QgsVectorLayerEditUtils utils( this );
1701 int result = utils.translateFeature( featureId, dx, dy );
1702
1703 if ( result == static_cast< int >( Qgis::GeometryOperationResult::Success ) )
1704 updateExtents();
1705 return result;
1706}
1707
1708Qgis::GeometryOperationResult QgsVectorLayer::splitParts( const QVector<QgsPointXY> &splitLine, bool topologicalEditing )
1709{
1711
1712 return splitParts( vectorPointXY2pointSequence( splitLine ), topologicalEditing );
1713}
1714
1716{
1718
1719 if ( !isValid() || !mEditBuffer || !mDataProvider )
1721
1722 QgsVectorLayerEditUtils utils( this );
1723 return utils.splitParts( splitLine, topologicalEditing );
1724}
1725
1726Qgis::GeometryOperationResult QgsVectorLayer::splitFeatures( const QVector<QgsPointXY> &splitLine, bool topologicalEditing )
1727{
1729
1730 return splitFeatures( vectorPointXY2pointSequence( splitLine ), topologicalEditing );
1731}
1732
1734{
1736
1737 QgsLineString splitLineString( splitLine );
1738 QgsPointSequence topologyTestPoints;
1739 bool preserveCircular = false;
1740 return splitFeatures( &splitLineString, topologyTestPoints, preserveCircular, topologicalEditing );
1741}
1742
1743Qgis::GeometryOperationResult QgsVectorLayer::splitFeatures( const QgsCurve *curve, QgsPointSequence &topologyTestPoints, bool preserveCircular, bool topologicalEditing )
1744{
1746
1747 if ( !isValid() || !mEditBuffer || !mDataProvider )
1749
1750 QgsVectorLayerEditUtils utils( this );
1751 return utils.splitFeatures( curve, topologyTestPoints, preserveCircular, topologicalEditing );
1752}
1753
1755{
1757
1758 if ( !isValid() || !mEditBuffer || !mDataProvider )
1759 return -1;
1760
1761 QgsVectorLayerEditUtils utils( this );
1762 return utils.addTopologicalPoints( geom );
1763}
1764
1771
1773{
1775
1776 if ( !isValid() || !mEditBuffer || !mDataProvider )
1777 return -1;
1778
1779 QgsVectorLayerEditUtils utils( this );
1780 return utils.addTopologicalPoints( p );
1781}
1782
1784{
1786
1787 if ( !mValid || !mEditBuffer || !mDataProvider )
1788 return -1;
1789
1790 QgsVectorLayerEditUtils utils( this );
1791 return utils.addTopologicalPoints( ps );
1792}
1793
1795{
1797
1798 if ( mLabeling == labeling )
1799 return;
1800
1801 delete mLabeling;
1802 mLabeling = labeling;
1803}
1804
1806{
1808
1809 if ( project() && project()->transactionMode() == Qgis::TransactionMode::BufferedGroups )
1810 return project()->startEditing( this );
1811
1812 if ( !isValid() || !mDataProvider )
1813 {
1814 return false;
1815 }
1816
1817 // allow editing if provider supports any of the capabilities
1818 if ( !supportsEditing() )
1819 {
1820 return false;
1821 }
1822
1823 if ( mEditBuffer )
1824 {
1825 // editing already underway
1826 return false;
1827 }
1828
1829 mDataProvider->enterUpdateMode();
1830
1831 emit beforeEditingStarted();
1832
1833 createEditBuffer();
1834
1835 updateFields();
1836
1837 emit editingStarted();
1838
1839 return true;
1840}
1841
1843{
1845
1846 if ( mDataProvider )
1847 mDataProvider->setTransformContext( transformContext );
1848}
1849
1851{
1853
1854 return mDataProvider ? mDataProvider->hasSpatialIndex() : Qgis::SpatialIndexPresence::Unknown;
1855}
1856
1858{
1860
1861 if ( mRenderer )
1862 if ( !mRenderer->accept( visitor ) )
1863 return false;
1864
1865 if ( mLabeling )
1866 if ( !mLabeling->accept( visitor ) )
1867 return false;
1868
1869 return true;
1870}
1871
1873{
1875
1876 if ( mActions )
1877 {
1878 const QList<QgsAction> actions = mActions->actions();
1879 for ( const QgsAction &action : actions )
1880 {
1881 if ( action.command().isEmpty() )
1882 {
1883 continue;
1884 }
1885
1886 switch ( action.type() )
1887 {
1892 {
1893 QgsEmbeddedScriptEntity entity( Qgis::EmbeddedScriptType::Action, tr( "%1: Action ’%2’" ).arg( name(), action.name() ), action.command() );
1894 if ( !visitor->visitEmbeddedScript( entity, context ) )
1895 {
1896 return false;
1897 }
1898 break;
1899 }
1900
1905 {
1906 break;
1907 }
1908 }
1909 }
1910 }
1911
1912 QString initCode;
1913 switch ( mEditFormConfig.initCodeSource() )
1914 {
1916 {
1917 initCode = u"# Calling function ’%1’\n\n%2"_s.arg( mEditFormConfig.initFunction(), mEditFormConfig.initCode() );
1918 break;
1919 }
1920
1922 {
1923 QFile *inputFile = QgsApplication::networkContentFetcherRegistry()->localFile( mEditFormConfig.initFilePath() );
1924 if ( inputFile && inputFile->open( QFile::ReadOnly ) )
1925 {
1926 // Read it into a string
1927 QTextStream inf( inputFile );
1928 initCode = inf.readAll();
1929 inputFile->close();
1930 initCode = u"# Calling function ’%1’\n# From file %2\n\n"_s.arg( mEditFormConfig.initFunction(), mEditFormConfig.initFilePath() ) + initCode;
1931 }
1932 break;
1933 }
1934
1936 {
1937 initCode = u"# Calling function ’%1’\n# From environment\n\n"_s.arg( mEditFormConfig.initFunction() );
1938 break;
1939 }
1940
1942 {
1943 break;
1944 }
1945 }
1946
1947 if ( !initCode.isEmpty() )
1948 {
1949 QgsEmbeddedScriptEntity entity( Qgis::EmbeddedScriptType::FormInitCode, tr( "%1: Attribute form init code" ).arg( name() ), initCode );
1950 if ( !visitor->visitEmbeddedScript( entity, context ) )
1951 {
1952 return false;
1953 }
1954 }
1955
1956 return true;
1957}
1958
1959bool QgsVectorLayer::readXml( const QDomNode &layer_node, QgsReadWriteContext &context )
1960{
1962
1963 QgsDebugMsgLevel( u"Datasource in QgsVectorLayer::readXml: %1"_s.arg( mDataSource.toLocal8Bit().data() ), 3 );
1964
1965 //process provider key
1966 QDomNode pkeyNode = layer_node.namedItem( u"provider"_s );
1967
1968 if ( pkeyNode.isNull() )
1969 {
1970 mProviderKey.clear();
1971 }
1972 else
1973 {
1974 QDomElement pkeyElt = pkeyNode.toElement();
1975 mProviderKey = pkeyElt.text();
1976 }
1977
1978 // determine type of vector layer
1979 if ( !mProviderKey.isNull() )
1980 {
1981 // if the provider string isn't empty, then we successfully
1982 // got the stored provider
1983 }
1984 else if ( mDataSource.contains( "dbname="_L1 ) )
1985 {
1986 mProviderKey = u"postgres"_s;
1987 }
1988 else
1989 {
1990 mProviderKey = u"ogr"_s;
1991 }
1992
1993 const QDomElement elem = layer_node.toElement();
1995
1996 mDataSourceReadOnly = mReadFlags & QgsMapLayer::FlagForceReadOnly;
1998
1999 if ( ( mReadFlags & QgsMapLayer::FlagDontResolveLayers ) || !setDataProvider( mProviderKey, options, flags ) )
2000 {
2002 {
2003 QgsDebugError( u"Could not set data provider for layer %1"_s.arg( publicSource() ) );
2004 }
2005
2006 // for invalid layer sources, we fallback to stored wkbType if available
2007 if ( elem.hasAttribute( u"wkbType"_s ) )
2008 mWkbType = qgsEnumKeyToValue( elem.attribute( u"wkbType"_s ), mWkbType );
2009 }
2010
2011 QDomElement pkeyElem = pkeyNode.toElement();
2012 if ( !pkeyElem.isNull() )
2013 {
2014 QString encodingString = pkeyElem.attribute( u"encoding"_s );
2015 if ( mDataProvider && !encodingString.isEmpty() )
2016 {
2017 mDataProvider->setEncoding( encodingString );
2018 }
2019 }
2020
2021 // load vector joins - does not resolve references to layers yet
2022 mJoinBuffer->readXml( layer_node );
2023
2024 updateFields();
2025
2026 // If style doesn't include a legend, we'll need to make a default one later...
2027 mSetLegendFromStyle = false;
2028
2029 QString errorMsg;
2030 if ( !readSymbology( layer_node, errorMsg, context ) )
2031 {
2032 return false;
2033 }
2034
2035 readStyleManager( layer_node );
2036
2037 QDomNode depsNode = layer_node.namedItem( u"dataDependencies"_s );
2038 QDomNodeList depsNodes = depsNode.childNodes();
2039 QSet<QgsMapLayerDependency> sources;
2040 for ( int i = 0; i < depsNodes.count(); i++ )
2041 {
2042 QString source = depsNodes.at( i ).toElement().attribute( u"id"_s );
2043 sources << QgsMapLayerDependency( source );
2044 }
2045 setDependencies( sources );
2046
2047 if ( !mSetLegendFromStyle )
2049
2050 // read extent
2052 {
2053 mReadExtentFromXml = true;
2054 }
2055 if ( mReadExtentFromXml )
2056 {
2057 const QDomNode extentNode = layer_node.namedItem( u"extent"_s );
2058 if ( !extentNode.isNull() )
2059 {
2060 mXmlExtent2D = QgsXmlUtils::readRectangle( extentNode.toElement() );
2061 }
2062 const QDomNode extent3DNode = layer_node.namedItem( u"extent3D"_s );
2063 if ( !extent3DNode.isNull() )
2064 {
2065 mXmlExtent3D = QgsXmlUtils::readBox3D( extent3DNode.toElement() );
2066 }
2067 }
2068
2069 // auxiliary layer
2070 const QDomNode asNode = layer_node.namedItem( u"auxiliaryLayer"_s );
2071 const QDomElement asElem = asNode.toElement();
2072 if ( !asElem.isNull() )
2073 {
2074 mAuxiliaryLayerKey = asElem.attribute( u"key"_s );
2075 }
2076
2077 // QGIS Server WMS Dimensions
2078 mServerProperties->readXml( layer_node );
2079
2080 return isValid(); // should be true if read successfully
2081
2082} // void QgsVectorLayer::readXml
2083
2084
2085void QgsVectorLayer::setDataSourcePrivate( const QString &dataSource, const QString &baseName, const QString &provider, const QgsDataProvider::ProviderOptions &options, Qgis::DataProviderReadFlags flags )
2086{
2088
2089 Qgis::GeometryType geomType = geometryType();
2090
2091 mDataSource = dataSource;
2092 setName( baseName );
2093 setDataProvider( provider, options, flags );
2094
2095 if ( !isValid() )
2096 {
2097 return;
2098 }
2099
2100 // Always set crs
2102
2103 bool loadDefaultStyleFlag = false;
2105 {
2106 loadDefaultStyleFlag = true;
2107 }
2108
2109 // reset style if loading default style, style is missing, or geometry type is has changed (and layer is valid)
2110 if ( !renderer() || !legend() || ( isValid() && geomType != geometryType() ) || loadDefaultStyleFlag )
2111 {
2112 std::unique_ptr< QgsScopedRuntimeProfile > profile;
2113 if ( QgsApplication::profiler()->groupIsActive( u"projectload"_s ) )
2114 profile = std::make_unique< QgsScopedRuntimeProfile >( tr( "Load layer style" ), u"projectload"_s );
2115
2116 bool defaultLoadedFlag = false;
2117
2118 // defer style changed signal until we've set the renderer, labeling, everything.
2119 // we don't want multiple signals!
2120 ScopedIntIncrementor styleChangedSignalBlocker( &mBlockStyleChangedSignal );
2121
2122 // need to check whether the default style included a legend, and if not, we need to make a default legend
2123 // later...
2124 mSetLegendFromStyle = false;
2125
2126 // first check if there is a default style / propertysheet defined
2127 // for this layer and if so apply it
2128 // this should take precedence over all
2129 if ( !defaultLoadedFlag && loadDefaultStyleFlag )
2130 {
2131 loadDefaultStyle( defaultLoadedFlag );
2132 }
2133
2134 if ( loadDefaultStyleFlag && !defaultLoadedFlag && isSpatial() && mDataProvider->capabilities() & Qgis::VectorProviderCapability::CreateRenderer )
2135 {
2136 // if we didn't load a default style for this layer, try to create a renderer directly from the data provider
2137 std::unique_ptr< QgsFeatureRenderer > defaultRenderer( mDataProvider->createRenderer() );
2138 if ( defaultRenderer )
2139 {
2140 defaultLoadedFlag = true;
2141 setRenderer( defaultRenderer.release() );
2142 }
2143 }
2144
2145 // if the default style failed to load or was disabled use some very basic defaults
2146 if ( !defaultLoadedFlag )
2147 {
2148 // add single symbol renderer for spatial layers
2150 }
2151
2152 if ( !mSetLegendFromStyle )
2154
2155 if ( mDataProvider->capabilities() & Qgis::VectorProviderCapability::CreateLabeling )
2156 {
2157 std::unique_ptr< QgsAbstractVectorLayerLabeling > defaultLabeling( mDataProvider->createLabeling() );
2158 if ( defaultLabeling )
2159 {
2160 setLabeling( defaultLabeling.release() );
2161 setLabelsEnabled( true );
2162 }
2163 }
2164
2165 styleChangedSignalBlocker.release();
2167 }
2168}
2169
2170QString QgsVectorLayer::loadDefaultStyle( bool &resultFlag )
2171{
2173
2174 // first try to load a user-defined default style - this should always take precedence
2175 QString styleXml = QgsMapLayer::loadDefaultStyle( resultFlag );
2176
2177 if ( resultFlag )
2178 {
2179 // Try to load all stored styles from DB
2180 if ( mLoadAllStoredStyle && mDataProvider && mDataProvider->styleStorageCapabilities().testFlag( Qgis::ProviderStyleStorageCapability::LoadFromDatabase ) )
2181 {
2182 QStringList ids, names, descriptions;
2183 QString errorMessage;
2184 // Get the number of styles related to current layer.
2185 const int relatedStylesCount { listStylesInDatabase( ids, names, descriptions, errorMessage ) };
2186 Q_ASSERT( ids.count() == names.count() );
2187 const QString currentStyleName { mStyleManager->currentStyle() };
2188 for ( int i = 0; i < relatedStylesCount; ++i )
2189 {
2190 if ( names.at( i ) == currentStyleName )
2191 {
2192 continue;
2193 }
2194 errorMessage.clear();
2195 const QString styleXml { getStyleFromDatabase( ids.at( i ), errorMessage ) };
2196 if ( !styleXml.isEmpty() && errorMessage.isEmpty() )
2197 {
2198 mStyleManager->addStyle( names.at( i ), QgsMapLayerStyle( styleXml ) );
2199 }
2200 else
2201 {
2202 QgsDebugMsgLevel( u"Error retrieving style %1 from DB: %2"_s.arg( ids.at( i ), errorMessage ), 2 );
2203 }
2204 }
2205 }
2206 return styleXml;
2207 }
2208
2209 if ( isSpatial() && mDataProvider->capabilities() & Qgis::VectorProviderCapability::CreateRenderer )
2210 {
2211 // otherwise try to create a renderer directly from the data provider
2212 std::unique_ptr< QgsFeatureRenderer > defaultRenderer( mDataProvider->createRenderer() );
2213 if ( defaultRenderer )
2214 {
2215 resultFlag = true;
2216 setRenderer( defaultRenderer.release() );
2217 return QString();
2218 }
2219 }
2220
2221 return QString();
2222}
2223
2224bool QgsVectorLayer::setDataProvider( QString const &provider, const QgsDataProvider::ProviderOptions &options, Qgis::DataProviderReadFlags flags )
2225{
2227
2228 mProviderKey = provider;
2229 delete mDataProvider;
2230
2231 // For Postgres provider primary key unicity is tested at construction time,
2232 // so it has to be set before initializing the provider,
2233 // this manipulation is necessary to preserve default behavior when
2234 // "trust layer metadata" project level option is set and checkPrimaryKeyUnicity
2235 // was not explicitly passed in the uri
2236 if ( provider.compare( "postgres"_L1 ) == 0 )
2237 {
2238 const QString checkUnicityKey { u"checkPrimaryKeyUnicity"_s };
2239 QgsDataSourceUri uri( mDataSource );
2240 if ( !uri.hasParam( checkUnicityKey ) )
2241 {
2242 uri.setParam( checkUnicityKey, mReadExtentFromXml ? "0" : "1" );
2243 mDataSource = uri.uri( false );
2244 }
2245 }
2246
2247 std::unique_ptr< QgsScopedRuntimeProfile > profile;
2248 if ( QgsApplication::profiler()->groupIsActive( u"projectload"_s ) )
2249 profile = std::make_unique< QgsScopedRuntimeProfile >( tr( "Create %1 provider" ).arg( provider ), u"projectload"_s );
2250
2251 if ( mPreloadedProvider )
2252 {
2253 QgsDebugMsgLevel( u"Attaching map layer %1 to preloaded data provider. Provider belongs to thread %2"_s.arg( id(), QgsThreadingUtils::threadDescription( mPreloadedProvider->thread() ) ), 2 );
2254 mDataProvider = qobject_cast< QgsVectorDataProvider * >( mPreloadedProvider.release() );
2255 }
2256 else
2257 {
2258 mDataProvider = qobject_cast<QgsVectorDataProvider *>( QgsProviderRegistry::instance()->createProvider( provider, mDataSource, options, flags ) );
2259 }
2260
2261 if ( !mDataProvider )
2262 {
2263 setValid( false );
2264 QgsDebugMsgLevel( u"Unable to get data provider"_s, 2 );
2265 return false;
2266 }
2267
2268 mDataProvider->setParent( this );
2269 connect( mDataProvider, &QgsVectorDataProvider::raiseError, this, &QgsVectorLayer::raiseError );
2270
2271 QgsDebugMsgLevel( u"Instantiated the data provider plugin"_s, 2 );
2272
2273 setValid( mDataProvider->isValid() );
2274 if ( !isValid() )
2275 {
2276 QgsDebugMsgLevel( u"Invalid provider plugin %1"_s.arg( QString( mDataSource.toUtf8() ) ), 2 );
2277 return false;
2278 }
2279
2280 if ( profile )
2281 profile->switchTask( tr( "Read layer metadata" ) );
2282 if ( mDataProvider->capabilities() & Qgis::VectorProviderCapability::ReadLayerMetadata )
2283 {
2284 // we combine the provider metadata with the layer's existing metadata, so as not to reset any user customizations to the metadata
2285 // back to the default if a layer's data source is changed
2286 QgsLayerMetadata newMetadata = mDataProvider->layerMetadata();
2287 // this overwrites the provider metadata with any properties which are non-empty from the existing layer metadata
2288 newMetadata.combine( &mMetadata );
2289
2290 setMetadata( newMetadata );
2291 QgsDebugMsgLevel( u"Set Data provider QgsLayerMetadata identifier[%1]"_s.arg( metadata().identifier() ), 4 );
2292 }
2293
2294 // TODO: Check if the provider has the capability to send fullExtentCalculated
2295 connect( mDataProvider, &QgsVectorDataProvider::fullExtentCalculated, this, [this] { updateExtents(); } );
2296
2297 // get and store the feature type
2298 mWkbType = mDataProvider->wkbType();
2299
2300 // before we update the layer fields from the provider, we first copy any default set alias and
2301 // editor widget config from the data provider fields, if present
2302 const QgsFields providerFields = mDataProvider->fields();
2303 for ( const QgsField &field : providerFields )
2304 {
2305 // we only copy defaults from the provider if we aren't overriding any configuration made in the layer
2306 if ( !field.editorWidgetSetup().isNull() && mFieldWidgetSetups.value( field.name() ).isNull() )
2307 {
2308 mFieldWidgetSetups[field.name()] = field.editorWidgetSetup();
2309 }
2310 if ( !field.alias().isEmpty() && mAttributeAliasMap.value( field.name() ).isEmpty() )
2311 {
2312 mAttributeAliasMap[field.name()] = field.alias();
2313 }
2314 if ( !mAttributeSplitPolicy.contains( field.name() ) )
2315 {
2316 mAttributeSplitPolicy[field.name()] = field.splitPolicy();
2317 }
2318 if ( !mAttributeDuplicatePolicy.contains( field.name() ) )
2319 {
2320 mAttributeDuplicatePolicy[field.name()] = field.duplicatePolicy();
2321 }
2322 if ( !mAttributeMergePolicy.contains( field.name() ) )
2323 {
2324 mAttributeMergePolicy[field.name()] = field.mergePolicy();
2325 }
2326 }
2327
2328 if ( profile )
2329 profile->switchTask( tr( "Read layer fields" ) );
2330 updateFields();
2331
2332 if ( mProviderKey == "postgres"_L1 )
2333 {
2334 // update datasource from data provider computed one
2335 mDataSource = mDataProvider->dataSourceUri( false );
2336
2337 QgsDebugMsgLevel( u"Beautifying layer name %1"_s.arg( name() ), 3 );
2338
2339 // adjust the display name for postgres layers
2340 const thread_local QRegularExpression reg( R"lit("[^"]+"\."([^"] + )"( \‍([^)]+\))?)lit" );
2341 const QRegularExpressionMatch match = reg.match( name() );
2342 if ( match.hasMatch() )
2343 {
2344 QStringList stuff = match.capturedTexts();
2345 QString lName = stuff[1];
2346
2347 const QMap<QString, QgsMapLayer *> &layers = QgsProject::instance()->mapLayers(); // skip-keyword-check
2348
2349 QMap<QString, QgsMapLayer *>::const_iterator it;
2350 for ( it = layers.constBegin(); it != layers.constEnd() && ( *it )->name() != lName; ++it )
2351 ;
2352
2353 if ( it != layers.constEnd() && stuff.size() > 2 )
2354 {
2355 lName += '.' + stuff[2].mid( 2, stuff[2].length() - 3 );
2356 }
2357
2358 if ( !lName.isEmpty() )
2359 setName( lName );
2360 }
2361 QgsDebugMsgLevel( u"Beautified layer name %1"_s.arg( name() ), 3 );
2362 }
2363 else if ( mProviderKey == "osm"_L1 )
2364 {
2365 // make sure that the "observer" has been removed from URI to avoid crashes
2366 mDataSource = mDataProvider->dataSourceUri();
2367 }
2368 else if ( provider == "ogr"_L1 )
2369 {
2370 // make sure that the /vsigzip or /vsizip is added to uri, if applicable
2371 mDataSource = mDataProvider->dataSourceUri();
2372 if ( mDataSource.right( 10 ) == "|layerid=0"_L1 )
2373 mDataSource.chop( 10 );
2374 }
2375 else if ( provider == "memory"_L1 )
2376 {
2377 // required so that source differs between memory layers
2378 mDataSource = mDataSource + u"&uid=%1"_s.arg( QUuid::createUuid().toString() );
2379 }
2380 else if ( provider == "hana"_L1 )
2381 {
2382 // update datasource from data provider computed one
2383 mDataSource = mDataProvider->dataSourceUri( false );
2384 }
2385
2386 connect( mDataProvider, &QgsVectorDataProvider::dataChanged, this, &QgsVectorLayer::emitDataChanged );
2388
2389 return true;
2390} // QgsVectorLayer:: setDataProvider
2391
2392
2393/* virtual */
2394bool QgsVectorLayer::writeXml( QDomNode &layer_node, QDomDocument &document, const QgsReadWriteContext &context ) const
2395{
2397
2398 // first get the layer element so that we can append the type attribute
2399
2400 QDomElement mapLayerNode = layer_node.toElement();
2401
2402 if ( mapLayerNode.isNull() || ( "maplayer" != mapLayerNode.nodeName() ) )
2403 {
2404 QgsDebugMsgLevel( u"can't find <maplayer>"_s, 2 );
2405 return false;
2406 }
2407
2408 mapLayerNode.setAttribute( u"type"_s, QgsMapLayerFactory::typeToString( Qgis::LayerType::Vector ) );
2409
2410 // set the geometry type
2411 mapLayerNode.setAttribute( u"geometry"_s, QgsWkbTypes::geometryDisplayString( geometryType() ) );
2412 mapLayerNode.setAttribute( u"wkbType"_s, qgsEnumValueToKey( wkbType() ) );
2413
2414 // add provider node
2415 if ( mDataProvider )
2416 {
2417 QDomElement provider = document.createElement( u"provider"_s );
2418 provider.setAttribute( u"encoding"_s, mDataProvider->encoding() );
2419 QDomText providerText = document.createTextNode( providerType() );
2420 provider.appendChild( providerText );
2421 layer_node.appendChild( provider );
2422 }
2423
2424 //save joins
2425 mJoinBuffer->writeXml( layer_node, document );
2426
2427 // dependencies
2428 QDomElement dependenciesElement = document.createElement( u"layerDependencies"_s );
2429 const auto constDependencies = dependencies();
2430 for ( const QgsMapLayerDependency &dep : constDependencies )
2431 {
2433 continue;
2434 QDomElement depElem = document.createElement( u"layer"_s );
2435 depElem.setAttribute( u"id"_s, dep.layerId() );
2436 dependenciesElement.appendChild( depElem );
2437 }
2438 layer_node.appendChild( dependenciesElement );
2439
2440 // change dependencies
2441 QDomElement dataDependenciesElement = document.createElement( u"dataDependencies"_s );
2442 for ( const QgsMapLayerDependency &dep : constDependencies )
2443 {
2444 if ( dep.type() != QgsMapLayerDependency::DataDependency )
2445 continue;
2446 QDomElement depElem = document.createElement( u"layer"_s );
2447 depElem.setAttribute( u"id"_s, dep.layerId() );
2448 dataDependenciesElement.appendChild( depElem );
2449 }
2450 layer_node.appendChild( dataDependenciesElement );
2451
2452 // save expression fields
2453 mExpressionFieldBuffer->writeXml( layer_node, document );
2454
2455 writeStyleManager( layer_node, document );
2456
2457 // auxiliary layer
2458 QDomElement asElem = document.createElement( u"auxiliaryLayer"_s );
2459 if ( mAuxiliaryLayer )
2460 {
2461 const QString pkField = mAuxiliaryLayer->joinInfo().targetFieldName();
2462 asElem.setAttribute( u"key"_s, pkField );
2463 }
2464 layer_node.appendChild( asElem );
2465
2466 // renderer specific settings
2467 QString errorMsg;
2468 return writeSymbology( layer_node, document, errorMsg, context );
2469}
2470
2471QString QgsVectorLayer::encodedSource( const QString &source, const QgsReadWriteContext &context ) const
2472{
2474
2475 if ( providerType() == "memory"_L1 )
2476 {
2477 // Refetch the source from the provider, because adding fields actually changes the source for this provider.
2478 return dataProvider()->dataSourceUri();
2479 }
2480
2482}
2483
2484QString QgsVectorLayer::decodedSource( const QString &source, const QString &provider, const QgsReadWriteContext &context ) const
2485{
2487
2488 return QgsProviderRegistry::instance()->relativeToAbsoluteUri( provider, source, context );
2489}
2490
2491
2499
2500
2501bool QgsVectorLayer::readSymbology( const QDomNode &layerNode, QString &errorMessage, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories )
2502{
2504
2505 if ( categories.testFlag( Fields ) )
2506 {
2507 if ( !mExpressionFieldBuffer )
2508 mExpressionFieldBuffer = new QgsExpressionFieldBuffer();
2509 mExpressionFieldBuffer->readXml( layerNode );
2510
2511 updateFields();
2512 }
2513
2514 if ( categories.testFlag( Relations ) )
2515 {
2516 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Relations" ) );
2517
2518 // Restore referenced layers: relations where "this" is the child layer (the referencing part, that holds the FK)
2519 QDomNodeList referencedLayersNodeList = layerNode.toElement().elementsByTagName( u"referencedLayers"_s );
2520 if ( referencedLayersNodeList.size() > 0 )
2521 {
2522 const QDomNodeList relationNodes { referencedLayersNodeList.at( 0 ).childNodes() };
2523 for ( int i = 0; i < relationNodes.length(); ++i )
2524 {
2525 const QDomElement relationElement = relationNodes.at( i ).toElement();
2526
2527 mWeakRelations.push_back( QgsWeakRelation::readXml( this, QgsWeakRelation::Referencing, relationElement, context.pathResolver() ) );
2528 }
2529 }
2530
2531 // Restore referencing layers: relations where "this" is the parent layer (the referenced part where the FK points to)
2532 QDomNodeList referencingLayersNodeList = layerNode.toElement().elementsByTagName( u"referencingLayers"_s );
2533 if ( referencingLayersNodeList.size() > 0 )
2534 {
2535 const QDomNodeList relationNodes { referencingLayersNodeList.at( 0 ).childNodes() };
2536 for ( int i = 0; i < relationNodes.length(); ++i )
2537 {
2538 const QDomElement relationElement = relationNodes.at( i ).toElement();
2539 mWeakRelations.push_back( QgsWeakRelation::readXml( this, QgsWeakRelation::Referenced, relationElement, context.pathResolver() ) );
2540 }
2541 }
2542 }
2543
2544 QDomElement layerElement = layerNode.toElement();
2545
2546 readCommonStyle( layerElement, context, categories );
2547
2548 readStyle( layerNode, errorMessage, context, categories );
2549
2550 if ( categories.testFlag( MapTips ) )
2551 {
2552 QDomElement mapTipElem = layerNode.namedItem( u"mapTip"_s ).toElement();
2553 setMapTipTemplate( mapTipElem.text() );
2554 setMapTipsEnabled( mapTipElem.attribute( u"enabled"_s, u"1"_s ).toInt() == 1 );
2555 }
2556
2557 if ( categories.testFlag( LayerConfiguration ) )
2558 mDisplayExpression = layerNode.namedItem( u"previewExpression"_s ).toElement().text();
2559
2560 // Try to migrate pre QGIS 3.0 display field property
2561 QString displayField = layerNode.namedItem( u"displayfield"_s ).toElement().text();
2562 if ( mFields.lookupField( displayField ) < 0 )
2563 {
2564 // if it's not a field, it's a maptip
2565 if ( mMapTipTemplate.isEmpty() && categories.testFlag( MapTips ) )
2566 mMapTipTemplate = displayField;
2567 }
2568 else
2569 {
2570 if ( mDisplayExpression.isEmpty() && categories.testFlag( LayerConfiguration ) )
2571 mDisplayExpression = QgsExpression::quotedColumnRef( displayField );
2572 }
2573
2574 // process the attribute actions
2575 if ( categories.testFlag( Actions ) )
2576 mActions->readXml( layerNode, context );
2577
2578 if ( categories.testFlag( Fields ) )
2579 {
2580 // IMPORTANT - we don't clear mAttributeAliasMap here, as it may contain aliases which are coming direct
2581 // from the data provider. Instead we leave any existing aliases and only overwrite them if the style
2582 // has a specific value for that field's alias
2583 QDomNode aliasesNode = layerNode.namedItem( u"aliases"_s );
2584 if ( !aliasesNode.isNull() )
2585 {
2586 QDomElement aliasElem;
2587
2588 QDomNodeList aliasNodeList = aliasesNode.toElement().elementsByTagName( u"alias"_s );
2589 for ( int i = 0; i < aliasNodeList.size(); ++i )
2590 {
2591 aliasElem = aliasNodeList.at( i ).toElement();
2592
2593 QString field;
2594 if ( aliasElem.hasAttribute( u"field"_s ) )
2595 {
2596 field = aliasElem.attribute( u"field"_s );
2597 }
2598 else
2599 {
2600 int index = aliasElem.attribute( u"index"_s ).toInt();
2601
2602 if ( index >= 0 && index < fields().count() )
2603 field = fields().at( index ).name();
2604 }
2605
2606 QString alias;
2607
2608 if ( !aliasElem.attribute( u"name"_s ).isEmpty() )
2609 {
2610 //if it has alias
2611 alias = context.projectTranslator()->translate( u"project:layers:%1:fieldaliases"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ), aliasElem.attribute( u"name"_s ) );
2612 QgsDebugMsgLevel( "context" + u"project:layers:%1:fieldaliases"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ) + " source " + aliasElem.attribute( u"name"_s ), 3 );
2613 }
2614 else
2615 {
2616 //if it has no alias, it should be the fields translation
2617 alias = context.projectTranslator()->translate( u"project:layers:%1:fieldaliases"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ), field );
2618 QgsDebugMsgLevel( "context" + u"project:layers:%1:fieldaliases"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ) + " source " + field, 3 );
2619 //if it gets the exact field value, there has been no translation (or not even translation loaded) - so no alias should be generated;
2620 if ( alias == aliasElem.attribute( u"field"_s ) )
2621 alias.clear();
2622 }
2623
2624 QgsDebugMsgLevel( "field " + field + " origalias " + aliasElem.attribute( u"name"_s ) + " trans " + alias, 3 );
2625 mAttributeAliasMap.insert( field, alias );
2626 }
2627 }
2628
2629 // custom comments
2630 // mAttributeCustomCommentMap is cleared, because when a custom comment is null the provider comment should be considered
2631 mAttributeCustomCommentMap.clear();
2632 QDomNode customCommentsNode = layerNode.namedItem( u"customComments"_s );
2633 if ( !customCommentsNode.isNull() )
2634 {
2635 QDomElement customCommentEntryElem;
2636
2637 QDomNodeList customCommentNodeList = customCommentsNode.toElement().elementsByTagName( u"customComment"_s );
2638 for ( int i = 0; i < customCommentNodeList.size(); ++i )
2639 {
2640 customCommentEntryElem = customCommentNodeList.at( i ).toElement();
2641
2642 const QString field = customCommentEntryElem.attribute( u"field"_s );
2643
2644 //empty values are important as well (to override provider comments with nothing)
2645 const QString customComment = customCommentEntryElem.attribute( u"value"_s );
2646
2647 if ( fields().lookupField( field ) < 0 )
2648 {
2649 QgsDebugMsgLevel( u"Warning: Field %1 not found in layer %2 to load custom comment from setting "_s.arg( field, name() ), 2 );
2650 continue;
2651 }
2652 mAttributeCustomCommentMap.insert( field, customComment );
2653 }
2654 }
2655
2656 // IMPORTANT - we don't clear mAttributeSplitPolicy here, as it may contain policies which are coming direct
2657 // from the data provider. Instead we leave any existing policies and only overwrite them if the style
2658 // has a specific value for that field's policy
2659 const QDomNode splitPoliciesNode = layerNode.namedItem( u"splitPolicies"_s );
2660 if ( !splitPoliciesNode.isNull() )
2661 {
2662 const QDomNodeList splitPolicyNodeList = splitPoliciesNode.toElement().elementsByTagName( u"policy"_s );
2663 for ( int i = 0; i < splitPolicyNodeList.size(); ++i )
2664 {
2665 const QDomElement splitPolicyElem = splitPolicyNodeList.at( i ).toElement();
2666 const QString field = splitPolicyElem.attribute( u"field"_s );
2667 const Qgis::FieldDomainSplitPolicy policy = qgsEnumKeyToValue( splitPolicyElem.attribute( u"policy"_s ), Qgis::FieldDomainSplitPolicy::Duplicate );
2668 mAttributeSplitPolicy.insert( field, policy );
2669 }
2670 }
2671
2672 // The duplicate policy is - unlike alias and split policy - never defined by the data provider, so we clear the map
2673 mAttributeDuplicatePolicy.clear();
2674 const QDomNode duplicatePoliciesNode = layerNode.namedItem( u"duplicatePolicies"_s );
2675 if ( !duplicatePoliciesNode.isNull() )
2676 {
2677 const QDomNodeList duplicatePolicyNodeList = duplicatePoliciesNode.toElement().elementsByTagName( u"policy"_s );
2678 for ( int i = 0; i < duplicatePolicyNodeList.size(); ++i )
2679 {
2680 const QDomElement duplicatePolicyElem = duplicatePolicyNodeList.at( i ).toElement();
2681 const QString field = duplicatePolicyElem.attribute( u"field"_s );
2682 const Qgis::FieldDuplicatePolicy policy = qgsEnumKeyToValue( duplicatePolicyElem.attribute( u"policy"_s ), Qgis::FieldDuplicatePolicy::Duplicate );
2683 mAttributeDuplicatePolicy.insert( field, policy );
2684 }
2685 }
2686
2687 const QDomNode mergePoliciesNode = layerNode.namedItem( u"mergePolicies"_s );
2688 if ( !mergePoliciesNode.isNull() )
2689 {
2690 const QDomNodeList mergePolicyNodeList = mergePoliciesNode.toElement().elementsByTagName( u"policy"_s );
2691 for ( int i = 0; i < mergePolicyNodeList.size(); ++i )
2692 {
2693 const QDomElement mergePolicyElem = mergePolicyNodeList.at( i ).toElement();
2694 const QString field = mergePolicyElem.attribute( u"field"_s );
2695 const Qgis::FieldDomainMergePolicy policy = qgsEnumKeyToValue( mergePolicyElem.attribute( u"policy"_s ), Qgis::FieldDomainMergePolicy::UnsetField );
2696 mAttributeMergePolicy.insert( field, policy );
2697 }
2698 }
2699
2700 // default expressions
2701 mDefaultExpressionMap.clear();
2702 QDomNode defaultsNode = layerNode.namedItem( u"defaults"_s );
2703 if ( !defaultsNode.isNull() )
2704 {
2705 QDomNodeList defaultNodeList = defaultsNode.toElement().elementsByTagName( u"default"_s );
2706 for ( int i = 0; i < defaultNodeList.size(); ++i )
2707 {
2708 QDomElement defaultElem = defaultNodeList.at( i ).toElement();
2709
2710 QString field = defaultElem.attribute( u"field"_s, QString() );
2711 QString expression = defaultElem.attribute( u"expression"_s, QString() );
2712 bool applyOnUpdate = defaultElem.attribute( u"applyOnUpdate"_s, u"0"_s ) == "1"_L1;
2713 if ( field.isEmpty() || expression.isEmpty() )
2714 continue;
2715
2716 mDefaultExpressionMap.insert( field, QgsDefaultValue( expression, applyOnUpdate ) );
2717 }
2718 }
2719
2720 // constraints
2721 mFieldConstraints.clear();
2722 mFieldConstraintStrength.clear();
2723 QDomNode constraintsNode = layerNode.namedItem( u"constraints"_s );
2724 if ( !constraintsNode.isNull() )
2725 {
2726 QDomNodeList constraintNodeList = constraintsNode.toElement().elementsByTagName( u"constraint"_s );
2727 for ( int i = 0; i < constraintNodeList.size(); ++i )
2728 {
2729 QDomElement constraintElem = constraintNodeList.at( i ).toElement();
2730
2731 QString field = constraintElem.attribute( u"field"_s, QString() );
2732 int constraints = constraintElem.attribute( u"constraints"_s, u"0"_s ).toInt();
2733 if ( field.isEmpty() || constraints == 0 )
2734 continue;
2735
2736 mFieldConstraints.insert( field, static_cast< QgsFieldConstraints::Constraints >( constraints ) );
2737
2738 int uniqueStrength = constraintElem.attribute( u"unique_strength"_s, u"1"_s ).toInt();
2739 int notNullStrength = constraintElem.attribute( u"notnull_strength"_s, u"1"_s ).toInt();
2740 int expStrength = constraintElem.attribute( u"exp_strength"_s, u"1"_s ).toInt();
2741
2742 mFieldConstraintStrength.insert( qMakePair( field, QgsFieldConstraints::ConstraintUnique ), static_cast< QgsFieldConstraints::ConstraintStrength >( uniqueStrength ) );
2743 mFieldConstraintStrength.insert( qMakePair( field, QgsFieldConstraints::ConstraintNotNull ), static_cast< QgsFieldConstraints::ConstraintStrength >( notNullStrength ) );
2744 mFieldConstraintStrength.insert( qMakePair( field, QgsFieldConstraints::ConstraintExpression ), static_cast< QgsFieldConstraints::ConstraintStrength >( expStrength ) );
2745 }
2746 }
2747 mFieldConstraintExpressions.clear();
2748 QDomNode constraintExpressionsNode = layerNode.namedItem( u"constraintExpressions"_s );
2749 if ( !constraintExpressionsNode.isNull() )
2750 {
2751 QDomNodeList constraintNodeList = constraintExpressionsNode.toElement().elementsByTagName( u"constraint"_s );
2752 for ( int i = 0; i < constraintNodeList.size(); ++i )
2753 {
2754 QDomElement constraintElem = constraintNodeList.at( i ).toElement();
2755
2756 QString field = constraintElem.attribute( u"field"_s, QString() );
2757 QString exp = constraintElem.attribute( u"exp"_s, QString() );
2758 QString desc
2759 = context.projectTranslator()->translate( u"project:layers:%1:constraintdescriptions"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ), constraintElem.attribute( u"desc"_s, QString() ) );
2760 QgsDebugMsgLevel( "context" + u"project:layers:%1:constraintdescriptions"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text() ) + " source " + constraintElem.attribute( u"desc"_s, QString() ), 3 );
2761 if ( field.isEmpty() || exp.isEmpty() )
2762 continue;
2763
2764 mFieldConstraintExpressions.insert( field, qMakePair( exp, desc ) );
2765 }
2766 }
2767
2768 updateFields();
2769 }
2770
2771 // load field configuration
2772 if ( categories.testFlag( Fields ) || categories.testFlag( Forms ) )
2773 {
2774 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Forms" ) );
2775
2776 QDomElement widgetsElem = layerNode.namedItem( u"fieldConfiguration"_s ).toElement();
2777 QDomNodeList fieldConfigurationElementList = widgetsElem.elementsByTagName( u"field"_s );
2778 for ( int i = 0; i < fieldConfigurationElementList.size(); ++i )
2779 {
2780 const QDomElement fieldConfigElement = fieldConfigurationElementList.at( i ).toElement();
2781 const QDomElement fieldWidgetElement = fieldConfigElement.elementsByTagName( u"editWidget"_s ).at( 0 ).toElement();
2782
2783 QString fieldName = fieldConfigElement.attribute( u"name"_s );
2784
2785 if ( categories.testFlag( Fields ) )
2786 mFieldConfigurationFlags[fieldName] = qgsFlagKeysToValue( fieldConfigElement.attribute( u"configurationFlags"_s ), Qgis::FieldConfigurationFlag::NoFlag );
2787
2788 // load editor widget configuration
2789 if ( categories.testFlag( Forms ) )
2790 {
2791 const QString widgetType = fieldWidgetElement.attribute( u"type"_s );
2792 const QDomElement cfgElem = fieldConfigElement.elementsByTagName( u"config"_s ).at( 0 ).toElement();
2793 const QDomElement optionsElem = cfgElem.childNodes().at( 0 ).toElement();
2794 QVariantMap optionsMap = QgsXmlUtils::readVariant( optionsElem ).toMap();
2795 // translate widget configuration strings
2796 if ( widgetType == "ValueRelation"_L1 )
2797 {
2798 optionsMap[u"Value"_s]
2799 = context.projectTranslator()->translate( u"project:layers:%1:fields:%2:valuerelationvalue"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text(), fieldName ), optionsMap[u"Value"_s].toString() );
2800 optionsMap[u"Description"_s]
2801 = context.projectTranslator()
2802 ->translate( u"project:layers:%1:fields:%2:valuerelationdescription"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text(), fieldName ), optionsMap[u"Description"_s].toString() );
2803 }
2804 if ( widgetType == "ValueMap"_L1 )
2805 {
2806 if ( optionsMap[u"map"_s].canConvert<QList<QVariant>>() )
2807 {
2808 QList<QVariant> translatedValueList;
2809 const QList<QVariant> valueList = optionsMap[u"map"_s].toList();
2810 for ( int i = 0; i < valueList.count(); i++ )
2811 {
2812 QMap<QString, QVariant> translatedValueMap;
2813 QString translatedKey
2814 = context.projectTranslator()
2815 ->translate( u"project:layers:%1:fields:%2:valuemapdescriptions"_s.arg( layerNode.namedItem( u"id"_s ).toElement().text(), fieldName ), valueList[i].toMap().constBegin().key() );
2816 translatedValueMap.insert( translatedKey, valueList[i].toMap().constBegin().value() );
2817 translatedValueList.append( translatedValueMap );
2818 }
2819 optionsMap.insert( u"map"_s, translatedValueList );
2820 }
2821 }
2822 QgsEditorWidgetSetup setup = QgsEditorWidgetSetup( widgetType, optionsMap );
2823 mFieldWidgetSetups[fieldName] = setup;
2824 }
2825 }
2826 }
2827
2828 // Legacy reading for QGIS 3.14 and older projects
2829 // Attributes excluded from WMS and WFS
2830 if ( categories.testFlag( Fields ) )
2831 {
2832 const QList<QPair<QString, Qgis::FieldConfigurationFlag>>
2833 legacyConfig { qMakePair( u"excludeAttributesWMS"_s, Qgis::FieldConfigurationFlag::HideFromWms ), qMakePair( u"excludeAttributesWFS"_s, Qgis::FieldConfigurationFlag::HideFromWfs ) };
2834 for ( const auto &config : legacyConfig )
2835 {
2836 QDomNode excludeNode = layerNode.namedItem( config.first );
2837 if ( !excludeNode.isNull() )
2838 {
2839 QDomNodeList attributeNodeList = excludeNode.toElement().elementsByTagName( u"attribute"_s );
2840 for ( int i = 0; i < attributeNodeList.size(); ++i )
2841 {
2842 QString fieldName = attributeNodeList.at( i ).toElement().text();
2843 if ( !mFieldConfigurationFlags.contains( fieldName ) )
2844 mFieldConfigurationFlags[fieldName] = config.second;
2845 else
2846 mFieldConfigurationFlags[fieldName].setFlag( config.second, true );
2847 }
2848 }
2849 }
2850 }
2851
2852 if ( categories.testFlag( GeometryOptions ) )
2853 mGeometryOptions->readXml( layerNode.namedItem( u"geometryOptions"_s ) );
2854
2855 if ( categories.testFlag( Forms ) )
2856 mEditFormConfig.readXml( layerNode, context );
2857
2858 if ( categories.testFlag( AttributeTable ) )
2859 {
2860 mAttributeTableConfig.readXml( layerNode );
2861 mConditionalStyles->readXml( layerNode, context );
2862 mStoredExpressionManager->readXml( layerNode );
2863 }
2864
2865 if ( categories.testFlag( CustomProperties ) )
2866 readCustomProperties( layerNode, u"variable"_s );
2867
2868 QDomElement mapLayerNode = layerNode.toElement();
2869 if ( categories.testFlag( LayerConfiguration ) && mapLayerNode.attribute( u"readOnly"_s, u"0"_s ).toInt() == 1 )
2870 mReadOnly = true;
2871
2872 updateFields();
2873
2874 if ( categories.testFlag( Legend ) )
2875 {
2876 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Legend" ) );
2877
2878 const QDomElement legendElem = layerNode.firstChildElement( u"legend"_s );
2879 if ( !legendElem.isNull() )
2880 {
2881 std::unique_ptr< QgsMapLayerLegend > legend( QgsMapLayerLegend::defaultVectorLegend( this ) );
2882 legend->readXml( legendElem, context );
2883 setLegend( legend.release() );
2884 mSetLegendFromStyle = true;
2885 }
2886 }
2887
2888 return true;
2889}
2890
2891bool QgsVectorLayer::readStyle( const QDomNode &node, QString &errorMessage, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories )
2892{
2894
2895 bool result = true;
2896 emit readCustomSymbology( node.toElement(), errorMessage );
2897
2898 // we must try to restore a renderer if our geometry type is unknown
2899 // as this allows the renderer to be correctly restored even for layers
2900 // with broken sources
2901 if ( isSpatial() || mWkbType == Qgis::WkbType::Unknown )
2902 {
2903 // defer style changed signal until we've set the renderer, labeling, everything.
2904 // we don't want multiple signals!
2905 ScopedIntIncrementor styleChangedSignalBlocker( &mBlockStyleChangedSignal );
2906
2907 // try renderer v2 first
2908 if ( categories.testFlag( Symbology ) )
2909 {
2910 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Symbology" ) );
2911
2912 QDomElement rendererElement = node.firstChildElement( RENDERER_TAG_NAME );
2913 if ( !rendererElement.isNull() )
2914 {
2915 QgsFeatureRenderer *r = QgsFeatureRenderer::load( rendererElement, context );
2916 if ( r )
2917 {
2918 setRenderer( r );
2919 }
2920 else
2921 {
2922 result = false;
2923 }
2924 }
2925 // make sure layer has a renderer - if none exists, fallback to a default renderer
2926 if ( isSpatial() && !renderer() )
2927 {
2929 }
2930
2931 if ( mSelectionProperties )
2932 mSelectionProperties->readXml( node.toElement(), context );
2933 }
2934
2935 // read labeling definition
2936 if ( categories.testFlag( Labeling ) )
2937 {
2938 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Labeling" ) );
2939
2940 QDomElement labelingElement = node.firstChildElement( u"labeling"_s );
2942 if ( labelingElement.isNull() || ( labelingElement.attribute( u"type"_s ) == "simple"_L1 && labelingElement.firstChildElement( u"settings"_s ).isNull() ) )
2943 {
2944 // make sure we have custom properties for labeling for 2.x projects
2945 // (custom properties should be already loaded when reading the whole layer from XML,
2946 // but when reading style, custom properties are not read)
2947 readCustomProperties( node, u"labeling"_s );
2948
2949 // support for pre-QGIS 3 labeling configurations written in custom properties
2950 labeling = readLabelingFromCustomProperties();
2951 }
2952 else
2953 {
2954 labeling = QgsAbstractVectorLayerLabeling::create( labelingElement, context );
2955 }
2957
2958 if ( node.toElement().hasAttribute( u"labelsEnabled"_s ) )
2959 mLabelsEnabled = node.toElement().attribute( u"labelsEnabled"_s ).toInt();
2960 else
2961 mLabelsEnabled = true;
2962 }
2963
2964 if ( categories.testFlag( Symbology ) )
2965 {
2966 // get and set the blend mode if it exists
2967 QDomNode blendModeNode = node.namedItem( u"blendMode"_s );
2968 if ( !blendModeNode.isNull() )
2969 {
2970 QDomElement e = blendModeNode.toElement();
2971 setBlendMode( QgsPainting::getCompositionMode( static_cast< Qgis::BlendMode >( e.text().toInt() ) ) );
2972 }
2973
2974 // get and set the feature blend mode if it exists
2975 QDomNode featureBlendModeNode = node.namedItem( u"featureBlendMode"_s );
2976 if ( !featureBlendModeNode.isNull() )
2977 {
2978 QDomElement e = featureBlendModeNode.toElement();
2979 setFeatureBlendMode( QgsPainting::getCompositionMode( static_cast< Qgis::BlendMode >( e.text().toInt() ) ) );
2980 }
2981 }
2982
2983 // get and set the layer transparency and scale visibility if they exists
2984 if ( categories.testFlag( Rendering ) )
2985 {
2986 QDomNode layerTransparencyNode = node.namedItem( u"layerTransparency"_s );
2987 if ( !layerTransparencyNode.isNull() )
2988 {
2989 QDomElement e = layerTransparencyNode.toElement();
2990 setOpacity( 1.0 - e.text().toInt() / 100.0 );
2991 }
2992 QDomNode layerOpacityNode = node.namedItem( u"layerOpacity"_s );
2993 if ( !layerOpacityNode.isNull() )
2994 {
2995 QDomElement e = layerOpacityNode.toElement();
2996 setOpacity( e.text().toDouble() );
2997 }
2998
2999 const bool hasScaleBasedVisibiliy { node.attributes().namedItem( u"hasScaleBasedVisibilityFlag"_s ).nodeValue() == '1' };
3000 setScaleBasedVisibility( hasScaleBasedVisibiliy );
3001 bool ok;
3002 const double maxScale { node.attributes().namedItem( u"maxScale"_s ).nodeValue().toDouble( &ok ) };
3003 if ( ok )
3004 {
3005 setMaximumScale( maxScale );
3006 }
3007 const double minScale { node.attributes().namedItem( u"minScale"_s ).nodeValue().toDouble( &ok ) };
3008 if ( ok )
3009 {
3010 setMinimumScale( minScale );
3011 }
3012
3013 QDomElement e = node.toElement();
3014
3015 // get the simplification drawing settings
3016 mSimplifyMethod.setSimplifyHints( static_cast< Qgis::VectorRenderingSimplificationFlags >( e.attribute( u"simplifyDrawingHints"_s, u"1"_s ).toInt() ) );
3017 mSimplifyMethod.setSimplifyAlgorithm( static_cast< Qgis::VectorSimplificationAlgorithm >( e.attribute( u"simplifyAlgorithm"_s, u"0"_s ).toInt() ) );
3018 mSimplifyMethod.setThreshold( e.attribute( u"simplifyDrawingTol"_s, u"1"_s ).toFloat() );
3019 mSimplifyMethod.setForceLocalOptimization( e.attribute( u"simplifyLocal"_s, u"1"_s ).toInt() );
3020 mSimplifyMethod.setMaximumScale( e.attribute( u"simplifyMaxScale"_s, u"1"_s ).toFloat() );
3021
3022 if ( mRenderer )
3023 mRenderer->setReferenceScale( e.attribute( u"symbologyReferenceScale"_s, u"-1"_s ).toDouble() );
3024 }
3025
3026 //diagram renderer and diagram layer settings
3027 if ( categories.testFlag( Diagrams ) )
3028 {
3029 QgsReadWriteContextCategoryPopper p = context.enterCategory( tr( "Diagrams" ) );
3030
3031 delete mDiagramRenderer;
3032 mDiagramRenderer = nullptr;
3033 QDomElement singleCatDiagramElem = node.firstChildElement( u"SingleCategoryDiagramRenderer"_s );
3034 if ( !singleCatDiagramElem.isNull() )
3035 {
3036 mDiagramRenderer = new QgsSingleCategoryDiagramRenderer();
3037 mDiagramRenderer->readXml( singleCatDiagramElem, context );
3038 }
3039 QDomElement linearDiagramElem = node.firstChildElement( u"LinearlyInterpolatedDiagramRenderer"_s );
3040 if ( !linearDiagramElem.isNull() )
3041 {
3042 if ( linearDiagramElem.hasAttribute( u"classificationAttribute"_s ) )
3043 {
3044 // fix project from before QGIS 3.0
3045 int idx = linearDiagramElem.attribute( u"classificationAttribute"_s ).toInt();
3046 if ( idx >= 0 && idx < mFields.count() )
3047 linearDiagramElem.setAttribute( u"classificationField"_s, mFields.at( idx ).name() );
3048 }
3049
3050 mDiagramRenderer = new QgsLinearlyInterpolatedDiagramRenderer();
3051 mDiagramRenderer->readXml( linearDiagramElem, context );
3052 }
3053 QDomElement stackedDiagramElem = node.firstChildElement( u"StackedDiagramRenderer"_s );
3054 if ( !stackedDiagramElem.isNull() )
3055 {
3056 mDiagramRenderer = new QgsStackedDiagramRenderer();
3057 mDiagramRenderer->readXml( stackedDiagramElem, context );
3058 }
3059
3060 if ( mDiagramRenderer )
3061 {
3062 QDomElement diagramSettingsElem = node.firstChildElement( u"DiagramLayerSettings"_s );
3063 if ( !diagramSettingsElem.isNull() )
3064 {
3065 bool oldXPos = diagramSettingsElem.hasAttribute( u"xPosColumn"_s );
3066 bool oldYPos = diagramSettingsElem.hasAttribute( u"yPosColumn"_s );
3067 bool oldShow = diagramSettingsElem.hasAttribute( u"showColumn"_s );
3068 if ( oldXPos || oldYPos || oldShow )
3069 {
3070 // fix project from before QGIS 3.0
3072 if ( oldXPos )
3073 {
3074 int xPosColumn = diagramSettingsElem.attribute( u"xPosColumn"_s ).toInt();
3075 if ( xPosColumn >= 0 && xPosColumn < mFields.count() )
3076 ddp.setProperty( QgsDiagramLayerSettings::Property::PositionX, QgsProperty::fromField( mFields.at( xPosColumn ).name(), true ) );
3077 }
3078 if ( oldYPos )
3079 {
3080 int yPosColumn = diagramSettingsElem.attribute( u"yPosColumn"_s ).toInt();
3081 if ( yPosColumn >= 0 && yPosColumn < mFields.count() )
3082 ddp.setProperty( QgsDiagramLayerSettings::Property::PositionY, QgsProperty::fromField( mFields.at( yPosColumn ).name(), true ) );
3083 }
3084 if ( oldShow )
3085 {
3086 int showColumn = diagramSettingsElem.attribute( u"showColumn"_s ).toInt();
3087 if ( showColumn >= 0 && showColumn < mFields.count() )
3088 ddp.setProperty( QgsDiagramLayerSettings::Property::Show, QgsProperty::fromField( mFields.at( showColumn ).name(), true ) );
3089 }
3090 QDomElement propertiesElem = diagramSettingsElem.ownerDocument().createElement( u"properties"_s );
3092 { static_cast< int >( QgsDiagramLayerSettings::Property::PositionX ), QgsPropertyDefinition( "positionX", QObject::tr( "Position (X)" ), QgsPropertyDefinition::Double ) },
3093 { static_cast< int >( QgsDiagramLayerSettings::Property::PositionY ), QgsPropertyDefinition( "positionY", QObject::tr( "Position (Y)" ), QgsPropertyDefinition::Double ) },
3094 { static_cast< int >( QgsDiagramLayerSettings::Property::Show ), QgsPropertyDefinition( "show", QObject::tr( "Show diagram" ), QgsPropertyDefinition::Boolean ) },
3095 };
3096 ddp.writeXml( propertiesElem, defs );
3097 diagramSettingsElem.appendChild( propertiesElem );
3098 }
3099
3100 delete mDiagramLayerSettings;
3101 mDiagramLayerSettings = new QgsDiagramLayerSettings();
3102 mDiagramLayerSettings->readXml( diagramSettingsElem );
3103 }
3104 }
3105 }
3106 // end diagram
3107
3108 styleChangedSignalBlocker.release();
3110 }
3111 return result;
3112}
3113
3114
3115bool QgsVectorLayer::writeSymbology( QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories ) const
3116{
3118
3119 QDomElement layerElement = node.toElement();
3120 writeCommonStyle( layerElement, doc, context, categories );
3121
3122 ( void ) writeStyle( node, doc, errorMessage, context, categories );
3123
3124 if ( categories.testFlag( GeometryOptions ) )
3125 mGeometryOptions->writeXml( node );
3126
3127 if ( categories.testFlag( Legend ) && legend() )
3128 {
3129 QDomElement legendElement = legend()->writeXml( doc, context );
3130 if ( !legendElement.isNull() )
3131 node.appendChild( legendElement );
3132 }
3133
3134 // Relation information for both referenced and referencing sides
3135 if ( categories.testFlag( Relations ) )
3136 {
3137 if ( QgsProject *p = project() )
3138 {
3139 // Store referenced layers: relations where "this" is the child layer (the referencing part, that holds the FK)
3140 QDomElement referencedLayersElement = doc.createElement( u"referencedLayers"_s );
3141 node.appendChild( referencedLayersElement );
3142
3143 const QList<QgsRelation> referencingRelations { p->relationManager()->referencingRelations( this ) };
3144 for ( const QgsRelation &rel : referencingRelations )
3145 {
3146 switch ( rel.type() )
3147 {
3149 QgsWeakRelation::writeXml( this, QgsWeakRelation::Referencing, rel, referencedLayersElement, doc );
3150 break;
3152 break;
3153 }
3154 }
3155
3156 // Store referencing layers: relations where "this" is the parent layer (the referenced part, that holds the FK)
3157 QDomElement referencingLayersElement = doc.createElement( u"referencingLayers"_s );
3158 node.appendChild( referencingLayersElement );
3159
3160 const QList<QgsRelation> referencedRelations { p->relationManager()->referencedRelations( this ) };
3161 for ( const QgsRelation &rel : referencedRelations )
3162 {
3163 switch ( rel.type() )
3164 {
3166 QgsWeakRelation::writeXml( this, QgsWeakRelation::Referenced, rel, referencingLayersElement, doc );
3167 break;
3169 break;
3170 }
3171 }
3172 }
3173 }
3174
3175 // write field configurations
3176 if ( categories.testFlag( Fields ) || categories.testFlag( Forms ) )
3177 {
3178 QDomElement fieldConfigurationElement;
3179 // field configuration flag
3180 fieldConfigurationElement = doc.createElement( u"fieldConfiguration"_s );
3181 node.appendChild( fieldConfigurationElement );
3182
3183 for ( const QgsField &field : std::as_const( mFields ) )
3184 {
3185 QDomElement fieldElement = doc.createElement( u"field"_s );
3186 fieldElement.setAttribute( u"name"_s, field.name() );
3187 fieldConfigurationElement.appendChild( fieldElement );
3188
3189 if ( categories.testFlag( Fields ) )
3190 {
3191 fieldElement.setAttribute( u"configurationFlags"_s, qgsFlagValueToKeys( field.configurationFlags() ) );
3192 }
3193
3194 if ( categories.testFlag( Forms ) )
3195 {
3196 QgsEditorWidgetSetup widgetSetup = field.editorWidgetSetup();
3197
3198 // TODO : wrap this part in an if to only save if it was user-modified
3199 QDomElement editWidgetElement = doc.createElement( u"editWidget"_s );
3200 fieldElement.appendChild( editWidgetElement );
3201 editWidgetElement.setAttribute( u"type"_s, field.editorWidgetSetup().type() );
3202 QDomElement editWidgetConfigElement = doc.createElement( u"config"_s );
3203
3204 editWidgetConfigElement.appendChild( QgsXmlUtils::writeVariant( widgetSetup.config(), doc ) );
3205 editWidgetElement.appendChild( editWidgetConfigElement );
3206 // END TODO : wrap this part in an if to only save if it was user-modified
3207 }
3208 }
3209 }
3210
3211 if ( categories.testFlag( Fields ) )
3212 {
3213 //attribute aliases
3214 QDomElement aliasElem = doc.createElement( u"aliases"_s );
3215 for ( const QgsField &field : std::as_const( mFields ) )
3216 {
3217 QDomElement aliasEntryElem = doc.createElement( u"alias"_s );
3218 aliasEntryElem.setAttribute( u"field"_s, field.name() );
3219 aliasEntryElem.setAttribute( u"index"_s, mFields.indexFromName( field.name() ) );
3220 aliasEntryElem.setAttribute( u"name"_s, field.alias() );
3221 aliasElem.appendChild( aliasEntryElem );
3222 }
3223 node.appendChild( aliasElem );
3224
3225 //custom comments
3226 QDomElement customCommentElem = doc.createElement( u"customComments"_s );
3227 bool hasCustomComments = false;
3228 for ( const QgsField &field : std::as_const( mFields ) )
3229 {
3230 //if empty ("") we store it, if null we don't store it
3231 const QString customComment = field.customComment();
3232 if ( customComment.isNull() )
3233 continue;
3234
3235 hasCustomComments = true;
3236 QDomElement customCommentEntryElem = doc.createElement( u"customComment"_s );
3237 customCommentEntryElem.setAttribute( u"field"_s, field.name() );
3238 customCommentEntryElem.setAttribute( u"value"_s, customComment );
3239 customCommentElem.appendChild( customCommentEntryElem );
3240 }
3241 if ( hasCustomComments )
3242 {
3243 node.appendChild( customCommentElem );
3244 }
3245
3246 //split policies
3247 {
3248 QDomElement splitPoliciesElement = doc.createElement( u"splitPolicies"_s );
3249 bool hasNonDefaultSplitPolicies = false;
3250 for ( const QgsField &field : std::as_const( mFields ) )
3251 {
3252 if ( field.splitPolicy() != Qgis::FieldDomainSplitPolicy::Duplicate )
3253 {
3254 QDomElement splitPolicyElem = doc.createElement( u"policy"_s );
3255 splitPolicyElem.setAttribute( u"field"_s, field.name() );
3256 splitPolicyElem.setAttribute( u"policy"_s, qgsEnumValueToKey( field.splitPolicy() ) );
3257 splitPoliciesElement.appendChild( splitPolicyElem );
3258 hasNonDefaultSplitPolicies = true;
3259 }
3260 }
3261 if ( hasNonDefaultSplitPolicies )
3262 node.appendChild( splitPoliciesElement );
3263 }
3264
3265 //duplicate policies
3266 {
3267 QDomElement duplicatePoliciesElement = doc.createElement( u"duplicatePolicies"_s );
3268 bool hasNonDefaultDuplicatePolicies = false;
3269 for ( const QgsField &field : std::as_const( mFields ) )
3270 {
3271 if ( field.duplicatePolicy() != Qgis::FieldDuplicatePolicy::Duplicate )
3272 {
3273 QDomElement duplicatePolicyElem = doc.createElement( u"policy"_s );
3274 duplicatePolicyElem.setAttribute( u"field"_s, field.name() );
3275 duplicatePolicyElem.setAttribute( u"policy"_s, qgsEnumValueToKey( field.duplicatePolicy() ) );
3276 duplicatePoliciesElement.appendChild( duplicatePolicyElem );
3277 hasNonDefaultDuplicatePolicies = true;
3278 }
3279 }
3280 if ( hasNonDefaultDuplicatePolicies )
3281 node.appendChild( duplicatePoliciesElement );
3282 }
3283
3284 //merge policies
3285 {
3286 QDomElement mergePoliciesElement = doc.createElement( u"mergePolicies"_s );
3287 bool hasNonDefaultMergePolicies = false;
3288 for ( const QgsField &field : std::as_const( mFields ) )
3289 {
3290 if ( field.mergePolicy() != Qgis::FieldDomainMergePolicy::UnsetField )
3291 {
3292 QDomElement mergePolicyElem = doc.createElement( u"policy"_s );
3293 mergePolicyElem.setAttribute( u"field"_s, field.name() );
3294 mergePolicyElem.setAttribute( u"policy"_s, qgsEnumValueToKey( field.mergePolicy() ) );
3295 mergePoliciesElement.appendChild( mergePolicyElem );
3296 hasNonDefaultMergePolicies = true;
3297 }
3298 }
3299 if ( hasNonDefaultMergePolicies )
3300 node.appendChild( mergePoliciesElement );
3301 }
3302
3303 //default expressions
3304 QDomElement defaultsElem = doc.createElement( u"defaults"_s );
3305 for ( const QgsField &field : std::as_const( mFields ) )
3306 {
3307 QDomElement defaultElem = doc.createElement( u"default"_s );
3308 defaultElem.setAttribute( u"field"_s, field.name() );
3309 defaultElem.setAttribute( u"expression"_s, field.defaultValueDefinition().expression() );
3310 defaultElem.setAttribute( u"applyOnUpdate"_s, field.defaultValueDefinition().applyOnUpdate() ? u"1"_s : u"0"_s );
3311 defaultsElem.appendChild( defaultElem );
3312 }
3313 node.appendChild( defaultsElem );
3314
3315 // constraints
3316 QDomElement constraintsElem = doc.createElement( u"constraints"_s );
3317 for ( const QgsField &field : std::as_const( mFields ) )
3318 {
3319 QDomElement constraintElem = doc.createElement( u"constraint"_s );
3320 constraintElem.setAttribute( u"field"_s, field.name() );
3321 constraintElem.setAttribute( u"constraints"_s, field.constraints().constraints() );
3322 constraintElem.setAttribute( u"unique_strength"_s, field.constraints().constraintStrength( QgsFieldConstraints::ConstraintUnique ) );
3323 constraintElem.setAttribute( u"notnull_strength"_s, field.constraints().constraintStrength( QgsFieldConstraints::ConstraintNotNull ) );
3324 constraintElem.setAttribute( u"exp_strength"_s, field.constraints().constraintStrength( QgsFieldConstraints::ConstraintExpression ) );
3325
3326 constraintsElem.appendChild( constraintElem );
3327 }
3328 node.appendChild( constraintsElem );
3329
3330 // constraint expressions
3331 QDomElement constraintExpressionsElem = doc.createElement( u"constraintExpressions"_s );
3332 for ( const QgsField &field : std::as_const( mFields ) )
3333 {
3334 QDomElement constraintExpressionElem = doc.createElement( u"constraint"_s );
3335 constraintExpressionElem.setAttribute( u"field"_s, field.name() );
3336 constraintExpressionElem.setAttribute( u"exp"_s, field.constraints().constraintExpression() );
3337 constraintExpressionElem.setAttribute( u"desc"_s, field.constraints().constraintDescription() );
3338 constraintExpressionsElem.appendChild( constraintExpressionElem );
3339 }
3340 node.appendChild( constraintExpressionsElem );
3341
3342 // save expression fields
3343 if ( !mExpressionFieldBuffer )
3344 {
3345 // can happen when saving style on a invalid layer
3347 dummy.writeXml( node, doc );
3348 }
3349 else
3350 {
3351 mExpressionFieldBuffer->writeXml( node, doc );
3352 }
3353 }
3354
3355 // add attribute actions
3356 if ( categories.testFlag( Actions ) )
3357 mActions->writeXml( node );
3358
3359 if ( categories.testFlag( AttributeTable ) )
3360 {
3361 mAttributeTableConfig.writeXml( node );
3362 mConditionalStyles->writeXml( node, doc, context );
3363 mStoredExpressionManager->writeXml( node );
3364 }
3365
3366 if ( categories.testFlag( Forms ) )
3367 mEditFormConfig.writeXml( node, context );
3368
3369 // save readonly state
3370 if ( categories.testFlag( LayerConfiguration ) )
3371 node.toElement().setAttribute( u"readOnly"_s, mReadOnly );
3372
3373 // save preview expression
3374 if ( categories.testFlag( LayerConfiguration ) )
3375 {
3376 QDomElement prevExpElem = doc.createElement( u"previewExpression"_s );
3377 QDomText prevExpText = doc.createTextNode( mDisplayExpression );
3378 prevExpElem.appendChild( prevExpText );
3379 node.appendChild( prevExpElem );
3380 }
3381
3382 // save map tip
3383 if ( categories.testFlag( MapTips ) )
3384 {
3385 QDomElement mapTipElem = doc.createElement( u"mapTip"_s );
3386 mapTipElem.setAttribute( u"enabled"_s, mapTipsEnabled() );
3387 QDomText mapTipText = doc.createTextNode( mMapTipTemplate );
3388 mapTipElem.appendChild( mapTipText );
3389 node.toElement().appendChild( mapTipElem );
3390 }
3391
3392 return true;
3393}
3394
3395bool QgsVectorLayer::writeStyle( QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories ) const
3396{
3398
3399 QDomElement mapLayerNode = node.toElement();
3400
3401 emit writeCustomSymbology( mapLayerNode, doc, errorMessage );
3402
3403 // we must try to write the renderer if our geometry type is unknown
3404 // as this allows the renderer to be correctly restored even for layers
3405 // with broken sources
3406 if ( isSpatial() || mWkbType == Qgis::WkbType::Unknown )
3407 {
3408 if ( categories.testFlag( Symbology ) )
3409 {
3410 if ( mRenderer )
3411 {
3412 QDomElement rendererElement = mRenderer->save( doc, context );
3413 node.appendChild( rendererElement );
3414 }
3415 if ( mSelectionProperties )
3416 {
3417 mSelectionProperties->writeXml( mapLayerNode, doc, context );
3418 }
3419 }
3420
3421 if ( categories.testFlag( Labeling ) )
3422 {
3423 if ( mLabeling )
3424 {
3425 QDomElement labelingElement = mLabeling->save( doc, context );
3426 node.appendChild( labelingElement );
3427 }
3428 mapLayerNode.setAttribute( u"labelsEnabled"_s, mLabelsEnabled ? u"1"_s : u"0"_s );
3429 }
3430
3431 // save the simplification drawing settings
3432 if ( categories.testFlag( Rendering ) )
3433 {
3434 mapLayerNode.setAttribute( u"simplifyDrawingHints"_s, QString::number( static_cast< int >( mSimplifyMethod.simplifyHints() ) ) );
3435 mapLayerNode.setAttribute( u"simplifyAlgorithm"_s, QString::number( static_cast< int >( mSimplifyMethod.simplifyAlgorithm() ) ) );
3436 mapLayerNode.setAttribute( u"simplifyDrawingTol"_s, QString::number( mSimplifyMethod.threshold() ) );
3437 mapLayerNode.setAttribute( u"simplifyLocal"_s, mSimplifyMethod.forceLocalOptimization() ? 1 : 0 );
3438 mapLayerNode.setAttribute( u"simplifyMaxScale"_s, QString::number( mSimplifyMethod.maximumScale() ) );
3439 }
3440
3441 //save customproperties
3442 if ( categories.testFlag( CustomProperties ) )
3443 {
3444 writeCustomProperties( node, doc );
3445 }
3446
3447 if ( categories.testFlag( Symbology ) )
3448 {
3449 // add the blend mode field
3450 QDomElement blendModeElem = doc.createElement( u"blendMode"_s );
3451 QDomText blendModeText = doc.createTextNode( QString::number( static_cast< int >( QgsPainting::getBlendModeEnum( blendMode() ) ) ) );
3452 blendModeElem.appendChild( blendModeText );
3453 node.appendChild( blendModeElem );
3454
3455 // add the feature blend mode field
3456 QDomElement featureBlendModeElem = doc.createElement( u"featureBlendMode"_s );
3457 QDomText featureBlendModeText = doc.createTextNode( QString::number( static_cast< int >( QgsPainting::getBlendModeEnum( featureBlendMode() ) ) ) );
3458 featureBlendModeElem.appendChild( featureBlendModeText );
3459 node.appendChild( featureBlendModeElem );
3460 }
3461
3462 // add the layer opacity and scale visibility
3463 if ( categories.testFlag( Rendering ) )
3464 {
3465 QDomElement layerOpacityElem = doc.createElement( u"layerOpacity"_s );
3466 QDomText layerOpacityText = doc.createTextNode( QString::number( opacity() ) );
3467 layerOpacityElem.appendChild( layerOpacityText );
3468 node.appendChild( layerOpacityElem );
3469 mapLayerNode.setAttribute( u"hasScaleBasedVisibilityFlag"_s, hasScaleBasedVisibility() ? 1 : 0 );
3470 mapLayerNode.setAttribute( u"maxScale"_s, maximumScale() );
3471 mapLayerNode.setAttribute( u"minScale"_s, minimumScale() );
3472
3473 mapLayerNode.setAttribute( u"symbologyReferenceScale"_s, mRenderer ? mRenderer->referenceScale() : -1 );
3474 }
3475
3476 if ( categories.testFlag( Diagrams ) && mDiagramRenderer )
3477 {
3478 mDiagramRenderer->writeXml( mapLayerNode, doc, context );
3479 if ( mDiagramLayerSettings )
3480 mDiagramLayerSettings->writeXml( mapLayerNode, doc );
3481 }
3482 }
3483 return true;
3484}
3485
3486bool QgsVectorLayer::readSld( const QDomNode &node, QString &errorMessage )
3487{
3489
3490 // get the Name element
3491 QDomElement nameElem = node.firstChildElement( u"Name"_s );
3492 if ( nameElem.isNull() )
3493 {
3494 errorMessage = u"Warning: Name element not found within NamedLayer while it's required."_s;
3495 }
3496
3497 if ( isSpatial() )
3498 {
3499 QgsFeatureRenderer *r = QgsFeatureRenderer::loadSld( node, geometryType(), errorMessage );
3500 if ( !r )
3501 return false;
3502
3503 // defer style changed signal until we've set the renderer, labeling, everything.
3504 // we don't want multiple signals!
3505 ScopedIntIncrementor styleChangedSignalBlocker( &mBlockStyleChangedSignal );
3506
3507 setRenderer( r );
3508
3509 // labeling
3510 readSldLabeling( node );
3511
3512 styleChangedSignalBlocker.release();
3514 }
3515 return true;
3516}
3517
3518bool QgsVectorLayer::writeSld( QDomNode &node, QDomDocument &doc, QString &, const QVariantMap &props ) const
3519{
3521 QgsSldExportContext context;
3522 context.setExtraProperties( props );
3523 writeSld( node, doc, context );
3524 return true;
3525}
3526
3527bool QgsVectorLayer::writeSld( QDomNode &node, QDomDocument &doc, QgsSldExportContext &context ) const
3528{
3530
3531 QVariantMap localProps = context.extraProperties();
3533 {
3535 }
3536 context.setExtraProperties( localProps );
3537
3538 if ( isSpatial() )
3539 {
3540 // store the Name element
3541 QDomElement nameNode = doc.createElement( u"se:Name"_s );
3542 nameNode.appendChild( doc.createTextNode( name() ) );
3543 node.appendChild( nameNode );
3544
3545 QDomElement userStyleElem = doc.createElement( u"UserStyle"_s );
3546 node.appendChild( userStyleElem );
3547
3548 QDomElement nameElem = doc.createElement( u"se:Name"_s );
3549 nameElem.appendChild( doc.createTextNode( name() ) );
3550
3551 userStyleElem.appendChild( nameElem );
3552
3553 QDomElement featureTypeStyleElem = doc.createElement( u"se:FeatureTypeStyle"_s );
3554 userStyleElem.appendChild( featureTypeStyleElem );
3555
3556 mRenderer->toSld( doc, featureTypeStyleElem, context );
3557 if ( labelsEnabled() )
3558 {
3559 mLabeling->toSld( featureTypeStyleElem, context );
3560 }
3561 }
3562 return true;
3563}
3564
3565bool QgsVectorLayer::changeGeometry( QgsFeatureId fid, QgsGeometry &geom, bool skipDefaultValue )
3566{
3568
3569 if ( !mEditBuffer || !mDataProvider )
3570 {
3571 return false;
3572 }
3573
3574 if ( mGeometryOptions->isActive() )
3575 mGeometryOptions->apply( geom );
3576
3577 updateExtents();
3578
3579 bool result = mEditBuffer->changeGeometry( fid, geom );
3580
3581 if ( result )
3582 {
3583 updateExtents();
3584 if ( !skipDefaultValue && !mDefaultValueOnUpdateFields.isEmpty() )
3585 updateDefaultValues( fid );
3586 }
3587 return result;
3588}
3589
3590
3591bool QgsVectorLayer::changeAttributeValue( QgsFeatureId fid, int field, const QVariant &newValue, const QVariant &oldValue, bool skipDefaultValues, QgsVectorLayerToolsContext *context )
3592{
3594
3595 bool result = false;
3596
3597 switch ( fields().fieldOrigin( field ) )
3598 {
3600 result = mJoinBuffer->changeAttributeValue( fid, field, newValue, oldValue );
3601 if ( result )
3602 emit attributeValueChanged( fid, field, newValue );
3603 break;
3604
3608 {
3609 if ( mEditBuffer && mDataProvider )
3610 result = mEditBuffer->changeAttributeValue( fid, field, newValue, oldValue );
3611 break;
3612 }
3613
3615 break;
3616 }
3617
3618 if ( result && !skipDefaultValues && !mDefaultValueOnUpdateFields.isEmpty() )
3619 updateDefaultValues( fid, QgsFeature(), context ? context->expressionContext() : nullptr );
3620
3621 return result;
3622}
3623
3624bool QgsVectorLayer::changeAttributeValues( QgsFeatureId fid, const QgsAttributeMap &newValues, const QgsAttributeMap &oldValues, bool skipDefaultValues, QgsVectorLayerToolsContext *context )
3625{
3627
3628 bool result = true;
3629
3630 QgsAttributeMap newValuesJoin;
3631 QgsAttributeMap oldValuesJoin;
3632
3633 QgsAttributeMap newValuesNotJoin;
3634 QgsAttributeMap oldValuesNotJoin;
3635
3636 for ( auto it = newValues.constBegin(); it != newValues.constEnd(); ++it )
3637 {
3638 const int field = it.key();
3639 const QVariant newValue = it.value();
3640 QVariant oldValue;
3641
3642 if ( oldValues.contains( field ) )
3643 oldValue = oldValues[field];
3644
3645 switch ( fields().fieldOrigin( field ) )
3646 {
3648 newValuesJoin[field] = newValue;
3649 oldValuesJoin[field] = oldValue;
3650 break;
3651
3655 {
3656 newValuesNotJoin[field] = newValue;
3657 oldValuesNotJoin[field] = oldValue;
3658 break;
3659 }
3660
3662 break;
3663 }
3664 }
3665
3666 if ( !newValuesJoin.isEmpty() && mJoinBuffer )
3667 {
3668 result = mJoinBuffer->changeAttributeValues( fid, newValuesJoin, oldValuesJoin );
3669 }
3670
3671 if ( !newValuesNotJoin.isEmpty() )
3672 {
3673 if ( mEditBuffer && mDataProvider )
3674 result &= mEditBuffer->changeAttributeValues( fid, newValuesNotJoin, oldValues );
3675 else
3676 result = false;
3677 }
3678
3679 if ( result && !skipDefaultValues && !mDefaultValueOnUpdateFields.isEmpty() )
3680 {
3681 updateDefaultValues( fid, QgsFeature(), context ? context->expressionContext() : nullptr );
3682 }
3683
3684 return result;
3685}
3686
3688{
3690
3691 if ( !mEditBuffer || !mDataProvider )
3692 return false;
3693
3694 return mEditBuffer->addAttribute( field );
3695}
3696
3698{
3700
3701 if ( attIndex < 0 || attIndex >= fields().count() )
3702 return;
3703
3704 QString name = fields().at( attIndex ).name();
3705 mFields[attIndex].setAlias( QString() );
3706 if ( mAttributeAliasMap.contains( name ) )
3707 {
3708 mAttributeAliasMap.remove( name );
3709 updateFields();
3710 mEditFormConfig.setFields( mFields );
3711 emit layerModified();
3712 }
3713}
3714
3715bool QgsVectorLayer::renameAttribute( int index, const QString &newName )
3716{
3718
3719 if ( index < 0 || index >= fields().count() )
3720 return false;
3721
3722 switch ( mFields.fieldOrigin( index ) )
3723 {
3725 {
3726 if ( mExpressionFieldBuffer )
3727 {
3728 int oi = mFields.fieldOriginIndex( index );
3729 mExpressionFieldBuffer->renameExpression( oi, newName );
3730 updateFields();
3731 return true;
3732 }
3733 else
3734 {
3735 return false;
3736 }
3737 }
3738
3741
3742 if ( !mEditBuffer || !mDataProvider )
3743 return false;
3744
3745 return mEditBuffer->renameAttribute( index, newName );
3746
3749 return false;
3750 }
3751
3752 return false; // avoid warning
3753}
3754
3755void QgsVectorLayer::setFieldAlias( int attIndex, const QString &aliasString )
3756{
3758
3759 if ( attIndex < 0 || attIndex >= fields().count() )
3760 return;
3761
3762 QString name = fields().at( attIndex ).name();
3763
3764 mAttributeAliasMap.insert( name, aliasString );
3765 mFields[attIndex].setAlias( aliasString );
3766 mEditFormConfig.setFields( mFields );
3767 emit layerModified(); // TODO[MD]: should have a different signal?
3768}
3769
3770QString QgsVectorLayer::attributeAlias( int index ) const
3771{
3773
3774 if ( index < 0 || index >= fields().count() )
3775 return QString();
3776
3777 return fields().at( index ).alias();
3778}
3779
3780void QgsVectorLayer::setFieldCustomComment( int attIndex, const QString &customCommentString )
3781{
3783
3784 if ( attIndex < 0 || attIndex >= fields().count() )
3785 return;
3786
3787 QString name = fields().at( attIndex ).name();
3788
3789 mAttributeCustomCommentMap.insert( name, customCommentString );
3790 mFields[attIndex].setCustomComment( customCommentString );
3791 mEditFormConfig.setFields( mFields );
3792 emit layerModified();
3793}
3794
3796{
3798
3799 if ( attIndex < 0 || attIndex >= fields().count() )
3800 return;
3801
3802 QString name = fields().at( attIndex ).name();
3803 mFields[attIndex].setCustomComment( QString() );
3804 if ( mAttributeCustomCommentMap.contains( name ) )
3805 {
3806 mAttributeCustomCommentMap.remove( name );
3807 updateFields();
3808 mEditFormConfig.setFields( mFields );
3809 emit layerModified();
3810 }
3811}
3812
3814{
3816
3817 if ( index < 0 || index >= fields().count() )
3818 return QString();
3819
3820 return fields().at( index ).customComment();
3821}
3822
3824{
3826
3827 return mAttributeCustomCommentMap;
3828}
3829
3831{
3833
3834 if ( index >= 0 && index < mFields.count() )
3835 return mFields.at( index ).displayName();
3836 else
3837 return QString();
3838}
3839
3841{
3843
3844 return mAttributeAliasMap;
3845}
3846
3848{
3850
3851 if ( index < 0 || index >= fields().count() )
3852 return;
3853
3854 const QString name = fields().at( index ).name();
3855
3856 mAttributeSplitPolicy.insert( name, policy );
3857 mFields[index].setSplitPolicy( policy );
3858 mEditFormConfig.setFields( mFields );
3859 emit layerModified(); // TODO[MD]: should have a different signal?
3860}
3861
3863{
3865
3866 if ( index < 0 || index >= fields().count() )
3867 return;
3868
3869 const QString name = fields().at( index ).name();
3870
3871 mAttributeDuplicatePolicy.insert( name, policy );
3872 mFields[index].setDuplicatePolicy( policy );
3873 mEditFormConfig.setFields( mFields );
3874 emit layerModified(); // TODO[MD]: should have a different signal?
3875}
3876
3878{
3880
3881 if ( index < 0 || index >= fields().count() )
3882 return;
3883
3884 const QString name = fields().at( index ).name();
3885
3886 mAttributeMergePolicy.insert( name, policy );
3887 mFields[index].setMergePolicy( policy );
3888 mEditFormConfig.setFields( mFields );
3889 emit layerModified(); // TODO[MD]: should have a different signal?
3890}
3891
3893{
3895
3896 QSet<QString> excludeList;
3897 QMap< QString, Qgis::FieldConfigurationFlags >::const_iterator flagsIt = mFieldConfigurationFlags.constBegin();
3898 for ( ; flagsIt != mFieldConfigurationFlags.constEnd(); ++flagsIt )
3899 {
3900 if ( flagsIt->testFlag( Qgis::FieldConfigurationFlag::HideFromWms ) )
3901 {
3902 excludeList << flagsIt.key();
3903 }
3904 }
3905 return excludeList;
3906}
3907
3908void QgsVectorLayer::setExcludeAttributesWms( const QSet<QString> &att )
3909{
3911
3912 QMap< QString, Qgis::FieldConfigurationFlags >::iterator flagsIt = mFieldConfigurationFlags.begin();
3913 for ( ; flagsIt != mFieldConfigurationFlags.end(); ++flagsIt )
3914 {
3915 flagsIt->setFlag( Qgis::FieldConfigurationFlag::HideFromWms, att.contains( flagsIt.key() ) );
3916 }
3917 updateFields();
3918}
3919
3921{
3923
3924 QSet<QString> excludeList;
3925 QMap< QString, Qgis::FieldConfigurationFlags >::const_iterator flagsIt = mFieldConfigurationFlags.constBegin();
3926 for ( ; flagsIt != mFieldConfigurationFlags.constEnd(); ++flagsIt )
3927 {
3928 if ( flagsIt->testFlag( Qgis::FieldConfigurationFlag::HideFromWfs ) )
3929 {
3930 excludeList << flagsIt.key();
3931 }
3932 }
3933 return excludeList;
3934}
3935
3936void QgsVectorLayer::setExcludeAttributesWfs( const QSet<QString> &att )
3937{
3939
3940 QMap< QString, Qgis::FieldConfigurationFlags >::iterator flagsIt = mFieldConfigurationFlags.begin();
3941 for ( ; flagsIt != mFieldConfigurationFlags.end(); ++flagsIt )
3942 {
3943 flagsIt->setFlag( Qgis::FieldConfigurationFlag::HideFromWfs, att.contains( flagsIt.key() ) );
3944 }
3945 updateFields();
3946}
3947
3949{
3951
3952 if ( index < 0 || index >= fields().count() )
3953 return false;
3954
3955 if ( mFields.fieldOrigin( index ) == Qgis::FieldOrigin::Expression )
3956 {
3957 removeExpressionField( index );
3958 return true;
3959 }
3960
3961 if ( !mEditBuffer || !mDataProvider )
3962 return false;
3963
3964 return mEditBuffer->deleteAttribute( index );
3965}
3966
3967bool QgsVectorLayer::deleteAttributes( const QList<int> &attrs )
3968{
3970
3971 bool deleted = false;
3972
3973 // Remove multiple occurrences of same attribute
3974 QList<int> attrList = qgis::setToList( qgis::listToSet( attrs ) );
3975
3976 std::sort( attrList.begin(), attrList.end(), std::greater<int>() );
3977
3978 for ( int attr : std::as_const( attrList ) )
3979 {
3980 if ( deleteAttribute( attr ) )
3981 {
3982 deleted = true;
3983 }
3984 }
3985
3986 return deleted;
3987}
3988
3989bool QgsVectorLayer::deleteFeatureCascade( QgsFeatureId fid, QgsVectorLayer::DeleteContext *context )
3990{
3992
3993 if ( !mEditBuffer )
3994 return false;
3995
3996 if ( context && context->cascade )
3997 {
3998 const QList<QgsRelation> relations = context->project->relationManager()->referencedRelations( this );
3999 const bool hasRelationsOrJoins = !relations.empty() || mJoinBuffer->containsJoins();
4000 if ( hasRelationsOrJoins )
4001 {
4002 if ( context->mHandledFeatures.contains( this ) )
4003 {
4004 QgsFeatureIds &handledFeatureIds = context->mHandledFeatures[this];
4005 if ( handledFeatureIds.contains( fid ) )
4006 {
4007 // avoid endless recursion
4008 return false;
4009 }
4010 else
4011 {
4012 // add feature id
4013 handledFeatureIds << fid;
4014 }
4015 }
4016 else
4017 {
4018 // add layer and feature id
4019 context->mHandledFeatures.insert( this, QgsFeatureIds() << fid );
4020 }
4021
4022 for ( const QgsRelation &relation : relations )
4023 {
4024 //check if composition (and not association)
4025 switch ( relation.strength() )
4026 {
4028 {
4029 //get features connected over this relation
4030 QgsFeatureIterator relatedFeaturesIt = relation.getRelatedFeatures( getFeature( fid ) );
4031 QgsFeatureIds childFeatureIds;
4032 QgsFeature childFeature;
4033 while ( relatedFeaturesIt.nextFeature( childFeature ) )
4034 {
4035 childFeatureIds.insert( childFeature.id() );
4036 }
4037 if ( childFeatureIds.count() > 0 )
4038 {
4039 relation.referencingLayer()->startEditing();
4040 relation.referencingLayer()->deleteFeatures( childFeatureIds, context );
4041 }
4042 break;
4043 }
4044
4046 break;
4047 }
4048 }
4049 }
4050 }
4051
4052 if ( mJoinBuffer->containsJoins() )
4053 mJoinBuffer->deleteFeature( fid, context );
4054
4055 bool res = mEditBuffer->deleteFeature( fid );
4056
4057 return res;
4058}
4059
4061{
4063
4064 if ( !mEditBuffer )
4065 return false;
4066
4067 return deleteFeatureCascade( fid, context );
4068}
4069
4071{
4073
4074 bool res = true;
4075
4076 if ( ( context && context->cascade ) || mJoinBuffer->containsJoins() )
4077 {
4078 // should ideally be "deleteFeaturesCascade" for performance!
4079 for ( QgsFeatureId fid : fids )
4080 res = deleteFeatureCascade( fid, context ) && res;
4081 }
4082 else
4083 {
4084 res = mEditBuffer && mEditBuffer->deleteFeatures( fids );
4085 }
4086
4087 if ( res )
4088 {
4089 mSelectedFeatureIds.subtract( fids ); // remove it from selection
4090 updateExtents();
4091 }
4092
4093 return res;
4094}
4095
4097{
4098 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
4100
4101 return mFields;
4102}
4103
4105{
4107
4108 QgsAttributeList pkAttributesList;
4109 if ( !mDataProvider )
4110 return pkAttributesList;
4111
4112 QgsAttributeList providerIndexes = mDataProvider->pkAttributeIndexes();
4113 for ( int i = 0; i < mFields.count(); ++i )
4114 {
4115 if ( mFields.fieldOrigin( i ) == Qgis::FieldOrigin::Provider && providerIndexes.contains( mFields.fieldOriginIndex( i ) ) )
4116 pkAttributesList << i;
4117 }
4118
4119 return pkAttributesList;
4120}
4121
4123{
4125
4126 if ( !mDataProvider )
4127 return static_cast< long long >( Qgis::FeatureCountState::UnknownCount );
4128 return mDataProvider->featureCount() + ( mEditBuffer && !mDataProvider->transaction() ? mEditBuffer->addedFeatures().size() - mEditBuffer->deletedFeatureIds().size() : 0 );
4129}
4130
4132{
4134
4135 const QgsFeatureIds deletedFeatures( mEditBuffer && !mDataProvider->transaction() ? mEditBuffer->deletedFeatureIds() : QgsFeatureIds() );
4136 const QgsFeatureMap addedFeatures( mEditBuffer && !mDataProvider->transaction() ? mEditBuffer->addedFeatures() : QgsFeatureMap() );
4137
4138 if ( mEditBuffer && !deletedFeatures.empty() )
4139 {
4140 if ( addedFeatures.size() > deletedFeatures.size() )
4142 else
4144 }
4145
4146 if ( ( !mEditBuffer || addedFeatures.empty() ) && mDataProvider && mDataProvider->empty() )
4148 else
4150}
4151
4152bool QgsVectorLayer::commitChanges( bool stopEditing )
4153{
4155
4156 if ( project() && project()->transactionMode() == Qgis::TransactionMode::BufferedGroups )
4157 return project()->commitChanges( mCommitErrors, stopEditing, this );
4158
4159 mCommitErrors.clear();
4160
4161 if ( !mDataProvider )
4162 {
4163 mCommitErrors << tr( "ERROR: no provider" );
4164 return false;
4165 }
4166
4167 if ( !mEditBuffer )
4168 {
4169 mCommitErrors << tr( "ERROR: layer not editable" );
4170 return false;
4171 }
4172
4173 emit beforeCommitChanges( stopEditing );
4174
4175 if ( !mAllowCommit )
4176 return false;
4177
4178 mCommitChangesActive = true;
4179
4180 bool success = false;
4181 if ( mEditBuffer->editBufferGroup() )
4182 success = mEditBuffer->editBufferGroup()->commitChanges( mCommitErrors, stopEditing );
4183 else
4184 success = mEditBuffer->commitChanges( mCommitErrors );
4185
4186 mCommitChangesActive = false;
4187
4188 if ( !mDeletedFids.empty() )
4189 {
4190 emit featuresDeleted( mDeletedFids );
4191 mDeletedFids.clear();
4192 }
4193
4194 if ( success )
4195 {
4196 if ( stopEditing )
4197 {
4198 clearEditBuffer();
4199 }
4200 undoStack()->clear();
4201 emit afterCommitChanges();
4202 if ( stopEditing )
4203 emit editingStopped();
4204 }
4205 else
4206 {
4207 QgsMessageLog::logMessage( tr( "Commit errors:\n %1" ).arg( mCommitErrors.join( "\n "_L1 ) ) );
4208 }
4209
4210 updateFields();
4211
4212 mDataProvider->updateExtents();
4213
4214 if ( stopEditing )
4215 {
4216 mDataProvider->leaveUpdateMode();
4217 }
4218
4219 // This second call is required because OGR provider with JSON
4220 // driver might have changed fields order after the call to
4221 // leaveUpdateMode
4222 if ( mFields.names() != mDataProvider->fields().names() )
4223 {
4224 updateFields();
4225 }
4226
4228
4229 return success;
4230}
4231
4233{
4235
4236 return mCommitErrors;
4237}
4238
4239bool QgsVectorLayer::rollBack( bool deleteBuffer )
4240{
4242
4243 if ( project() && project()->transactionMode() == Qgis::TransactionMode::BufferedGroups )
4244 return project()->rollBack( mCommitErrors, deleteBuffer, this );
4245
4246 if ( !mEditBuffer )
4247 {
4248 return false;
4249 }
4250
4251 if ( !mDataProvider )
4252 {
4253 mCommitErrors << tr( "ERROR: no provider" );
4254 return false;
4255 }
4256
4257 bool rollbackExtent = !mDataProvider->transaction() && ( !mEditBuffer->deletedFeatureIds().isEmpty() || !mEditBuffer->addedFeatures().isEmpty() || !mEditBuffer->changedGeometries().isEmpty() );
4258
4259 emit beforeRollBack();
4260
4261 mEditBuffer->rollBack();
4262
4263 emit afterRollBack();
4264
4265 if ( isModified() )
4266 {
4267 // new undo stack roll back method
4268 // old method of calling every undo could cause many canvas refreshes
4269 undoStack()->setIndex( 0 );
4270 }
4271
4272 updateFields();
4273
4274 if ( deleteBuffer )
4275 {
4276 delete mEditBuffer;
4277 mEditBuffer = nullptr;
4278 undoStack()->clear();
4279 }
4280 emit editingStopped();
4281
4282 if ( rollbackExtent )
4283 updateExtents();
4284
4285 mDataProvider->leaveUpdateMode();
4286
4288 return true;
4289}
4290
4292{
4294
4295 return mSelectedFeatureIds.size();
4296}
4297
4299{
4300 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
4302
4303 return mSelectedFeatureIds;
4304}
4305
4307{
4309
4310 QgsFeatureList features;
4311 features.reserve( mSelectedFeatureIds.count() );
4312 QgsFeature f;
4313
4315
4316 while ( it.nextFeature( f ) )
4317 {
4318 features.push_back( f );
4319 }
4320
4321 return features;
4322}
4323
4325{
4327
4328 if ( mSelectedFeatureIds.isEmpty() )
4329 return QgsFeatureIterator();
4330
4333
4334 if ( mSelectedFeatureIds.count() == 1 )
4335 request.setFilterFid( *mSelectedFeatureIds.constBegin() );
4336 else
4337 request.setFilterFids( mSelectedFeatureIds );
4338
4339 return getFeatures( request );
4340}
4341
4343{
4345
4346 if ( !mEditBuffer || !mDataProvider )
4347 return false;
4348
4349 if ( mGeometryOptions->isActive() )
4350 {
4351 for ( auto feature = features.begin(); feature != features.end(); ++feature )
4352 {
4353 QgsGeometry geom = feature->geometry();
4354 mGeometryOptions->apply( geom );
4355 feature->setGeometry( geom );
4356 }
4357 }
4358
4359 bool res = mEditBuffer->addFeatures( features );
4360 updateExtents();
4361
4362 if ( res && mJoinBuffer->containsJoins() )
4363 res = mJoinBuffer->addFeatures( features );
4364
4365 return res;
4366}
4367
4369{
4371
4372 // if layer is not spatial, it has not CRS!
4373 setCrs( ( isSpatial() && mDataProvider ) ? mDataProvider->crs() : QgsCoordinateReferenceSystem() );
4374}
4375
4377{
4379
4381 if ( exp.isField() )
4382 {
4383 return static_cast<const QgsExpressionNodeColumnRef *>( exp.rootNode() )->name();
4384 }
4385
4386 return QString();
4387}
4388
4390{
4392
4393 if ( mDisplayExpression == displayExpression )
4394 return;
4395
4396 mDisplayExpression = displayExpression;
4398}
4399
4401{
4403
4404 if ( !mDisplayExpression.isEmpty() || mFields.isEmpty() )
4405 {
4406 return mDisplayExpression;
4407 }
4408 else
4409 {
4410 const QString candidateName = QgsVectorLayerUtils::guessFriendlyIdentifierField( mFields );
4411 if ( !candidateName.isEmpty() )
4412 {
4413 return QgsExpression::quotedColumnRef( candidateName );
4414 }
4415 else
4416 {
4417 return QString();
4418 }
4419 }
4420}
4421
4423{
4425
4426 // display expressions are used as a fallback when no explicit map tip template is set
4427 return mapTipsEnabled() && ( !mapTipTemplate().isEmpty() || !displayExpression().isEmpty() );
4428}
4429
4431{
4433
4434 return ( mEditBuffer && mDataProvider );
4435}
4436
4438{
4439 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
4441
4444}
4445
4446bool QgsVectorLayer::isReadOnly() const
4447{
4449
4450 return mDataSourceReadOnly || mReadOnly;
4451}
4452
4453bool QgsVectorLayer::setReadOnly( bool readonly )
4454{
4456
4457 // exit if the layer is in editing mode
4458 if ( readonly && mEditBuffer )
4459 return false;
4460
4461 // exit if the data source is in read-only mode
4462 if ( !readonly && mDataSourceReadOnly )
4463 return false;
4464
4465 mReadOnly = readonly;
4466 emit readOnlyChanged();
4467 return true;
4468}
4469
4471{
4473
4474 if ( !mDataProvider )
4475 return false;
4476
4477 if ( mDataSourceReadOnly )
4478 return false;
4479
4480 return mDataProvider->capabilities() & QgsVectorDataProvider::EditingCapabilities && !mReadOnly;
4481}
4482
4484{
4486
4487 emit beforeModifiedCheck();
4488 return mEditBuffer && mEditBuffer->isModified();
4489}
4490
4491bool QgsVectorLayer::isAuxiliaryField( int index, int &srcIndex ) const
4492{
4494
4495 bool auxiliaryField = false;
4496 srcIndex = -1;
4497
4498 if ( !auxiliaryLayer() )
4499 return auxiliaryField;
4500
4501 if ( index >= 0 && fields().fieldOrigin( index ) == Qgis::FieldOrigin::Join )
4502 {
4503 const QgsVectorLayerJoinInfo *info = mJoinBuffer->joinForFieldIndex( index, fields(), srcIndex );
4504
4505 if ( info && info->joinLayerId() == auxiliaryLayer()->id() )
4506 auxiliaryField = true;
4507 }
4508
4509 return auxiliaryField;
4510}
4511
4513{
4515
4516 // we must allow setting a renderer if our geometry type is unknown
4517 // as this allows the renderer to be correctly set even for layers
4518 // with broken sources
4519 // (note that we allow REMOVING the renderer for non-spatial layers,
4520 // e.g. to permit removing the renderer when the layer changes from
4521 // a spatial layer to a non-spatial one)
4522 if ( r && !isSpatial() && mWkbType != Qgis::WkbType::Unknown )
4523 return;
4524
4525 if ( r != mRenderer )
4526 {
4527 delete mRenderer;
4528 mRenderer = r;
4529 mSymbolFeatureCounted = false;
4530 mSymbolFeatureCountMap.clear();
4531 mSymbolFeatureIdMap.clear();
4532
4533 if ( mRenderer )
4534 {
4535 const double refreshRate = QgsSymbolLayerUtils::rendererFrameRate( mRenderer );
4536 if ( refreshRate <= 0 )
4537 {
4538 mRefreshRendererTimer->stop();
4539 mRefreshRendererTimer->setInterval( 0 );
4540 }
4541 else
4542 {
4543 mRefreshRendererTimer->setInterval( 1000 / refreshRate );
4544 mRefreshRendererTimer->start();
4545 }
4546 }
4547
4548 emit rendererChanged();
4550 }
4551}
4552
4554{
4556
4557 if ( generator )
4558 {
4559 mRendererGenerators << generator;
4560 }
4561}
4562
4564{
4566
4567 for ( int i = mRendererGenerators.count() - 1; i >= 0; --i )
4568 {
4569 if ( mRendererGenerators.at( i )->id() == id )
4570 {
4571 delete mRendererGenerators.at( i );
4572 mRendererGenerators.removeAt( i );
4573 }
4574 }
4575}
4576
4577QList<const QgsFeatureRendererGenerator *> QgsVectorLayer::featureRendererGenerators() const
4578{
4579 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
4581
4582 QList< const QgsFeatureRendererGenerator * > res;
4583 for ( const QgsFeatureRendererGenerator *generator : mRendererGenerators )
4584 res << generator;
4585 return res;
4586}
4587
4588void QgsVectorLayer::beginEditCommand( const QString &text )
4589{
4591
4592 if ( !mDataProvider )
4593 {
4594 return;
4595 }
4596 if ( mDataProvider->transaction() )
4597 {
4598 QString ignoredError;
4599 mDataProvider->transaction()->createSavepoint( ignoredError );
4600 }
4601 undoStack()->beginMacro( text );
4602 mEditCommandActive = true;
4603 emit editCommandStarted( text );
4604}
4605
4607{
4609
4610 if ( !mDataProvider )
4611 {
4612 return;
4613 }
4614 undoStack()->endMacro();
4615 mEditCommandActive = false;
4616 if ( !mDeletedFids.isEmpty() )
4617 {
4618 if ( selectedFeatureCount() > 0 )
4619 {
4620 mSelectedFeatureIds.subtract( mDeletedFids );
4621 }
4622 emit featuresDeleted( mDeletedFids );
4623 mDeletedFids.clear();
4624 }
4625 emit editCommandEnded();
4626}
4627
4629{
4631
4632 if ( !mDataProvider )
4633 {
4634 return;
4635 }
4636 undoStack()->endMacro();
4637 undoStack()->undo();
4638
4639 // it's not directly possible to pop the last command off the stack (the destroyed one)
4640 // and delete, so we add a dummy obsolete command to force this to occur.
4641 // Pushing the new command deletes the destroyed one, and since the new
4642 // command is obsolete it's automatically deleted by the undo stack.
4643 auto command = std::make_unique< QUndoCommand >();
4644 command->setObsolete( true );
4645 undoStack()->push( command.release() );
4646
4647 mEditCommandActive = false;
4648 mDeletedFids.clear();
4649 emit editCommandDestroyed();
4650}
4651
4653{
4655
4656 return mJoinBuffer->addJoin( joinInfo );
4657}
4658
4659bool QgsVectorLayer::removeJoin( const QString &joinLayerId )
4660{
4662
4663 return mJoinBuffer->removeJoin( joinLayerId );
4664}
4665
4666const QList< QgsVectorLayerJoinInfo > QgsVectorLayer::vectorJoins() const
4667{
4669
4670 return mJoinBuffer->vectorJoins();
4671}
4672
4673int QgsVectorLayer::addExpressionField( const QString &exp, const QgsField &fld )
4674{
4676
4677 emit beforeAddingExpressionField( fld.name() );
4678 mExpressionFieldBuffer->addExpression( exp, fld );
4679 updateFields();
4680 int idx = mFields.indexFromName( fld.name() );
4681 emit attributeAdded( idx );
4682 return idx;
4683}
4684
4686{
4688
4689 emit beforeRemovingExpressionField( index );
4690 int oi = mFields.fieldOriginIndex( index );
4691 mExpressionFieldBuffer->removeExpression( oi );
4692 updateFields();
4693 emit attributeDeleted( index );
4694}
4695
4696QString QgsVectorLayer::expressionField( int index ) const
4697{
4699
4700 if ( mFields.fieldOrigin( index ) != Qgis::FieldOrigin::Expression )
4701 return QString();
4702
4703 int oi = mFields.fieldOriginIndex( index );
4704 if ( oi < 0 || oi >= mExpressionFieldBuffer->expressions().size() )
4705 return QString();
4706
4707 return mExpressionFieldBuffer->expressions().at( oi ).cachedExpression.expression();
4708}
4709
4710void QgsVectorLayer::updateExpressionField( int index, const QString &exp )
4711{
4713
4714 int oi = mFields.fieldOriginIndex( index );
4715 mExpressionFieldBuffer->updateExpression( oi, exp );
4716}
4717
4719{
4720 // non fatal for now -- the QgsVirtualLayerTask class is not thread safe and calls this
4722
4723 if ( !mDataProvider )
4724 return;
4725
4726 QgsFields oldFields = mFields;
4727
4728 mFields = mDataProvider->fields();
4729
4730 // added / removed fields
4731 if ( mEditBuffer )
4732 mEditBuffer->updateFields( mFields );
4733
4734 // joined fields
4735 if ( mJoinBuffer->containsJoins() )
4736 mJoinBuffer->updateFields( mFields );
4737
4738 if ( mExpressionFieldBuffer )
4739 mExpressionFieldBuffer->updateFields( mFields );
4740
4741 // set aliases and default values
4742 for ( auto aliasIt = mAttributeAliasMap.constBegin(); aliasIt != mAttributeAliasMap.constEnd(); ++aliasIt )
4743 {
4744 int index = mFields.lookupField( aliasIt.key() );
4745 if ( index < 0 )
4746 continue;
4747
4748 mFields[index].setAlias( aliasIt.value() );
4749 }
4750
4751 // set custom comments
4752 for ( auto customCommentIt = mAttributeCustomCommentMap.constBegin(); customCommentIt != mAttributeCustomCommentMap.constEnd(); ++customCommentIt )
4753 {
4754 int index = mFields.lookupField( customCommentIt.key() );
4755 if ( index < 0 )
4756 continue;
4757
4758 mFields[index].setCustomComment( customCommentIt.value() );
4759 }
4760
4761 for ( auto splitPolicyIt = mAttributeSplitPolicy.constBegin(); splitPolicyIt != mAttributeSplitPolicy.constEnd(); ++splitPolicyIt )
4762 {
4763 int index = mFields.lookupField( splitPolicyIt.key() );
4764 if ( index < 0 )
4765 continue;
4766
4767 mFields[index].setSplitPolicy( splitPolicyIt.value() );
4768 }
4769
4770 for ( auto duplicatePolicyIt = mAttributeDuplicatePolicy.constBegin(); duplicatePolicyIt != mAttributeDuplicatePolicy.constEnd(); ++duplicatePolicyIt )
4771 {
4772 int index = mFields.lookupField( duplicatePolicyIt.key() );
4773 if ( index < 0 )
4774 continue;
4775
4776 mFields[index].setDuplicatePolicy( duplicatePolicyIt.value() );
4777 }
4778
4779 for ( auto mergePolicyIt = mAttributeMergePolicy.constBegin(); mergePolicyIt != mAttributeMergePolicy.constEnd(); ++mergePolicyIt )
4780 {
4781 int index = mFields.lookupField( mergePolicyIt.key() );
4782 if ( index < 0 )
4783 continue;
4784
4785 mFields[index].setMergePolicy( mergePolicyIt.value() );
4786 }
4787
4788 // Update configuration flags
4789 QMap< QString, Qgis::FieldConfigurationFlags >::const_iterator flagsIt = mFieldConfigurationFlags.constBegin();
4790 for ( ; flagsIt != mFieldConfigurationFlags.constEnd(); ++flagsIt )
4791 {
4792 int index = mFields.lookupField( flagsIt.key() );
4793 if ( index < 0 )
4794 continue;
4795
4796 mFields[index].setConfigurationFlags( flagsIt.value() );
4797 }
4798
4799 // Update default values
4800 mDefaultValueOnUpdateFields.clear();
4801 QMap< QString, QgsDefaultValue >::const_iterator defaultIt = mDefaultExpressionMap.constBegin();
4802 for ( ; defaultIt != mDefaultExpressionMap.constEnd(); ++defaultIt )
4803 {
4804 int index = mFields.lookupField( defaultIt.key() );
4805 if ( index < 0 )
4806 continue;
4807
4808 mFields[index].setDefaultValueDefinition( defaultIt.value() );
4809 if ( defaultIt.value().applyOnUpdate() )
4810 mDefaultValueOnUpdateFields.insert( index );
4811 }
4812
4813 QMap< QString, QgsFieldConstraints::Constraints >::const_iterator constraintIt = mFieldConstraints.constBegin();
4814 for ( ; constraintIt != mFieldConstraints.constEnd(); ++constraintIt )
4815 {
4816 int index = mFields.lookupField( constraintIt.key() );
4817 if ( index < 0 )
4818 continue;
4819
4820 QgsFieldConstraints constraints = mFields.at( index ).constraints();
4821
4822 // always keep provider constraints intact
4823 if ( !( constraints.constraints() & QgsFieldConstraints::ConstraintNotNull ) && ( constraintIt.value() & QgsFieldConstraints::ConstraintNotNull ) )
4825 if ( !( constraints.constraints() & QgsFieldConstraints::ConstraintUnique ) && ( constraintIt.value() & QgsFieldConstraints::ConstraintUnique ) )
4827 if ( !( constraints.constraints() & QgsFieldConstraints::ConstraintExpression ) && ( constraintIt.value() & QgsFieldConstraints::ConstraintExpression ) )
4829 mFields[index].setConstraints( constraints );
4830 }
4831
4832 QMap< QString, QPair< QString, QString > >::const_iterator constraintExpIt = mFieldConstraintExpressions.constBegin();
4833 for ( ; constraintExpIt != mFieldConstraintExpressions.constEnd(); ++constraintExpIt )
4834 {
4835 int index = mFields.lookupField( constraintExpIt.key() );
4836 if ( index < 0 )
4837 continue;
4838
4839 QgsFieldConstraints constraints = mFields.at( index ).constraints();
4840
4841 // always keep provider constraints intact
4843 continue;
4844
4845 constraints.setConstraintExpression( constraintExpIt.value().first, constraintExpIt.value().second );
4846 mFields[index].setConstraints( constraints );
4847 }
4848
4849 QMap< QPair< QString, QgsFieldConstraints::Constraint >, QgsFieldConstraints::ConstraintStrength >::const_iterator constraintStrengthIt = mFieldConstraintStrength.constBegin();
4850 for ( ; constraintStrengthIt != mFieldConstraintStrength.constEnd(); ++constraintStrengthIt )
4851 {
4852 int index = mFields.lookupField( constraintStrengthIt.key().first );
4853 if ( index < 0 )
4854 continue;
4855
4856 QgsFieldConstraints constraints = mFields.at( index ).constraints();
4857
4858 // always keep provider constraints intact
4860 continue;
4861
4862 constraints.setConstraintStrength( constraintStrengthIt.key().second, constraintStrengthIt.value() );
4863 mFields[index].setConstraints( constraints );
4864 }
4865
4866 auto fieldWidgetIterator = mFieldWidgetSetups.constBegin();
4867 for ( ; fieldWidgetIterator != mFieldWidgetSetups.constEnd(); ++fieldWidgetIterator )
4868 {
4869 int index = mFields.indexOf( fieldWidgetIterator.key() );
4870 if ( index < 0 )
4871 continue;
4872
4873 mFields[index].setEditorWidgetSetup( fieldWidgetIterator.value() );
4874 }
4875
4876 if ( oldFields != mFields )
4877 {
4878 emit updatedFields();
4879 mEditFormConfig.setFields( mFields );
4880 }
4881}
4882
4883QVariant QgsVectorLayer::defaultValue( int index, const QgsFeature &feature, QgsExpressionContext *context ) const
4884{
4886
4887 if ( index < 0 || index >= mFields.count() || !mDataProvider )
4888 return QVariant();
4889
4890 QString expression = mFields.at( index ).defaultValueDefinition().expression();
4891 if ( expression.isEmpty() )
4892 return mDataProvider->defaultValue( index );
4893
4894 QgsExpressionContext *evalContext = context;
4895 std::unique_ptr< QgsExpressionContext > tempContext;
4896 if ( !evalContext )
4897 {
4898 // no context passed, so we create a default one
4899 tempContext = std::make_unique<QgsExpressionContext>( QgsExpressionContextUtils::globalProjectLayerScopes( this ) );
4900 evalContext = tempContext.get();
4901 }
4902
4903 if ( feature.isValid() )
4904 {
4906 featScope->setFeature( feature );
4907 featScope->setFields( feature.fields() );
4908 evalContext->appendScope( featScope );
4909 }
4910
4911 QVariant val;
4912 QgsExpression exp( expression );
4913 exp.prepare( evalContext );
4914 if ( exp.hasEvalError() )
4915 {
4916 QgsLogger::warning( "Error evaluating default value: " + exp.evalErrorString() );
4917 }
4918 else
4919 {
4920 val = exp.evaluate( evalContext );
4921 }
4922
4923 if ( feature.isValid() )
4924 {
4925 delete evalContext->popScope();
4926 }
4927
4928 return val;
4929}
4930
4932{
4934
4935 if ( index < 0 || index >= mFields.count() )
4936 return;
4937
4938 if ( definition.isValid() )
4939 {
4940 mDefaultExpressionMap.insert( mFields.at( index ).name(), definition );
4941 }
4942 else
4943 {
4944 mDefaultExpressionMap.remove( mFields.at( index ).name() );
4945 }
4946 updateFields();
4947}
4948
4950{
4952
4953 if ( index < 0 || index >= mFields.count() )
4954 return QgsDefaultValue();
4955 else
4956 return mFields.at( index ).defaultValueDefinition();
4957}
4958
4959QSet<QVariant> QgsVectorLayer::uniqueValues( int index, int limit ) const
4960{
4962
4963 QSet<QVariant> uniqueValues;
4964 if ( !mDataProvider )
4965 {
4966 return uniqueValues;
4967 }
4968
4969 Qgis::FieldOrigin origin = mFields.fieldOrigin( index );
4970 switch ( origin )
4971 {
4973 return uniqueValues;
4974
4975 case Qgis::FieldOrigin::Provider: //a provider field
4976 {
4977 uniqueValues = mDataProvider->uniqueValues( index, limit );
4978
4979 if ( mEditBuffer && !mDataProvider->transaction() )
4980 {
4981 QSet<QString> vals;
4982 const auto constUniqueValues = uniqueValues;
4983 for ( const QVariant &v : constUniqueValues )
4984 {
4985 vals << v.toString();
4986 }
4987
4988 QgsFeatureMap added = mEditBuffer->addedFeatures();
4989 QMapIterator< QgsFeatureId, QgsFeature > addedIt( added );
4990 while ( addedIt.hasNext() && ( limit < 0 || uniqueValues.count() < limit ) )
4991 {
4992 addedIt.next();
4993 QVariant v = addedIt.value().attribute( index );
4994 if ( v.isValid() )
4995 {
4996 QString vs = v.toString();
4997 if ( !vals.contains( vs ) )
4998 {
4999 vals << vs;
5000 uniqueValues << v;
5001 }
5002 }
5003 }
5004
5005 QMapIterator< QgsFeatureId, QgsAttributeMap > it( mEditBuffer->changedAttributeValues() );
5006 while ( it.hasNext() && ( limit < 0 || uniqueValues.count() < limit ) )
5007 {
5008 it.next();
5009 QVariant v = it.value().value( index );
5010 if ( v.isValid() )
5011 {
5012 QString vs = v.toString();
5013 if ( !vals.contains( vs ) )
5014 {
5015 vals << vs;
5016 uniqueValues << v;
5017 }
5018 }
5019 }
5020 }
5021
5022 return uniqueValues;
5023 }
5024
5026 // the layer is editable, but in certain cases it can still be avoided going through all features
5027 if ( mDataProvider->transaction()
5028 || ( mEditBuffer->deletedFeatureIds().isEmpty() && mEditBuffer->addedFeatures().isEmpty() && !mEditBuffer->deletedAttributeIds().contains( index ) && mEditBuffer->changedAttributeValues().isEmpty() ) )
5029 {
5030 uniqueValues = mDataProvider->uniqueValues( index, limit );
5031 return uniqueValues;
5032 }
5033 [[fallthrough]];
5034 //we need to go through each feature
5037 {
5038 QgsAttributeList attList;
5039 attList << index;
5040
5042
5043 QgsFeature f;
5044 QVariant currentValue;
5045 QHash<QString, QVariant> val;
5046 while ( fit.nextFeature( f ) )
5047 {
5048 currentValue = f.attribute( index );
5049 val.insert( currentValue.toString(), currentValue );
5050 if ( limit >= 0 && val.size() >= limit )
5051 {
5052 break;
5053 }
5054 }
5055
5056 return qgis::listToSet( val.values() );
5057 }
5058 }
5059
5060 Q_ASSERT_X( false, "QgsVectorLayer::uniqueValues()", "Unknown source of the field!" );
5061 return uniqueValues;
5062}
5063
5064QStringList QgsVectorLayer::uniqueStringsMatching( int index, const QString &substring, int limit, QgsFeedback *feedback ) const
5065{
5067
5068 QStringList results;
5069 if ( !mDataProvider )
5070 {
5071 return results;
5072 }
5073
5074 Qgis::FieldOrigin origin = mFields.fieldOrigin( index );
5075 switch ( origin )
5076 {
5078 return results;
5079
5080 case Qgis::FieldOrigin::Provider: //a provider field
5081 {
5082 results = mDataProvider->uniqueStringsMatching( index, substring, limit, feedback );
5083
5084 if ( mEditBuffer && !mDataProvider->transaction() )
5085 {
5086 QgsFeatureMap added = mEditBuffer->addedFeatures();
5087 QMapIterator< QgsFeatureId, QgsFeature > addedIt( added );
5088 while ( addedIt.hasNext() && ( limit < 0 || results.count() < limit ) && ( !feedback || !feedback->isCanceled() ) )
5089 {
5090 addedIt.next();
5091 QVariant v = addedIt.value().attribute( index );
5092 if ( v.isValid() )
5093 {
5094 QString vs = v.toString();
5095 if ( vs.contains( substring, Qt::CaseInsensitive ) && !results.contains( vs ) )
5096 {
5097 results << vs;
5098 }
5099 }
5100 }
5101
5102 QMapIterator< QgsFeatureId, QgsAttributeMap > it( mEditBuffer->changedAttributeValues() );
5103 while ( it.hasNext() && ( limit < 0 || results.count() < limit ) && ( !feedback || !feedback->isCanceled() ) )
5104 {
5105 it.next();
5106 QVariant v = it.value().value( index );
5107 if ( v.isValid() )
5108 {
5109 QString vs = v.toString();
5110 if ( vs.contains( substring, Qt::CaseInsensitive ) && !results.contains( vs ) )
5111 {
5112 results << vs;
5113 }
5114 }
5115 }
5116 }
5117
5118 return results;
5119 }
5120
5122 // the layer is editable, but in certain cases it can still be avoided going through all features
5123 if ( mDataProvider->transaction()
5124 || ( mEditBuffer->deletedFeatureIds().isEmpty() && mEditBuffer->addedFeatures().isEmpty() && !mEditBuffer->deletedAttributeIds().contains( index ) && mEditBuffer->changedAttributeValues().isEmpty() ) )
5125 {
5126 return mDataProvider->uniqueStringsMatching( index, substring, limit, feedback );
5127 }
5128 [[fallthrough]];
5129 //we need to go through each feature
5132 {
5133 QgsAttributeList attList;
5134 attList << index;
5135
5136 QgsFeatureRequest request;
5137 request.setSubsetOfAttributes( attList );
5139 QString fieldName = mFields.at( index ).name();
5140 request.setFilterExpression( u"\"%1\" ILIKE '%%2%'"_s.arg( fieldName, substring ) );
5141 QgsFeatureIterator fit = getFeatures( request );
5142
5143 QgsFeature f;
5144 QString currentValue;
5145 while ( fit.nextFeature( f ) )
5146 {
5147 currentValue = f.attribute( index ).toString();
5148 if ( !results.contains( currentValue ) )
5149 results << currentValue;
5150
5151 if ( ( limit >= 0 && results.size() >= limit ) || ( feedback && feedback->isCanceled() ) )
5152 {
5153 break;
5154 }
5155 }
5156
5157 return results;
5158 }
5159 }
5160
5161 Q_ASSERT_X( false, "QgsVectorLayer::uniqueStringsMatching()", "Unknown source of the field!" );
5162 return results;
5163}
5164
5165QVariant QgsVectorLayer::minimumValue( int index ) const
5166{
5168
5169 QVariant minimum;
5170 minimumOrMaximumValue( index, &minimum, nullptr );
5171 return minimum;
5172}
5173
5174QVariant QgsVectorLayer::maximumValue( int index ) const
5175{
5177
5178 QVariant maximum;
5179 minimumOrMaximumValue( index, nullptr, &maximum );
5180 return maximum;
5181}
5182
5183void QgsVectorLayer::minimumAndMaximumValue( int index, QVariant &minimum, QVariant &maximum ) const
5184{
5186
5187 minimumOrMaximumValue( index, &minimum, &maximum );
5188}
5189
5190void QgsVectorLayer::minimumOrMaximumValue( int index, QVariant *minimum, QVariant *maximum ) const
5191{
5193
5194 if ( minimum )
5195 *minimum = QVariant();
5196 if ( maximum )
5197 *maximum = QVariant();
5198
5199 if ( !mDataProvider )
5200 {
5201 return;
5202 }
5203
5204 Qgis::FieldOrigin origin = mFields.fieldOrigin( index );
5205
5206 switch ( origin )
5207 {
5209 {
5210 return;
5211 }
5212
5213 case Qgis::FieldOrigin::Provider: //a provider field
5214 {
5215 if ( minimum )
5216 *minimum = mDataProvider->minimumValue( index );
5217 if ( maximum )
5218 *maximum = mDataProvider->maximumValue( index );
5219 if ( mEditBuffer && !mDataProvider->transaction() )
5220 {
5221 const QgsFeatureMap added = mEditBuffer->addedFeatures();
5222 QMapIterator< QgsFeatureId, QgsFeature > addedIt( added );
5223 while ( addedIt.hasNext() )
5224 {
5225 addedIt.next();
5226 const QVariant v = addedIt.value().attribute( index );
5227 if ( minimum && v.isValid() && qgsVariantLessThan( v, *minimum ) )
5228 *minimum = v;
5229 if ( maximum && v.isValid() && qgsVariantGreaterThan( v, *maximum ) )
5230 *maximum = v;
5231 }
5232
5233 QMapIterator< QgsFeatureId, QgsAttributeMap > it( mEditBuffer->changedAttributeValues() );
5234 while ( it.hasNext() )
5235 {
5236 it.next();
5237 const QVariant v = it.value().value( index );
5238 if ( minimum && v.isValid() && qgsVariantLessThan( v, *minimum ) )
5239 *minimum = v;
5240 if ( maximum && v.isValid() && qgsVariantGreaterThan( v, *maximum ) )
5241 *maximum = v;
5242 }
5243 }
5244 return;
5245 }
5246
5248 {
5249 // the layer is editable, but in certain cases it can still be avoided going through all features
5250 if ( mDataProvider->transaction()
5251 || ( mEditBuffer->deletedFeatureIds().isEmpty() && mEditBuffer->addedFeatures().isEmpty() && !mEditBuffer->deletedAttributeIds().contains( index ) && mEditBuffer->changedAttributeValues().isEmpty() ) )
5252 {
5253 if ( minimum )
5254 *minimum = mDataProvider->minimumValue( index );
5255 if ( maximum )
5256 *maximum = mDataProvider->maximumValue( index );
5257 return;
5258 }
5259 }
5260 [[fallthrough]];
5261 // no choice but to go through all features
5264 {
5265 // we need to go through each feature
5266 QgsAttributeList attList;
5267 attList << index;
5268
5269 QgsFeatureIterator fit = getFeatures( QgsFeatureRequest().setFlags( Qgis::FeatureRequestFlag::NoGeometry ).setSubsetOfAttributes( attList ) );
5270
5271 QgsFeature f;
5272 bool firstValue = true;
5273 while ( fit.nextFeature( f ) )
5274 {
5275 const QVariant currentValue = f.attribute( index );
5276 if ( QgsVariantUtils::isNull( currentValue ) )
5277 continue;
5278
5279 if ( firstValue )
5280 {
5281 if ( minimum )
5282 *minimum = currentValue;
5283 if ( maximum )
5284 *maximum = currentValue;
5285 firstValue = false;
5286 }
5287 else
5288 {
5289 if ( minimum && currentValue.isValid() && qgsVariantLessThan( currentValue, *minimum ) )
5290 *minimum = currentValue;
5291 if ( maximum && currentValue.isValid() && qgsVariantGreaterThan( currentValue, *maximum ) )
5292 *maximum = currentValue;
5293 }
5294 }
5295 return;
5296 }
5297 }
5298
5299 Q_ASSERT_X( false, "QgsVectorLayer::minimumOrMaximumValue()", "Unknown source of the field!" );
5300}
5301
5302void QgsVectorLayer::createEditBuffer()
5303{
5305
5306 if ( mEditBuffer )
5307 clearEditBuffer();
5308
5309 if ( mDataProvider->transaction() )
5310 {
5311 mEditBuffer = new QgsVectorLayerEditPassthrough( this );
5312
5313 connect( mDataProvider->transaction(), &QgsTransaction::dirtied, this, &QgsVectorLayer::onDirtyTransaction, Qt::UniqueConnection );
5314 }
5315 else
5316 {
5317 mEditBuffer = new QgsVectorLayerEditBuffer( this );
5318 }
5319 // forward signals
5320 connect( mEditBuffer, &QgsVectorLayerEditBuffer::layerModified, this, &QgsVectorLayer::invalidateSymbolCountedFlag );
5321 connect( mEditBuffer, &QgsVectorLayerEditBuffer::layerModified, this, &QgsVectorLayer::layerModified ); // TODO[MD]: necessary?
5322 //connect( mEditBuffer, &QgsVectorLayerEditBuffer::layerModified, this, &QgsVectorLayer::triggerRepaint ); // TODO[MD]: works well?
5323 connect( mEditBuffer, &QgsVectorLayerEditBuffer::featureAdded, this, &QgsVectorLayer::onFeatureAdded );
5324 connect( mEditBuffer, &QgsVectorLayerEditBuffer::featureDeleted, this, &QgsVectorLayer::onFeatureDeleted );
5335}
5336
5337void QgsVectorLayer::clearEditBuffer()
5338{
5340
5341 delete mEditBuffer;
5342 mEditBuffer = nullptr;
5343}
5344
5347 const QString &fieldOrExpression,
5349 QgsExpressionContext *context,
5350 bool *ok,
5351 QgsFeatureIds *fids,
5352 QgsFeedback *feedback,
5353 QString *error
5354) const
5355{
5356 // non fatal for now -- the aggregate expression functions are not thread safe and call this
5358
5359 if ( ok )
5360 *ok = false;
5361 if ( error )
5362 error->clear();
5363
5364 if ( !mDataProvider )
5365 {
5366 if ( error )
5367 *error = tr( "Layer is invalid" );
5368 return QVariant();
5369 }
5370
5371 // test if we are calculating based on a field
5372 const int attrIndex = QgsExpression::expressionToLayerFieldIndex( fieldOrExpression, this );
5373 if ( attrIndex >= 0 )
5374 {
5375 // aggregate is based on a field - if it's a provider field, we could possibly hand over the calculation
5376 // to the provider itself
5377 Qgis::FieldOrigin origin = mFields.fieldOrigin( attrIndex );
5378 if ( origin == Qgis::FieldOrigin::Provider )
5379 {
5380 bool providerOk = false;
5381 QVariant val = mDataProvider->aggregate( aggregate, attrIndex, parameters, context, providerOk, fids );
5382 if ( providerOk )
5383 {
5384 // provider handled calculation
5385 if ( ok )
5386 *ok = true;
5387 return val;
5388 }
5389 }
5390 }
5391
5392 // fallback to using aggregate calculator to determine aggregate
5393 QgsAggregateCalculator c( this );
5394 if ( fids )
5395 c.setFidsFilter( *fids );
5396 c.setParameters( parameters );
5397 bool aggregateOk = false;
5398 const QVariant result = c.calculate( aggregate, fieldOrExpression, context, &aggregateOk, feedback );
5399 if ( ok )
5400 *ok = aggregateOk;
5401 if ( !aggregateOk && error )
5402 *error = c.lastError();
5403
5404 return result;
5405}
5406
5408{
5410
5411 if ( mFeatureBlendMode == featureBlendMode )
5412 return;
5413
5414 mFeatureBlendMode = featureBlendMode;
5417}
5418
5419QPainter::CompositionMode QgsVectorLayer::featureBlendMode() const
5420{
5421 // non fatal for now -- the "rasterize" processing algorithm is not thread safe and calls this
5423
5424 return mFeatureBlendMode;
5425}
5426
5427void QgsVectorLayer::readSldLabeling( const QDomNode &node )
5428{
5430
5431 setLabeling( nullptr ); // start with no labeling
5432 setLabelsEnabled( false );
5433
5434 QDomElement element = node.toElement();
5435 if ( element.isNull() )
5436 return;
5437
5438 QDomElement userStyleElem = element.firstChildElement( u"UserStyle"_s );
5439 if ( userStyleElem.isNull() )
5440 {
5441 QgsDebugMsgLevel( u"Info: UserStyle element not found."_s, 4 );
5442 return;
5443 }
5444
5445 QDomElement featTypeStyleElem = userStyleElem.firstChildElement( u"FeatureTypeStyle"_s );
5446 if ( featTypeStyleElem.isNull() )
5447 {
5448 QgsDebugMsgLevel( u"Info: FeatureTypeStyle element not found."_s, 4 );
5449 return;
5450 }
5451
5452 // create empty FeatureTypeStyle element to merge TextSymbolizer's Rule's from all FeatureTypeStyle's
5453 QDomElement mergedFeatTypeStyle = featTypeStyleElem.cloneNode( false ).toElement();
5454
5455 // use the RuleRenderer when more rules are present or the rule
5456 // has filters or min/max scale denominators set,
5457 // otherwise use the Simple labeling
5458 bool needRuleBasedLabeling = false;
5459 int ruleCount = 0;
5460
5461 while ( !featTypeStyleElem.isNull() )
5462 {
5463 QDomElement ruleElem = featTypeStyleElem.firstChildElement( u"Rule"_s );
5464 while ( !ruleElem.isNull() )
5465 {
5466 // test rule children element to check if we need to create RuleRenderer
5467 // and if the rule has a symbolizer
5468 bool hasTextSymbolizer = false;
5469 bool hasRuleBased = false;
5470 QDomElement ruleChildElem = ruleElem.firstChildElement();
5471 while ( !ruleChildElem.isNull() )
5472 {
5473 // rule has filter or min/max scale denominator, use the RuleRenderer
5474 if ( ruleChildElem.localName() == "Filter"_L1 || ruleChildElem.localName() == "MinScaleDenominator"_L1 || ruleChildElem.localName() == "MaxScaleDenominator"_L1 )
5475 {
5476 hasRuleBased = true;
5477 }
5478 // rule has a renderer symbolizer, not a text symbolizer
5479 else if ( ruleChildElem.localName() == "TextSymbolizer"_L1 )
5480 {
5481 QgsDebugMsgLevel( u"Info: TextSymbolizer element found"_s, 4 );
5482 hasTextSymbolizer = true;
5483 }
5484
5485 ruleChildElem = ruleChildElem.nextSiblingElement();
5486 }
5487
5488 if ( hasTextSymbolizer )
5489 {
5490 ruleCount++;
5491
5492 // append a clone of all Rules to the merged FeatureTypeStyle element
5493 mergedFeatTypeStyle.appendChild( ruleElem.cloneNode().toElement() );
5494
5495 if ( hasRuleBased )
5496 {
5497 QgsDebugMsgLevel( u"Info: Filter or Min/MaxScaleDenominator element found: need a RuleBasedLabeling"_s, 4 );
5498 needRuleBasedLabeling = true;
5499 }
5500 }
5501
5502 // more rules present, use the RuleRenderer
5503 if ( ruleCount > 1 )
5504 {
5505 QgsDebugMsgLevel( u"Info: More Rule elements found: need a RuleBasedLabeling"_s, 4 );
5506 needRuleBasedLabeling = true;
5507 }
5508
5509 // not use the rule based labeling if no rules with textSymbolizer
5510 if ( ruleCount == 0 )
5511 {
5512 needRuleBasedLabeling = false;
5513 }
5514
5515 ruleElem = ruleElem.nextSiblingElement( u"Rule"_s );
5516 }
5517 featTypeStyleElem = featTypeStyleElem.nextSiblingElement( u"FeatureTypeStyle"_s );
5518 }
5519
5520 if ( ruleCount == 0 )
5521 {
5522 QgsDebugMsgLevel( u"Info: No TextSymbolizer element."_s, 4 );
5523 return;
5524 }
5525
5526 QDomElement ruleElem = mergedFeatTypeStyle.firstChildElement( u"Rule"_s );
5527
5528 if ( needRuleBasedLabeling )
5529 {
5530 QgsDebugMsgLevel( u"Info: rule based labeling"_s, 4 );
5531 QgsRuleBasedLabeling::Rule *rootRule = new QgsRuleBasedLabeling::Rule( nullptr );
5532 while ( !ruleElem.isNull() )
5533 {
5534 QString label, description, filterExp;
5535 int scaleMinDenom = 0, scaleMaxDenom = 0;
5536 QgsPalLayerSettings settings;
5537
5538 // retrieve the Rule element child nodes
5539 QDomElement childElem = ruleElem.firstChildElement();
5540 while ( !childElem.isNull() )
5541 {
5542 if ( childElem.localName() == "Name"_L1 )
5543 {
5544 // <se:Name> tag contains the rule identifier,
5545 // so prefer title tag for the label property value
5546 if ( label.isEmpty() )
5547 label = childElem.firstChild().nodeValue();
5548 }
5549 else if ( childElem.localName() == "Description"_L1 )
5550 {
5551 // <se:Description> can contains a title and an abstract
5552 QDomElement titleElem = childElem.firstChildElement( u"Title"_s );
5553 if ( !titleElem.isNull() )
5554 {
5555 label = titleElem.firstChild().nodeValue();
5556 }
5557
5558 QDomElement abstractElem = childElem.firstChildElement( u"Abstract"_s );
5559 if ( !abstractElem.isNull() )
5560 {
5561 description = abstractElem.firstChild().nodeValue();
5562 }
5563 }
5564 else if ( childElem.localName() == "Abstract"_L1 )
5565 {
5566 // <sld:Abstract> (v1.0)
5567 description = childElem.firstChild().nodeValue();
5568 }
5569 else if ( childElem.localName() == "Title"_L1 )
5570 {
5571 // <sld:Title> (v1.0)
5572 label = childElem.firstChild().nodeValue();
5573 }
5574 else if ( childElem.localName() == "Filter"_L1 )
5575 {
5576 QgsExpression *filter = QgsOgcUtils::expressionFromOgcFilter( childElem );
5577 if ( filter )
5578 {
5579 if ( filter->hasParserError() )
5580 {
5581 QgsDebugMsgLevel( u"SLD Filter parsing error: %1"_s.arg( filter->parserErrorString() ), 3 );
5582 }
5583 else
5584 {
5585 filterExp = filter->expression();
5586 }
5587 delete filter;
5588 }
5589 }
5590 else if ( childElem.localName() == "MinScaleDenominator"_L1 )
5591 {
5592 bool ok;
5593 int v = childElem.firstChild().nodeValue().toInt( &ok );
5594 if ( ok )
5595 scaleMinDenom = v;
5596 }
5597 else if ( childElem.localName() == "MaxScaleDenominator"_L1 )
5598 {
5599 bool ok;
5600 int v = childElem.firstChild().nodeValue().toInt( &ok );
5601 if ( ok )
5602 scaleMaxDenom = v;
5603 }
5604 else if ( childElem.localName() == "TextSymbolizer"_L1 )
5605 {
5606 readSldTextSymbolizer( childElem, settings );
5607 }
5608
5609 childElem = childElem.nextSiblingElement();
5610 }
5611
5612 QgsRuleBasedLabeling::Rule *ruleLabeling = new QgsRuleBasedLabeling::Rule( new QgsPalLayerSettings( settings ), scaleMinDenom, scaleMaxDenom, filterExp, label );
5613 rootRule->appendChild( ruleLabeling );
5614
5615 ruleElem = ruleElem.nextSiblingElement();
5616 }
5617
5618 setLabeling( new QgsRuleBasedLabeling( rootRule ) );
5619 setLabelsEnabled( true );
5620 }
5621 else
5622 {
5623 QgsDebugMsgLevel( u"Info: simple labeling"_s, 4 );
5624 // retrieve the TextSymbolizer element child node
5625 QDomElement textSymbolizerElem = ruleElem.firstChildElement( u"TextSymbolizer"_s );
5626 QgsPalLayerSettings s;
5627 if ( readSldTextSymbolizer( textSymbolizerElem, s ) )
5628 {
5629 setLabeling( new QgsVectorLayerSimpleLabeling( s ) );
5630 setLabelsEnabled( true );
5631 }
5632 }
5633}
5634
5635bool QgsVectorLayer::readSldTextSymbolizer( const QDomNode &node, QgsPalLayerSettings &settings ) const
5636{
5638
5639 if ( node.localName() != "TextSymbolizer"_L1 )
5640 {
5641 QgsDebugMsgLevel( u"Not a TextSymbolizer element: %1"_s.arg( node.localName() ), 3 );
5642 return false;
5643 }
5644 QDomElement textSymbolizerElem = node.toElement();
5645 // Label
5646 QDomElement labelElem = textSymbolizerElem.firstChildElement( u"Label"_s );
5647 if ( !labelElem.isNull() )
5648 {
5649 QDomElement propertyNameElem = labelElem.firstChildElement( u"PropertyName"_s );
5650 if ( !propertyNameElem.isNull() )
5651 {
5652 // set labeling defaults
5653
5654 // label attribute
5655 QString labelAttribute = propertyNameElem.text();
5656 settings.fieldName = labelAttribute;
5657 settings.isExpression = false;
5658
5659 int fieldIndex = mFields.lookupField( labelAttribute );
5660 if ( fieldIndex == -1 )
5661 {
5662 // label attribute is not in columns, check if it is an expression
5663 QgsExpression exp( labelAttribute );
5664 if ( !exp.hasEvalError() )
5665 {
5666 settings.isExpression = true;
5667 }
5668 else
5669 {
5670 QgsDebugMsgLevel( u"SLD label attribute error: %1"_s.arg( exp.evalErrorString() ), 3 );
5671 }
5672 }
5673 }
5674 else
5675 {
5676 QgsDebugMsgLevel( u"Info: PropertyName element not found."_s, 4 );
5677 return false;
5678 }
5679 }
5680 else
5681 {
5682 QgsDebugMsgLevel( u"Info: Label element not found."_s, 4 );
5683 return false;
5684 }
5685
5687 if ( textSymbolizerElem.hasAttribute( u"uom"_s ) )
5688 {
5689 sldUnitSize = QgsSymbolLayerUtils::decodeSldUom( textSymbolizerElem.attribute( u"uom"_s ) );
5690 }
5691
5692 QString fontFamily = u"Sans-Serif"_s;
5693 double fontPointSize = 10;
5695 int fontWeight = -1;
5696 bool fontItalic = false;
5697 bool fontUnderline = false;
5698
5699 // Font
5700 QDomElement fontElem = textSymbolizerElem.firstChildElement( u"Font"_s );
5701 if ( !fontElem.isNull() )
5702 {
5703 QgsStringMap fontSvgParams = QgsSymbolLayerUtils::getSvgParameterList( fontElem );
5704 for ( QgsStringMap::iterator it = fontSvgParams.begin(); it != fontSvgParams.end(); ++it )
5705 {
5706 QgsDebugMsgLevel( u"found fontSvgParams %1: %2"_s.arg( it.key(), it.value() ), 4 );
5707
5708 if ( it.key() == "font-family"_L1 )
5709 {
5710 fontFamily = it.value();
5711 }
5712 else if ( it.key() == "font-style"_L1 )
5713 {
5714 fontItalic = ( it.value() == "italic"_L1 ) || ( it.value() == "Italic"_L1 );
5715 }
5716 else if ( it.key() == "font-size"_L1 )
5717 {
5718 bool ok;
5719 double fontSize = it.value().toDouble( &ok );
5720 if ( ok )
5721 {
5722 fontPointSize = fontSize;
5723 fontUnitSize = sldUnitSize;
5724 }
5725 }
5726 else if ( it.key() == "font-weight"_L1 )
5727 {
5728 if ( ( it.value() == "bold"_L1 ) || ( it.value() == "Bold"_L1 ) )
5729 fontWeight = QFont::Bold;
5730 }
5731 else if ( it.key() == "font-underline"_L1 )
5732 {
5733 fontUnderline = ( it.value() == "underline"_L1 ) || ( it.value() == "Underline"_L1 );
5734 }
5735 }
5736 }
5737
5738 QgsTextFormat format;
5739 QFont font( fontFamily, 1, fontWeight, fontItalic );
5740 font.setUnderline( fontUnderline );
5741 format.setFont( font );
5742 format.setSize( fontPointSize );
5743 format.setSizeUnit( fontUnitSize );
5744
5745 // Fill
5746 QDomElement fillElem = textSymbolizerElem.firstChildElement( u"Fill"_s );
5747 QColor textColor;
5748 Qt::BrushStyle textBrush = Qt::SolidPattern;
5749 QgsSymbolLayerUtils::fillFromSld( fillElem, textBrush, textColor );
5750 if ( textColor.isValid() )
5751 {
5752 QgsDebugMsgLevel( u"Info: textColor %1."_s.arg( QVariant( textColor ).toString() ), 4 );
5753 format.setColor( textColor );
5754 }
5755
5756 QgsTextBufferSettings bufferSettings;
5757
5758 // Halo
5759 QDomElement haloElem = textSymbolizerElem.firstChildElement( u"Halo"_s );
5760 if ( !haloElem.isNull() )
5761 {
5762 bufferSettings.setEnabled( true );
5763 bufferSettings.setSize( 1 );
5764
5765 QDomElement radiusElem = haloElem.firstChildElement( u"Radius"_s );
5766 if ( !radiusElem.isNull() )
5767 {
5768 bool ok;
5769 double bufferSize = radiusElem.text().toDouble( &ok );
5770 if ( ok )
5771 {
5772 bufferSettings.setSize( bufferSize );
5773 bufferSettings.setSizeUnit( sldUnitSize );
5774 }
5775 }
5776
5777 QDomElement haloFillElem = haloElem.firstChildElement( u"Fill"_s );
5778 QColor bufferColor;
5779 Qt::BrushStyle bufferBrush = Qt::SolidPattern;
5780 QgsSymbolLayerUtils::fillFromSld( haloFillElem, bufferBrush, bufferColor );
5781 if ( bufferColor.isValid() )
5782 {
5783 QgsDebugMsgLevel( u"Info: bufferColor %1."_s.arg( QVariant( bufferColor ).toString() ), 4 );
5784 bufferSettings.setColor( bufferColor );
5785 }
5786 }
5787
5788 // LabelPlacement
5789 QDomElement labelPlacementElem = textSymbolizerElem.firstChildElement( u"LabelPlacement"_s );
5790 if ( !labelPlacementElem.isNull() )
5791 {
5792 // PointPlacement
5793 QDomElement pointPlacementElem = labelPlacementElem.firstChildElement( u"PointPlacement"_s );
5794 if ( !pointPlacementElem.isNull() )
5795 {
5798 {
5800 }
5801
5802 QDomElement displacementElem = pointPlacementElem.firstChildElement( u"Displacement"_s );
5803 if ( !displacementElem.isNull() )
5804 {
5805 QDomElement displacementXElem = displacementElem.firstChildElement( u"DisplacementX"_s );
5806 if ( !displacementXElem.isNull() )
5807 {
5808 bool ok;
5809 double xOffset = displacementXElem.text().toDouble( &ok );
5810 if ( ok )
5811 {
5812 settings.xOffset = xOffset;
5813 settings.offsetUnits = sldUnitSize;
5814 }
5815 }
5816 QDomElement displacementYElem = displacementElem.firstChildElement( u"DisplacementY"_s );
5817 if ( !displacementYElem.isNull() )
5818 {
5819 bool ok;
5820 double yOffset = displacementYElem.text().toDouble( &ok );
5821 if ( ok )
5822 {
5823 settings.yOffset = yOffset;
5824 settings.offsetUnits = sldUnitSize;
5825 }
5826 }
5827 }
5828 QDomElement anchorPointElem = pointPlacementElem.firstChildElement( u"AnchorPoint"_s );
5829 if ( !anchorPointElem.isNull() )
5830 {
5831 QDomElement anchorPointXElem = anchorPointElem.firstChildElement( u"AnchorPointX"_s );
5832 if ( !anchorPointXElem.isNull() )
5833 {
5834 bool ok;
5835 double xOffset = anchorPointXElem.text().toDouble( &ok );
5836 if ( ok )
5837 {
5838 settings.xOffset = xOffset;
5839 settings.offsetUnits = sldUnitSize;
5840 }
5841 }
5842 QDomElement anchorPointYElem = anchorPointElem.firstChildElement( u"AnchorPointY"_s );
5843 if ( !anchorPointYElem.isNull() )
5844 {
5845 bool ok;
5846 double yOffset = anchorPointYElem.text().toDouble( &ok );
5847 if ( ok )
5848 {
5849 settings.yOffset = yOffset;
5850 settings.offsetUnits = sldUnitSize;
5851 }
5852 }
5853 }
5854
5855 QDomElement rotationElem = pointPlacementElem.firstChildElement( u"Rotation"_s );
5856 if ( !rotationElem.isNull() )
5857 {
5858 bool ok;
5859 double rotation = rotationElem.text().toDouble( &ok );
5860 if ( ok )
5861 {
5862 settings.angleOffset = 360 - rotation;
5863 }
5864 }
5865 }
5866 else
5867 {
5868 // PointPlacement
5869 QDomElement linePlacementElem = labelPlacementElem.firstChildElement( u"LinePlacement"_s );
5870 if ( !linePlacementElem.isNull() )
5871 {
5873 }
5874 }
5875 }
5876
5877 // read vendor options
5878 QgsStringMap vendorOptions;
5879 QDomElement vendorOptionElem = textSymbolizerElem.firstChildElement( u"VendorOption"_s );
5880 while ( !vendorOptionElem.isNull() && vendorOptionElem.localName() == "VendorOption"_L1 )
5881 {
5882 QString optionName = vendorOptionElem.attribute( u"name"_s );
5883 QString optionValue;
5884 if ( vendorOptionElem.firstChild().nodeType() == QDomNode::TextNode )
5885 {
5886 optionValue = vendorOptionElem.firstChild().nodeValue();
5887 }
5888 else
5889 {
5890 if ( vendorOptionElem.firstChild().nodeType() == QDomNode::ElementNode && vendorOptionElem.firstChild().localName() == "Literal"_L1 )
5891 {
5892 QgsDebugMsgLevel( vendorOptionElem.firstChild().localName(), 2 );
5893 optionValue = vendorOptionElem.firstChild().firstChild().nodeValue();
5894 }
5895 else
5896 {
5897 QgsDebugError( u"unexpected child of %1 named %2"_s.arg( vendorOptionElem.localName(), optionName ) );
5898 }
5899 }
5900
5901 if ( !optionName.isEmpty() && !optionValue.isEmpty() )
5902 {
5903 vendorOptions[optionName] = optionValue;
5904 }
5905
5906 vendorOptionElem = vendorOptionElem.nextSiblingElement();
5907 }
5908 if ( !vendorOptions.isEmpty() )
5909 {
5910 for ( QgsStringMap::iterator it = vendorOptions.begin(); it != vendorOptions.end(); ++it )
5911 {
5912 if ( it.key() == "underlineText"_L1 && it.value() == "true"_L1 )
5913 {
5914 font.setUnderline( true );
5915 format.setFont( font );
5916 }
5917 else if ( it.key() == "strikethroughText"_L1 && it.value() == "true"_L1 )
5918 {
5919 font.setStrikeOut( true );
5920 format.setFont( font );
5921 }
5922 else if ( it.key() == "maxDisplacement"_L1 )
5923 {
5925 }
5926 else if ( it.key() == "followLine"_L1 && it.value() == "true"_L1 )
5927 {
5929 {
5931 }
5932 else
5933 {
5935 }
5936 }
5937 else if ( it.key() == "maxAngleDelta"_L1 )
5938 {
5939 bool ok;
5940 double angle = it.value().toDouble( &ok );
5941 if ( ok )
5942 {
5943 settings.maxCurvedCharAngleIn = angle;
5944 settings.maxCurvedCharAngleOut = angle;
5945 }
5946 }
5947 // miscellaneous options
5948 else if ( it.key() == "conflictResolution"_L1 && it.value() == "false"_L1 )
5949 {
5951 }
5952 else if ( it.key() == "forceLeftToRight"_L1 && it.value() == "false"_L1 )
5953 {
5955 }
5956 else if ( it.key() == "group"_L1 && it.value() == "yes"_L1 )
5957 {
5958 settings.lineSettings().setMergeLines( true );
5959 }
5960 else if ( it.key() == "labelAllGroup"_L1 && it.value() == "true"_L1 )
5961 {
5962 settings.lineSettings().setMergeLines( true );
5963 }
5964 }
5965 }
5966
5967 format.setBuffer( bufferSettings );
5968 settings.setFormat( format );
5969 return true;
5970}
5971
5973{
5975
5976 return mEditFormConfig;
5977}
5978
5980{
5982
5983 if ( mEditFormConfig == editFormConfig )
5984 return;
5985
5986 mEditFormConfig = editFormConfig;
5987 mEditFormConfig.onRelationsLoaded();
5988 emit editFormConfigChanged();
5989}
5990
5992{
5994
5995 QgsAttributeTableConfig config = mAttributeTableConfig;
5996
5997 if ( config.isEmpty() )
5998 config.update( fields() );
5999
6000 return config;
6001}
6002
6004{
6006
6007 if ( mAttributeTableConfig != attributeTableConfig )
6008 {
6009 mAttributeTableConfig = attributeTableConfig;
6010 emit configChanged();
6011 }
6012}
6013
6015{
6016 // called in a non-thread-safe way in some cases when calculating aggregates in a different thread
6018
6020}
6021
6028
6030{
6032
6033 if ( !mDiagramLayerSettings )
6034 mDiagramLayerSettings = new QgsDiagramLayerSettings();
6035 *mDiagramLayerSettings = s;
6036}
6037
6039{
6041
6042 QgsLayerMetadataFormatter htmlFormatter( metadata() );
6043 QString myMetadata = u"<html><head></head>\n<body>\n"_s;
6044
6045 myMetadata += generalHtmlMetadata();
6046
6047 // Begin Provider section
6048 myMetadata += u"<h1>"_s + tr( "Information from provider" ) + u"</h1>\n<hr>\n"_s;
6049 myMetadata += "<table class=\"list-view\">\n"_L1;
6050
6051 // storage type
6052 if ( !storageType().isEmpty() )
6053 {
6054 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Storage" ) + u"</td><td>"_s + storageType() + u"</td></tr>\n"_s;
6055 }
6056
6057 // comment
6058 if ( !dataComment().isEmpty() )
6059 {
6060 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Comment" ) + u"</td><td>"_s + dataComment() + u"</td></tr>\n"_s;
6061 }
6062
6063 // encoding
6064 if ( const QgsVectorDataProvider *provider = dataProvider() )
6065 {
6066 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Encoding" ) + u"</td><td>"_s + provider->encoding() + u"</td></tr>\n"_s;
6067 myMetadata += provider->htmlMetadata();
6068 }
6069
6070 if ( isSpatial() )
6071 {
6072 // geom type
6074 if ( static_cast<int>( type ) < 0 || static_cast< int >( type ) > static_cast< int >( Qgis::GeometryType::Null ) )
6075 {
6076 QgsDebugMsgLevel( u"Invalid vector type"_s, 2 );
6077 }
6078 else
6079 {
6080 QString typeString( u"%1 (%2)"_s.arg( QgsWkbTypes::geometryDisplayString( geometryType() ), QgsWkbTypes::displayString( wkbType() ) ) );
6081 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Geometry type" ) + u"</td><td>"_s + typeString + u"</td></tr>\n"_s;
6082 }
6083
6084 // geom column name
6085 if ( const QgsVectorDataProvider *provider = dataProvider(); provider && !provider->geometryColumnName().isEmpty() )
6086 {
6087 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Geometry column" ) + u"</td><td>"_s + provider->geometryColumnName() + u"</td></tr>\n"_s;
6088 }
6089
6090 // Extent
6091 // Try to display extent 3D by default. If empty (probably because the data is 2D), fallback to the 2D version
6092 const QgsBox3D extentBox3D = extent3D();
6093 const QString extentAsStr = !extentBox3D.isEmpty() ? extentBox3D.toString() : extent().toString();
6094 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Extent" ) + u"</td><td>"_s + extentAsStr + u"</td></tr>\n"_s;
6095 }
6096
6097 // feature count
6098 QLocale locale = QLocale();
6099 locale.setNumberOptions( locale.numberOptions() &= ~QLocale::NumberOption::OmitGroupSeparator );
6100 myMetadata += u"<tr><td class=\"highlight\">"_s
6101 + tr( "Feature count" )
6102 + u"</td><td>"_s
6103 + ( featureCount() == -1 ? tr( "unknown" ) : locale.toString( static_cast<qlonglong>( featureCount() ) ) )
6104 + u"</td></tr>\n"_s;
6105
6106 // End Provider section
6107 myMetadata += "</table>\n<br><br>"_L1;
6108
6109 if ( isSpatial() )
6110 {
6111 // CRS
6112 myMetadata += crsHtmlMetadata();
6113 }
6114
6115 // identification section
6116 myMetadata += u"<h1>"_s + tr( "Identification" ) + u"</h1>\n<hr>\n"_s;
6117 myMetadata += htmlFormatter.identificationSectionHtml();
6118 myMetadata += "<br><br>\n"_L1;
6119
6120 // extent section
6121 myMetadata += u"<h1>"_s + tr( "Extent" ) + u"</h1>\n<hr>\n"_s;
6122 myMetadata += htmlFormatter.extentSectionHtml( isSpatial() );
6123 myMetadata += "<br><br>\n"_L1;
6124
6125 // Start the Access section
6126 myMetadata += u"<h1>"_s + tr( "Access" ) + u"</h1>\n<hr>\n"_s;
6127 myMetadata += htmlFormatter.accessSectionHtml();
6128 myMetadata += "<br><br>\n"_L1;
6129
6130 // Fields section
6131 myMetadata += u"<h1>"_s + tr( "Fields" ) + u"</h1>\n<hr>\n<table class=\"list-view\">\n"_s;
6132
6133 // primary key
6135 if ( !pkAttrList.isEmpty() )
6136 {
6137 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Primary key attributes" ) + u"</td><td>"_s;
6138 const auto constPkAttrList = pkAttrList;
6139 for ( int idx : constPkAttrList )
6140 {
6141 myMetadata += fields().at( idx ).name() + ' ';
6142 }
6143 myMetadata += "</td></tr>\n"_L1;
6144 }
6145
6146 const QgsFields myFields = fields();
6147
6148 // count fields
6149 myMetadata += u"<tr><td class=\"highlight\">"_s + tr( "Count" ) + u"</td><td>"_s + QString::number( myFields.size() ) + u"</td></tr>\n"_s;
6150
6151 myMetadata += "</table>\n<br><table width=\"100%\" class=\"tabular-view\">\n"_L1;
6152 myMetadata += "<tr><th>"_L1 + tr( "Field" ) + "</th><th>"_L1 + tr( "Type" ) + "</th><th>"_L1 + tr( "Length" ) + "</th><th>"_L1 + tr( "Precision" ) + "</th><th>"_L1 + tr( "Comment" ) + "</th></tr>\n"_L1;
6153
6154 for ( int i = 0; i < myFields.size(); ++i )
6155 {
6156 QgsField myField = myFields.at( i );
6157 QString rowClass;
6158 if ( i % 2 )
6159 rowClass = u"class=\"odd-row\""_s;
6160 myMetadata += "<tr "_L1
6161 + rowClass
6162 + "><td>"_L1
6163 + myField.displayNameWithAlias()
6164 + "</td><td>"_L1
6165 + myField.typeName()
6166 + "</td><td>"_L1
6167 + QString::number( myField.length() )
6168 + "</td><td>"_L1
6169 + QString::number( myField.precision() )
6170 + "</td><td>"_L1
6171 + myField.comment()
6172 + "</td></tr>\n"_L1;
6173 }
6174
6175 //close field list
6176 myMetadata += "</table>\n<br><br>"_L1;
6177
6178 // Start the contacts section
6179 myMetadata += u"<h1>"_s + tr( "Contacts" ) + u"</h1>\n<hr>\n"_s;
6180 myMetadata += htmlFormatter.contactsSectionHtml();
6181 myMetadata += "<br><br>\n"_L1;
6182
6183 // Start the links section
6184 myMetadata += u"<h1>"_s + tr( "Links" ) + u"</h1>\n<hr>\n"_s;
6185 myMetadata += htmlFormatter.linksSectionHtml();
6186 myMetadata += "<br><br>\n"_L1;
6187
6188 // Start the history section
6189 myMetadata += u"<h1>"_s + tr( "History" ) + u"</h1>\n<hr>\n"_s;
6190 myMetadata += htmlFormatter.historySectionHtml();
6191 myMetadata += "<br><br>\n"_L1;
6192
6193 myMetadata += customPropertyHtmlMetadata();
6194
6195 myMetadata += "\n</body>\n</html>\n"_L1;
6196 return myMetadata;
6197}
6198
6199void QgsVectorLayer::invalidateSymbolCountedFlag()
6200{
6202
6203 mSymbolFeatureCounted = false;
6204}
6205
6206void QgsVectorLayer::onFeatureCounterCompleted()
6207{
6209
6210 onSymbolsCounted();
6211 mFeatureCounter = nullptr;
6212}
6213
6214void QgsVectorLayer::onFeatureCounterTerminated()
6215{
6217
6218 mFeatureCounter = nullptr;
6219}
6220
6221void QgsVectorLayer::onJoinedFieldsChanged()
6222{
6224
6225 // some of the fields of joined layers have changed -> we need to update this layer's fields too
6226 updateFields();
6227}
6228
6229void QgsVectorLayer::onFeatureAdded( QgsFeatureId fid )
6230{
6232
6233 updateExtents();
6234
6235 emit featureAdded( fid );
6236}
6237
6238void QgsVectorLayer::onFeatureDeleted( QgsFeatureId fid )
6239{
6241
6242 updateExtents();
6243
6244 if ( mEditCommandActive || mCommitChangesActive )
6245 {
6246 mDeletedFids << fid;
6247 }
6248 else
6249 {
6250 mSelectedFeatureIds.remove( fid );
6251 emit featuresDeleted( QgsFeatureIds() << fid );
6252 }
6253
6254 emit featureDeleted( fid );
6255}
6256
6257void QgsVectorLayer::onRelationsLoaded()
6258{
6260
6261 mEditFormConfig.onRelationsLoaded();
6262}
6263
6264void QgsVectorLayer::onSymbolsCounted()
6265{
6267
6268 if ( mFeatureCounter )
6269 {
6270 mSymbolFeatureCounted = true;
6271 mSymbolFeatureCountMap = mFeatureCounter->symbolFeatureCountMap();
6272 mSymbolFeatureIdMap = mFeatureCounter->symbolFeatureIdMap();
6274 }
6275}
6276
6277QList<QgsRelation> QgsVectorLayer::referencingRelations( int idx ) const
6278{
6280
6281 if ( QgsProject *p = project() )
6282 return p->relationManager()->referencingRelations( this, idx );
6283 else
6284 return {};
6285}
6286
6287QList<QgsWeakRelation> QgsVectorLayer::weakRelations() const
6288{
6290
6291 return mWeakRelations;
6292}
6293
6294void QgsVectorLayer::setWeakRelations( const QList<QgsWeakRelation> &relations )
6295{
6297
6298 mWeakRelations = relations;
6299}
6300
6301bool QgsVectorLayer::loadAuxiliaryLayer( const QgsAuxiliaryStorage &storage, const QString &key )
6302{
6304
6305 bool rc = false;
6306
6307 QString joinKey = mAuxiliaryLayerKey;
6308 if ( !key.isEmpty() )
6309 joinKey = key;
6310
6311 if ( storage.isValid() && !joinKey.isEmpty() )
6312 {
6313 QgsAuxiliaryLayer *alayer = nullptr;
6314
6315 int idx = fields().lookupField( joinKey );
6316
6317 if ( idx >= 0 )
6318 {
6319 alayer = storage.createAuxiliaryLayer( fields().field( idx ), this );
6320
6321 if ( alayer )
6322 {
6323 setAuxiliaryLayer( alayer );
6324 rc = true;
6325 }
6326 }
6327 }
6328
6329 return rc;
6330}
6331
6333{
6335
6336 mAuxiliaryLayerKey.clear();
6337
6338 if ( mAuxiliaryLayer )
6339 removeJoin( mAuxiliaryLayer->id() );
6340
6341 if ( alayer )
6342 {
6343 addJoin( alayer->joinInfo() );
6344
6345 if ( !alayer->isEditable() )
6346 alayer->startEditing();
6347
6348 mAuxiliaryLayerKey = alayer->joinInfo().targetFieldName();
6349 }
6350
6351 mAuxiliaryLayer.reset( alayer );
6352 if ( mAuxiliaryLayer )
6353 mAuxiliaryLayer->setParent( this );
6354 updateFields();
6355}
6356
6358{
6360
6361 return mAuxiliaryLayer.get();
6362}
6363
6365{
6367
6368 return mAuxiliaryLayer.get();
6369}
6370
6371QSet<QgsMapLayerDependency> QgsVectorLayer::dependencies() const
6372{
6374
6375 if ( mDataProvider )
6376 return mDataProvider->dependencies() + mDependencies;
6377 return mDependencies;
6378}
6379
6380void QgsVectorLayer::emitDataChanged()
6381{
6383
6384 if ( mDataChangedFired )
6385 return;
6386
6387 // If we are asked to fire dataChanged from a layer we depend on,
6388 // be sure that this layer is not in the process of committing its changes, because
6389 // we will be asked to fire dataChanged at the end of his commit, and we don't
6390 // want to fire this signal more than necessary.
6391 if ( QgsVectorLayer *layerWeDependUpon = qobject_cast<QgsVectorLayer *>( sender() ); layerWeDependUpon && layerWeDependUpon->mCommitChangesActive )
6392 return;
6393
6394 updateExtents(); // reset cached extent to reflect data changes
6395
6396 mDataChangedFired = true;
6397 emit dataChanged();
6398 mDataChangedFired = false;
6399}
6400
6401void QgsVectorLayer::onDependencyAfterCommitChanges()
6402{
6404
6405 if ( mDataProvider && mDataProvider->capabilities().testFlag( Qgis::VectorProviderCapability::CacheData ) )
6406 mDataProvider->reloadData();
6407 else
6408 emitDataChanged();
6409}
6410
6411bool QgsVectorLayer::setDependencies( const QSet<QgsMapLayerDependency> &oDeps )
6412{
6414
6415 QSet<QgsMapLayerDependency> deps;
6416 const auto constODeps = oDeps;
6417 for ( const QgsMapLayerDependency &dep : constODeps )
6418 {
6419 if ( dep.origin() == QgsMapLayerDependency::FromUser )
6420 deps << dep;
6421 }
6422
6423 QSet<QgsMapLayerDependency> toAdd = deps - dependencies();
6424
6425 // disconnect layers that are not present in the list of dependencies anymore
6426 if ( QgsProject *p = project() )
6427 {
6428 for ( const QgsMapLayerDependency &dep : std::as_const( mDependencies ) )
6429 {
6430 QgsVectorLayer *lyr = static_cast<QgsVectorLayer *>( p->mapLayer( dep.layerId() ) );
6431 if ( !lyr )
6432 continue;
6433 disconnect( lyr, &QgsVectorLayer::featureAdded, this, &QgsVectorLayer::emitDataChanged );
6434 disconnect( lyr, &QgsVectorLayer::featureDeleted, this, &QgsVectorLayer::emitDataChanged );
6435 disconnect( lyr, &QgsVectorLayer::geometryChanged, this, &QgsVectorLayer::emitDataChanged );
6436 disconnect( lyr, &QgsVectorLayer::dataChanged, this, &QgsVectorLayer::emitDataChanged );
6438 disconnect( lyr, &QgsVectorLayer::afterCommitChanges, this, &QgsVectorLayer::onDependencyAfterCommitChanges );
6439 }
6440 }
6441
6442 // assign new dependencies
6443 if ( mDataProvider )
6444 mDependencies = mDataProvider->dependencies() + deps;
6445 else
6446 mDependencies = deps;
6447 emit dependenciesChanged();
6448
6449 // connect to new layers
6450 if ( QgsProject *p = project() )
6451 {
6452 for ( const QgsMapLayerDependency &dep : std::as_const( mDependencies ) )
6453 {
6454 QgsVectorLayer *lyr = static_cast<QgsVectorLayer *>( p->mapLayer( dep.layerId() ) );
6455 if ( !lyr )
6456 continue;
6457 connect( lyr, &QgsVectorLayer::featureAdded, this, &QgsVectorLayer::emitDataChanged );
6458 connect( lyr, &QgsVectorLayer::featureDeleted, this, &QgsVectorLayer::emitDataChanged );
6459 connect( lyr, &QgsVectorLayer::geometryChanged, this, &QgsVectorLayer::emitDataChanged );
6460 connect( lyr, &QgsVectorLayer::dataChanged, this, &QgsVectorLayer::emitDataChanged );
6462 connect( lyr, &QgsVectorLayer::afterCommitChanges, this, &QgsVectorLayer::onDependencyAfterCommitChanges );
6463 }
6464 }
6465
6466 // if new layers are present, emit a data change
6467 if ( !toAdd.isEmpty() )
6468 emitDataChanged();
6469
6470 return true;
6471}
6472
6474{
6476
6477 if ( fieldIndex < 0 || fieldIndex >= mFields.count() || !mDataProvider )
6479
6480 QgsFieldConstraints::Constraints constraints = mFields.at( fieldIndex ).constraints().constraints();
6481
6482 // make sure provider constraints are always present!
6483 if ( mFields.fieldOrigin( fieldIndex ) == Qgis::FieldOrigin::Provider )
6484 {
6485 constraints |= mDataProvider->fieldConstraints( mFields.fieldOriginIndex( fieldIndex ) );
6486 }
6487
6488 return constraints;
6489}
6490
6491QMap< QgsFieldConstraints::Constraint, QgsFieldConstraints::ConstraintStrength> QgsVectorLayer::fieldConstraintsAndStrength( int fieldIndex ) const
6492{
6494
6495 QMap< QgsFieldConstraints::Constraint, QgsFieldConstraints::ConstraintStrength > m;
6496
6497 if ( fieldIndex < 0 || fieldIndex >= mFields.count() )
6498 return m;
6499
6500 QString name = mFields.at( fieldIndex ).name();
6501
6502 QMap< QPair< QString, QgsFieldConstraints::Constraint >, QgsFieldConstraints::ConstraintStrength >::const_iterator conIt = mFieldConstraintStrength.constBegin();
6503 for ( ; conIt != mFieldConstraintStrength.constEnd(); ++conIt )
6504 {
6505 if ( conIt.key().first == name )
6506 {
6507 m[conIt.key().second] = mFieldConstraintStrength.value( conIt.key() );
6508 }
6509 }
6510
6511 return m;
6512}
6513
6515{
6517
6518 if ( index < 0 || index >= mFields.count() )
6519 return;
6520
6521 QString name = mFields.at( index ).name();
6522
6523 // add constraint to existing constraints
6524 QgsFieldConstraints::Constraints constraints = mFieldConstraints.value( name, QgsFieldConstraints::Constraints() );
6525 constraints |= constraint;
6526 mFieldConstraints.insert( name, constraints );
6527
6528 mFieldConstraintStrength.insert( qMakePair( name, constraint ), strength );
6529
6530 updateFields();
6531}
6532
6534{
6536
6537 if ( index < 0 || index >= mFields.count() )
6538 return;
6539
6540 QString name = mFields.at( index ).name();
6541
6542 // remove constraint from existing constraints
6543 QgsFieldConstraints::Constraints constraints = mFieldConstraints.value( name, QgsFieldConstraints::Constraints() );
6544 constraints &= ~constraint;
6545 mFieldConstraints.insert( name, constraints );
6546
6547 mFieldConstraintStrength.remove( qMakePair( name, constraint ) );
6548
6549 updateFields();
6550}
6551
6553{
6555
6556 if ( index < 0 || index >= mFields.count() )
6557 return QString();
6558
6559 return mFields.at( index ).constraints().constraintExpression();
6560}
6561
6563{
6565
6566 if ( index < 0 || index >= mFields.count() )
6567 return QString();
6568
6569 return mFields.at( index ).constraints().constraintDescription();
6570}
6571
6572void QgsVectorLayer::setConstraintExpression( int index, const QString &expression, const QString &description )
6573{
6575
6576 if ( index < 0 || index >= mFields.count() )
6577 return;
6578
6579 if ( expression.isEmpty() )
6580 {
6581 mFieldConstraintExpressions.remove( mFields.at( index ).name() );
6582 }
6583 else
6584 {
6585 mFieldConstraintExpressions.insert( mFields.at( index ).name(), qMakePair( expression, description ) );
6586 }
6587 updateFields();
6588}
6589
6591{
6593
6594 if ( index < 0 || index >= mFields.count() )
6595 return;
6596
6597 mFieldConfigurationFlags.insert( mFields.at( index ).name(), flags );
6598 updateFields();
6599}
6600
6602{
6604
6605 if ( index < 0 || index >= mFields.count() )
6606 return;
6607 Qgis::FieldConfigurationFlags flags = mFields.at( index ).configurationFlags();
6608 flags.setFlag( flag, active );
6610}
6611
6613{
6615
6616 if ( index < 0 || index >= mFields.count() )
6618
6619 return mFields.at( index ).configurationFlags();
6620}
6621
6623{
6625
6626 if ( index < 0 || index >= mFields.count() )
6627 return;
6628
6629 if ( setup.isNull() )
6630 mFieldWidgetSetups.remove( mFields.at( index ).name() );
6631 else
6632 mFieldWidgetSetups.insert( mFields.at( index ).name(), setup );
6633 updateFields();
6634}
6635
6637{
6639
6640 if ( index < 0 || index >= mFields.count() )
6641 return QgsEditorWidgetSetup();
6642
6643 return mFields.at( index ).editorWidgetSetup();
6644}
6645
6646QgsAbstractVectorLayerLabeling *QgsVectorLayer::readLabelingFromCustomProperties()
6647{
6649
6651 if ( customProperty( u"labeling"_s ).toString() == "pal"_L1 )
6652 {
6653 if ( customProperty( u"labeling/enabled"_s, QVariant( false ) ).toBool() )
6654 {
6655 // try to load from custom properties
6656 QgsPalLayerSettings settings;
6657 settings.readFromLayerCustomProperties( this );
6658 labeling = new QgsVectorLayerSimpleLabeling( settings );
6659 }
6660
6661 // also clear old-style labeling config
6662 removeCustomProperty( u"labeling"_s );
6663 const auto constCustomPropertyKeys = customPropertyKeys();
6664 for ( const QString &key : constCustomPropertyKeys )
6665 {
6666 if ( key.startsWith( "labeling/"_L1 ) )
6667 removeCustomProperty( key );
6668 }
6669 }
6670
6671 return labeling;
6672}
6673
6675{
6677
6678 return mAllowCommit;
6679}
6680
6682{
6684
6685 if ( mAllowCommit == allowCommit )
6686 return;
6687
6688 mAllowCommit = allowCommit;
6689 emit allowCommitChanged();
6690}
6691
6693{
6695
6696 return mGeometryOptions.get();
6697}
6698
6705
6707{
6709
6710 return mReadExtentFromXml;
6711}
6712
6713void QgsVectorLayer::onDirtyTransaction( const QString &sql, const QString &name )
6714{
6716
6718 if ( tr && mEditBuffer )
6719 {
6720 qobject_cast<QgsVectorLayerEditPassthrough *>( mEditBuffer )->update( tr, sql, name );
6721 }
6722}
6723
6724QList<QgsVectorLayer *> QgsVectorLayer::DeleteContext::handledLayers( bool includeAuxiliaryLayers ) const
6725{
6726 QList<QgsVectorLayer *> layers;
6727 QMap<QgsVectorLayer *, QgsFeatureIds>::const_iterator i;
6728 for ( i = mHandledFeatures.begin(); i != mHandledFeatures.end(); ++i )
6729 {
6730 if ( includeAuxiliaryLayers || !qobject_cast< QgsAuxiliaryLayer * >( i.key() ) )
6731 layers.append( i.key() );
6732 }
6733 return layers;
6734}
6735
Provides global constants and enumerations for use throughout the application.
Definition qgis.h:62
@ Action
Expression functions.
Definition qgis.h:466
@ FormInitCode
Map layers' action.
Definition qgis.h:467
@ SelectAtId
Fast access to features using their ID.
Definition qgis.h:533
@ CacheData
Provider caches source data and should force provider data reloads when dependent layers are committe...
Definition qgis.h:551
@ CreateRenderer
Provider can create feature renderers using backend-specific formatting information....
Definition qgis.h:547
@ CreateLabeling
Provider can set labeling settings using backend-specific formatting information. Since QGIS 3....
Definition qgis.h:548
@ ReadLayerMetadata
Provider can read layer metadata from data store. Since QGIS 3.0. See QgsDataProvider::layerMetadata(...
Definition qgis.h:544
@ DeleteFeatures
Allows deletion of features.
Definition qgis.h:528
QFlags< VectorRenderingSimplificationFlag > VectorRenderingSimplificationFlags
Simplification flags for vector feature rendering.
Definition qgis.h:3212
@ Composition
Fix relation, related elements are part of the parent and a parent copy will copy any children or del...
Definition qgis.h:4707
@ Association
Loose relation, related elements are not part of the parent and a parent copy will not copy any child...
Definition qgis.h:4706
GeometryOperationResult
Success or failure of a geometry operation.
Definition qgis.h:2162
@ InvalidInputGeometryType
The input geometry (ring, part, split line, etc.) has not the correct geometry type.
Definition qgis.h:2166
@ Success
Operation succeeded.
Definition qgis.h:2163
@ SelectionIsEmpty
No features were selected.
Definition qgis.h:2167
@ AddRingNotInExistingFeature
The input ring doesn't have any existing ring to fit into.
Definition qgis.h:2178
@ AddRingNotClosed
The input ring is not closed.
Definition qgis.h:2175
@ SelectionIsGreaterThanOne
More than one features were selected.
Definition qgis.h:2168
@ LayerNotEditable
Cannot edit layer.
Definition qgis.h:2170
SpatialIndexPresence
Enumeration of spatial index presence states.
Definition qgis.h:585
@ Unknown
Spatial index presence cannot be determined, index may or may not exist.
Definition qgis.h:586
VectorRenderingSimplificationFlag
Simplification flags for vector feature rendering.
Definition qgis.h:3197
@ NoSimplification
No simplification can be applied.
Definition qgis.h:3198
@ OverPoint
Arranges candidates over a point (or centroid of a polygon), or at a preset offset from the point....
Definition qgis.h:1275
@ Curved
Arranges candidates following the curvature of a line feature. Applies to line layers only.
Definition qgis.h:1277
@ AroundPoint
Arranges candidates in a circle around a point (or centroid of a polygon). Applies to point or polygo...
Definition qgis.h:1274
@ Line
Arranges candidates parallel to a generalised line representing the feature or parallel to a polygon'...
Definition qgis.h:1276
@ Horizontal
Arranges horizontal candidates scattered throughout a polygon feature. Applies to polygon layers only...
Definition qgis.h:1278
@ PerimeterCurved
Arranges candidates following the curvature of a polygon's boundary. Applies to polygon layers only.
Definition qgis.h:1281
QFlags< VectorLayerTypeFlag > VectorLayerTypeFlags
Vector layer type flags.
Definition qgis.h:440
VectorSimplificationAlgorithm
Simplification algorithms for vector features.
Definition qgis.h:3181
@ Distance
The simplification uses the distance between points to remove duplicate points.
Definition qgis.h:3182
@ File
Load the Python code from an external file.
Definition qgis.h:6029
@ Environment
Use the Python code available in the Python environment.
Definition qgis.h:6031
@ NoSource
Do not use Python code at all.
Definition qgis.h:6028
@ Dialog
Use the Python code provided in the dialog.
Definition qgis.h:6030
@ ExactIntersect
Use exact geometry intersection (slower) instead of bounding boxes.
Definition qgis.h:2331
@ SubsetOfAttributes
Fetch only a subset of attributes (setSubsetOfAttributes sets this flag).
Definition qgis.h:2330
@ NoGeometry
Geometry is not required. It may still be returned if e.g. required for a filter condition.
Definition qgis.h:2329
@ FastExtent3D
Provider's 3D extent retrieval via QgsDataProvider::extent3D() is always guaranteed to be trivial/fas...
Definition qgis.h:2446
@ FastExtent2D
Provider's 2D extent retrieval via QgsDataProvider::extent() is always guaranteed to be trivial/fast ...
Definition qgis.h:2445
@ BufferedGroups
Buffered transactional editing means that all editable layers in the buffered transaction group are t...
Definition qgis.h:4156
@ Mac
MacOS specific.
Definition qgis.h:4991
@ OpenUrl
Open URL action.
Definition qgis.h:4994
@ Unix
Unix specific.
Definition qgis.h:4993
@ SubmitUrlMultipart
POST data to an URL using "multipart/form-data".
Definition qgis.h:4996
@ Windows
Windows specific.
Definition qgis.h:4992
@ SubmitUrlEncoded
POST data to an URL, using "application/x-www-form-urlencoded" or "application/json" if the body is v...
Definition qgis.h:4995
FieldDomainMergePolicy
Merge policy for field domains.
Definition qgis.h:4107
@ UnsetField
Clears the field value so that the data provider backend will populate using any backend triggers or ...
Definition qgis.h:4111
FieldDomainSplitPolicy
Split policy for field domains.
Definition qgis.h:4090
@ Duplicate
Duplicate original value.
Definition qgis.h:4092
BlendMode
Blending modes defining the available composition modes that can be used when painting.
Definition qgis.h:5243
GeometryType
The geometry types are used to group Qgis::WkbType in a coarse way.
Definition qgis.h:379
@ Point
Points.
Definition qgis.h:380
@ Line
Lines.
Definition qgis.h:381
@ Polygon
Polygons.
Definition qgis.h:382
@ Unknown
Unknown types.
Definition qgis.h:383
@ Null
No geometry.
Definition qgis.h:384
@ Generated
A generated relation is a child of a polymorphic relation.
Definition qgis.h:4693
@ Normal
A normal relation.
Definition qgis.h:4692
FieldDuplicatePolicy
Duplicate policy for fields.
Definition qgis.h:4127
@ Duplicate
Duplicate original value.
Definition qgis.h:4129
static const float DEFAULT_MAPTOPIXEL_THRESHOLD
Default threshold between map coordinates and device coordinates for map2pixel simplification.
Definition qgis.h:6702
QFlags< DataProviderReadFlag > DataProviderReadFlags
Flags which control data provider construction.
Definition qgis.h:512
FeatureAvailability
Possible return value for QgsFeatureSource::hasFeatures() to determine if a source is empty.
Definition qgis.h:604
@ FeaturesMaybeAvailable
There may be features available in this source.
Definition qgis.h:607
@ FeaturesAvailable
There is at least one feature available in this source.
Definition qgis.h:606
@ NoFeaturesAvailable
There are certainly no features available in this source.
Definition qgis.h:605
@ Vector
Vector layer.
Definition qgis.h:207
FieldOrigin
Field origin.
Definition qgis.h:1824
@ Provider
Field originates from the underlying data provider of the vector layer.
Definition qgis.h:1826
@ Edit
Field has been temporarily added in editing mode.
Definition qgis.h:1828
@ Unknown
The field origin has not been specified.
Definition qgis.h:1825
@ Expression
Field is calculated from an expression.
Definition qgis.h:1829
@ Join
Field originates from a joined layer.
Definition qgis.h:1827
RenderUnit
Rendering size units.
Definition qgis.h:5496
@ Points
Points (e.g., for font sizes).
Definition qgis.h:5501
@ Pixels
Pixels.
Definition qgis.h:5499
@ LoadDefaultStyle
Reset the layer's style to the default for the datasource.
Definition qgis.h:495
@ ForceReadOnly
Open layer in a read-only mode.
Definition qgis.h:498
Aggregate
Available aggregates to calculate.
Definition qgis.h:6330
VertexMarkerType
Editing vertex markers, used for showing vertices during a edit operation.
Definition qgis.h:1953
@ NoMarker
No marker.
Definition qgis.h:1956
@ SemiTransparentCircle
Semi-transparent circle marker.
Definition qgis.h:1954
@ Cross
Cross marker.
Definition qgis.h:1955
VectorEditResult
Specifies the result of a vector layer edit operation.
Definition qgis.h:1938
@ Success
Edit operation was successful.
Definition qgis.h:1939
@ InvalidLayer
Edit failed due to invalid layer.
Definition qgis.h:1943
WkbType
The WKB type describes the number of dimensions a geometry has.
Definition qgis.h:294
@ Unknown
Unknown.
Definition qgis.h:295
FieldConfigurationFlag
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1841
@ HideFromWfs
Field is not available if layer is served as WFS from QGIS server.
Definition qgis.h:1845
@ NoFlag
No flag is defined.
Definition qgis.h:1842
@ HideFromWms
Field is not available if layer is served as WMS from QGIS server.
Definition qgis.h:1844
@ AllowOverlapIfRequired
Avoids overlapping labels when possible, but permit overlaps if labels for features cannot otherwise ...
Definition qgis.h:1236
QFlags< FieldConfigurationFlag > FieldConfigurationFlags
Configuration flags for fields These flags are meant to be user-configurable and are not describing a...
Definition qgis.h:1856
@ AlwaysAllowUpsideDown
Show upside down for all labels, including dynamic ones.
Definition qgis.h:1432
SelectBehavior
Specifies how a selection should be applied.
Definition qgis.h:1891
@ SetSelection
Set selection, removing any existing selection.
Definition qgis.h:1892
@ AddToSelection
Add selection to current selection.
Definition qgis.h:1893
@ IntersectSelection
Modify current selection to include only select features which match.
Definition qgis.h:1894
@ RemoveFromSelection
Remove from current selection.
Definition qgis.h:1895
Abstract base class for objects which generate elevation profiles.
virtual bool writeXml(QDomElement &collectionElem, const QgsPropertiesDefinition &definitions) const
Writes the current state of the property collection into an XML element.
Abstract base class - its implementations define different approaches to the labeling of a vector lay...
static QgsAbstractVectorLayerLabeling * create(const QDomElement &element, const QgsReadWriteContext &context)
Try to create instance of an implementation based on the XML data.
Storage and management of actions associated with a layer.
QList< QgsAction > actions(const QString &actionScope=QString()) const
Returns a list of actions that are available in the given action scope.
QUuid addAction(Qgis::AttributeActionType type, const QString &name, const QString &command, bool capture=false)
Add an action with the given name and action details.
Utility class that encapsulates an action based on vector attributes.
Definition qgsaction.h:38
Utility class for calculating aggregates for a field (or expression) over the features from a vector ...
static QgsNetworkContentFetcherRegistry * networkContentFetcherRegistry()
Returns the application's network content registry used for fetching temporary files during QGIS sess...
static QgsRuntimeProfiler * profiler()
Returns the application runtime profiler.
static QgsTaskManager * taskManager()
Returns the application's task manager, used for managing application wide background task handling.
A container for configuration of the attribute table.
void update(const QgsFields &fields)
Update the configuration with the given fields.
A vector of attributes.
Allows managing the auxiliary storage for a vector layer.
QgsVectorLayerJoinInfo joinInfo() const
Returns information to use for joining with primary key and so on.
Providing some utility methods to manage auxiliary storage.
QgsAuxiliaryLayer * createAuxiliaryLayer(const QgsField &field, QgsVectorLayer *layer) const
Creates an auxiliary layer for a vector layer.
bool isValid() const
Returns the status of the auxiliary storage currently defined.
A 3-dimensional box composed of x, y, z coordinates.
Definition qgsbox3d.h:45
QString toString(int precision=16) const
Returns a string representation of form xmin,ymin,zmin : xmax,ymax,zmax Coordinates will be truncated...
Definition qgsbox3d.cpp:326
bool isEmpty() const
Returns true if the box is empty.
Definition qgsbox3d.cpp:321
Holds conditional style information for a layer.
Represents a coordinate reference system (CRS).
bool isValid() const
Returns whether this CRS is correctly initialized and usable.
Contains information about the context in which a coordinate transform is executed.
Abstract base class for curved geometry type.
Definition qgscurve.h:36
virtual bool isClosed() const
Returns true if the curve is closed.
Definition qgscurve.cpp:53
QgsCurve * clone() const override=0
Clones the geometry by performing a deep copy.
void dataChanged()
Emitted whenever a change is made to the data provider which may have caused changes in the provider'...
void fullExtentCalculated()
Emitted whenever a deferred extent calculation is completed by the provider.
virtual QString dataSourceUri(bool expandAuthConfig=false) const
Gets the data source specification.
Stores the component parts of a data source URI (e.g.
bool useEstimatedMetadata() const
Returns true if estimated metadata should be used for the connection.
Provides a container for managing client side default values for fields.
bool isValid() const
Returns if this default value should be applied.
Stores the settings for rendering of all diagrams for a layer.
@ PositionX
X-coordinate data defined diagram position.
@ PositionY
Y-coordinate data defined diagram position.
@ Show
Whether to show the diagram.
Evaluates and returns the diagram settings relating to a diagram for a specific feature.
Contains configuration settings for an editor form.
Holder for the widget type and its configuration for a field.
QVariantMap config() const
Returns the widget configuration.
bool isNull() const
Returns true if there is no widget configured.
A embedded script entity for QgsObjectEntityVisitorInterface.
Single scope for storing variables and functions for use within a QgsExpressionContext.
void setFields(const QgsFields &fields)
Convenience function for setting a fields for the scope.
void setFeature(const QgsFeature &feature)
Convenience function for setting a feature for the scope.
static QList< QgsExpressionContextScope * > globalProjectLayerScopes(const QgsMapLayer *layer)
Creates a list of three scopes: global, layer's project and layer.
static QgsExpressionContextScope * layerScope(const QgsMapLayer *layer)
Creates a new scope which contains variables and functions relating to a QgsMapLayer.
Expression contexts are used to encapsulate the parameters around which a QgsExpression should be eva...
QgsExpressionContextScope * popScope()
Removes the last scope from the expression context and return it.
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.
Buffers information about expression fields for a vector layer.
void writeXml(QDomNode &layer_node, QDomDocument &document) const
Saves expressions to xml under the layer node.
An expression node which takes its value from a feature's field.
QString name() const
The name of the column.
Handles parsing and evaluation of expressions (formerly called "search strings").
bool prepare(const QgsExpressionContext *context)
Gets the expression ready for evaluation - find out column indexes.
QString expression() const
Returns the original, unmodified expression string.
bool hasParserError() const
Returns true if an error occurred when parsing the input expression.
QString evalErrorString() const
Returns evaluation error.
QString parserErrorString() const
Returns parser error.
QSet< QString > referencedColumns() const
Gets list of columns referenced by the expression.
static QString quotedColumnRef(QString name)
Returns a quoted column reference (in double quotes).
bool hasEvalError() const
Returns true if an error occurred when evaluating last input.
static int expressionToLayerFieldIndex(const QString &expression, const QgsVectorLayer *layer)
Attempts to resolve an expression to a field index from the given layer.
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.
bool close()
Call to end the iteration.
An interface for objects which generate feature renderers for vector layers.
Abstract base class for all 2D vector feature renderers.
static QgsFeatureRenderer * defaultRenderer(Qgis::GeometryType geomType)
Returns a new renderer - used by default in vector layers.
static QgsFeatureRenderer * load(QDomElement &symbologyElem, const QgsReadWriteContext &context)
create a renderer from XML element
static QgsFeatureRenderer * loadSld(const QDomNode &node, Qgis::GeometryType geomType, QString &errorMessage)
Create a new renderer according to the information contained in the UserStyle element of a SLD style ...
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 & setFilterFids(const QgsFeatureIds &fids)
Sets the feature IDs that should be fetched.
QgsFeatureRequest & setSubsetOfAttributes(const QgsAttributeList &attrs)
Set a subset of attributes that will be fetched.
QgsFeatureRequest & setFilterExpression(const QString &expression)
Set the filter expression.
QgsFeatureRequest & setExpressionContext(const QgsExpressionContext &context)
Sets the expression context used to evaluate filter expressions.
QgsFeatureRequest & setFilterFid(QgsFeatureId fid)
Sets the feature ID that should be fetched.
QFlags< Flag > Flags
virtual QgsFeatureIds allFeatureIds() const
Returns a list of all feature IDs for features present in the source.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
Definition qgsfeature.h:60
Q_INVOKABLE bool setAttribute(int field, const QVariant &attr)
Sets an attribute's value by field index.
QgsAttributes attributes
Definition qgsfeature.h:69
QgsFields fields
Definition qgsfeature.h:70
QgsFeatureId id
Definition qgsfeature.h:68
QgsGeometry geometry
Definition qgsfeature.h:71
bool hasGeometry() const
Returns true if the feature has an associated geometry.
bool isValid() const
Returns the validity of this feature.
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.
Definition qgsfeedback.h:44
bool isCanceled() const
Tells whether the operation has been canceled already.
Definition qgsfeedback.h:56
Stores information about constraints which may be present on a field.
ConstraintStrength
Strength of constraints.
void setConstraintStrength(Constraint constraint, ConstraintStrength strength)
Sets the strength of a constraint.
void setConstraintExpression(const QString &expression, const QString &description=QString())
Set the constraint expression for the field.
@ ConstraintOriginProvider
Constraint was set at data provider.
@ ConstraintOriginLayer
Constraint was set by layer.
ConstraintOrigin constraintOrigin(Constraint constraint) const
Returns the origin of a field constraint, or ConstraintOriginNotSet if the constraint is not present ...
Constraint
Constraints which may be present on a field.
@ ConstraintNotNull
Field may not be null.
@ ConstraintUnique
Field must have a unique value.
@ ConstraintExpression
Field has an expression constraint set. See constraintExpression().
void setConstraint(Constraint constraint, ConstraintOrigin origin=ConstraintOriginLayer)
Sets a constraint on the field.
QFlags< Constraint > Constraints
Encapsulate a field in an attribute table or data source.
Definition qgsfield.h:56
QString typeName() const
Gets the field type.
Definition qgsfield.cpp:158
QString name
Definition qgsfield.h:65
int precision
Definition qgsfield.h:62
int length
Definition qgsfield.h:61
QString displayNameWithAlias() const
Returns the name to use when displaying this field and adds the alias in parenthesis if it is defined...
Definition qgsfield.cpp:104
QString alias
Definition qgsfield.h:66
QString customComment
Definition qgsfield.h:71
QString comment
Definition qgsfield.h:64
Container of fields for a vector layer.
Definition qgsfields.h:46
int count
Definition qgsfields.h:50
int size() const
Returns number of items.
QgsField at(int i) const
Returns the field at particular index (must be in range 0..N-1).
Q_INVOKABLE int lookupField(const QString &fieldName) const
Looks up field's index from the field name.
const_iterator constBegin() const noexcept
Returns a const STL-style iterator pointing to the first item in the list.
Contains options to automatically adjust geometries to constraints on a layer.
A geometry is the spatial representation of a feature.
bool isExactlyEqual(const QgsGeometry &geometry, Qgis::GeometryBackend backend=Qgis::GeometryBackend::QGIS) const
Compares the geometry with another geometry using the specified backend.
QgsBox3D boundingBox3D() const
Returns the 3D bounding box of the geometry.
Qgis::GeometryType type
QgsRectangle boundingBox() const
Returns the bounding box of the geometry.
void setMergeLines(bool merge)
Sets whether connected line features with identical label text should be merged prior to generating l...
void setOverlapHandling(Qgis::LabelOverlapHandling handling)
Sets the technique used to handle overlapping labels.
Formats layer metadata into HTML.
void combine(const QgsAbstractMetadataBase *other) override
Combines the metadata from this object with the metadata from an other object.
Line string geometry type, with support for z-dimension and m-values.
Alters the size of rendered diagrams using linear scaling.
static void warning(const QString &msg)
Goes to qWarning.
Models dependencies with or between map layers.
Base class for storage of map layer elevation properties.
static QString typeToString(Qgis::LayerType type)
Converts a map layer type to a string value.
virtual QDomElement writeXml(QDomDocument &doc, const QgsReadWriteContext &context) const
Writes configuration to a DOM element, to be used later with readXml().
static QgsMapLayerLegend * defaultVectorLegend(QgsVectorLayer *vl)
Create new legend implementation for vector layer.
Base class for utility classes that encapsulate information necessary for rendering of map layers.
Base class for storage of map layer selection properties.
Stores style information (renderer, opacity, labeling, diagrams etc.) applicable to a map layer.
Base class for storage of map layer temporal properties.
QString name
Definition qgsmaplayer.h:87
void readStyleManager(const QDomNode &layerNode)
Read style manager's configuration (if any). To be called by subclasses.
void dependenciesChanged()
Emitted when dependencies are changed.
void writeStyleManager(QDomNode &layerNode, QDomDocument &doc) const
Write style manager's configuration (if exists). To be called by subclasses.
QgsMapLayerLegend * legend() const
Can be nullptr.
void editingStopped()
Emitted when edited changes have been successfully written to the data provider.
void recalculateExtents() const
This is used to send a request that any mapcanvas using this layer update its extents.
virtual Q_INVOKABLE QgsRectangle extent() const
Returns the extent of the layer.
QString source() const
Returns the source for the layer.
Q_INVOKABLE QVariant customProperty(const QString &value, const QVariant &defaultValue=QVariant()) const
Read a custom property from layer.
int mBlockStyleChangedSignal
If non-zero, the styleChanged signal should not be emitted.
QString providerType() const
Returns the provider type (provider key) for this layer.
virtual void setExtent3D(const QgsBox3D &box)
Sets the extent.
void removeCustomProperty(const QString &key)
Remove a custom property from layer.
void setBlendMode(QPainter::CompositionMode blendMode)
Set the blending mode used for rendering a layer.
void configChanged()
Emitted whenever the configuration is changed.
void setMinimumScale(double scale)
Sets the minimum map scale (i.e.
static Qgis::DataProviderReadFlags providerReadFlags(const QDomNode &layerNode, QgsMapLayer::ReadFlags layerReadFlags)
Returns provider read flag deduced from layer read flags layerReadFlags and a dom node layerNode that...
void editingStarted()
Emitted when editing on this layer has started.
QgsCoordinateReferenceSystem crs
Definition qgsmaplayer.h:90
friend class QgsVectorLayer
void writeCustomProperties(QDomNode &layerNode, QDomDocument &doc) const
Write custom properties to project file.
virtual int listStylesInDatabase(QStringList &ids, QStringList &names, QStringList &descriptions, QString &msgError)
Lists all the style in db split into related to the layer and not related to.
virtual QString loadDefaultStyle(bool &resultFlag)
Retrieve the default style for this layer if one exists (either as a .qml file on disk or as a record...
void setDataSource(const QString &dataSource, const QString &baseName=QString(), const QString &provider=QString(), bool loadDefaultStyleFlag=false)
Updates the data source of the layer.
void triggerRepaint(bool deferredUpdate=false)
Will advise the map canvas (and any other interested party) that this layer requires to be repainted.
QString crsHtmlMetadata() const
Returns a HTML fragment containing the layer's CRS metadata, for use in the htmlMetadata() method.
void setMaximumScale(double scale)
Sets the maximum map scale (i.e.
QgsLayerMetadata metadata
Definition qgsmaplayer.h:89
QgsMapLayer(Qgis::LayerType type=Qgis::LayerType::Vector, const QString &name=QString(), const QString &source=QString())
Constructor for QgsMapLayer.
Qgis::LayerType type
Definition qgsmaplayer.h:93
QPainter::CompositionMode blendMode() const
Returns the current blending mode for a layer.
virtual void setOpacity(double opacity)
Sets the opacity for the layer, where opacity is a value between 0 (totally transparent) and 1....
void setFlags(QgsMapLayer::LayerFlags flags)
Returns the flags for this layer.
QString publicSource(bool hidePassword=false) const
Gets a version of the internal layer definition that has sensitive bits removed (for example,...
QSet< QgsMapLayerDependency > mDependencies
List of layers that may modify this layer on modification.
void readCustomProperties(const QDomNode &layerNode, const QString &keyStartsWith=QString())
Read custom properties from project file.
virtual void setMetadata(const QgsLayerMetadata &metadata)
Sets the layer's metadata store.
QFlags< StyleCategory > StyleCategories
Q_INVOKABLE void setCustomProperty(const QString &key, const QVariant &value)
Set a custom property for layer.
QString mProviderKey
Data provider key (name of the data provider).
QgsCoordinateTransformContext transformContext() const
Returns the layer data provider coordinate transform context or a default transform context if the la...
void styleChanged()
Signal emitted whenever a change affects the layer's style.
QUndoStack * undoStack()
Returns pointer to layer's undo stack.
std::unique_ptr< QgsDataProvider > mPreloadedProvider
Optionally used when loading a project, it is released when the layer is effectively created.
void rendererChanged()
Signal emitted when renderer is changed.
virtual QgsError error() const
Gets current status error.
void setScaleBasedVisibility(bool enabled)
Sets whether scale based visibility is enabled for the layer.
void dataSourceChanged()
Emitted whenever the layer's data source has been changed.
QgsMapLayer::LayerFlags flags
Definition qgsmaplayer.h:99
bool hasScaleBasedVisibility() const
Returns whether scale based visibility is enabled for the layer.
virtual QString getStyleFromDatabase(const QString &styleId, QString &msgError)
Returns the named style corresponding to style id provided.
void emitStyleChanged()
Triggers an emission of the styleChanged() signal.
void dataChanged()
Data of layer changed.
void willBeDeleted()
Emitted in the destructor when the layer is about to be deleted, but it is still in a perfectly valid...
virtual QgsBox3D extent3D() const
Returns the 3D extent of the layer.
virtual QgsMapLayer * clone() const =0
Returns a new instance equivalent to this one except for the id which is still unique.
void setName(const QString &name)
Set the display name of the layer.
virtual void setExtent(const QgsRectangle &rect)
Sets the extent.
virtual void resolveReferences(QgsProject *project)
Resolve references to other layers (kept as layer IDs after reading XML) into layer objects.
QString mDataSource
Data source description string, varies by layer type.
void setMapTipsEnabled(bool enabled)
Enable or disable map tips for this layer.
@ FlagReadExtentFromXml
Read extent from xml and skip get extent from provider.
@ FlagForceReadOnly
Force open as read only.
@ FlagDontResolveLayers
Don't resolve layer paths or create data providers for layers.
void setValid(bool valid)
Sets whether layer is valid or not.
void readCommonStyle(const QDomElement &layerElement, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories)
Read style data common to all layer types.
QgsMapLayer::ReadFlags mReadFlags
Read flags. It's up to the subclass to respect these when restoring state from XML.
double minimumScale() const
Returns the minimum map scale (i.e.
void repaintRequested(bool deferredUpdate=false)
By emitting this signal the layer tells that either appearance or content have been changed and any v...
void setMapTipTemplate(const QString &mapTipTemplate)
The mapTip is a pretty, html representation for feature information.
Q_INVOKABLE QStringList customPropertyKeys() const
Returns list of all keys within custom properties.
QgsProject * project() const
Returns the parent project if this map layer is added to a project.
bool mapTipsEnabled
Definition qgsmaplayer.h:97
void setLegend(QgsMapLayerLegend *legend)
Assign a legend controller to the map layer.
double opacity
Definition qgsmaplayer.h:95
bool mValid
Indicates if the layer is valid and can be drawn.
@ GeometryOptions
Geometry validation configuration.
@ AttributeTable
Attribute table settings: choice and order of columns, conditional styling.
@ LayerConfiguration
General configuration: identifiable, removable, searchable, display expression, read-only.
@ Symbology
Symbology.
@ MapTips
Map tips.
@ Rendering
Rendering: scale visibility, simplify method, opacity.
@ Relations
Relations.
@ CustomProperties
Custom properties (by plugins for instance).
@ Actions
Actions.
@ Forms
Feature form.
@ Fields
Aliases, widgets, WMS/WFS, expressions, constraints, virtual fields.
@ Legend
Legend settings.
@ Diagrams
Diagrams.
@ Labeling
Labeling.
void layerModified()
Emitted when modifications has been done on layer.
void setProviderType(const QString &providerType)
Sets the providerType (provider key).
QString customPropertyHtmlMetadata() const
Returns an HTML fragment containing custom property information, for use in the htmlMetadata() method...
QString generalHtmlMetadata() const
Returns an HTML fragment containing general metadata information, for use in the htmlMetadata() metho...
void writeCommonStyle(QDomElement &layerElement, QDomDocument &document, const QgsReadWriteContext &context, StyleCategories categories=AllStyleCategories) const
Write style data common to all layer types.
double maximumScale() const
Returns the maximum map scale (i.e.
QString mapTipTemplate
Definition qgsmaplayer.h:96
bool mShouldValidateCrs
true if the layer's CRS should be validated and invalid CRSes are not permitted.
void setCrs(const QgsCoordinateReferenceSystem &srs, bool emitSignal=true)
Sets layer's spatial reference system.
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).
QFile * localFile(const QString &filePathOrUrl)
Returns a QFile from a local file or to a temporary file previously fetched by the registry.
An interface for classes which can visit various object entity (e.g.
virtual bool visitEmbeddedScript(const QgsEmbeddedScriptEntity &entity, const QgsObjectVisitorContext &context)
Called when the visitor will visit an embedded script entity.
A QgsObjectEntityVisitorInterface context object.
static QgsExpression * expressionFromOgcFilter(const QDomElement &element, QgsVectorLayer *layer=nullptr)
Parse XML with OGC filter into QGIS expression.
static Qgis::BlendMode getBlendModeEnum(QPainter::CompositionMode blendMode)
Returns a Qgis::BlendMode corresponding to a QPainter::CompositionMode.
static QPainter::CompositionMode getCompositionMode(Qgis::BlendMode blendMode)
Returns a QPainter::CompositionMode corresponding to a Qgis::BlendMode.
Contains settings for how a map layer will be labeled.
double yOffset
Vertical offset of label.
const QgsLabelPlacementSettings & placementSettings() const
Returns the label placement settings.
double maxCurvedCharAngleIn
Maximum angle between inside curved label characters (valid range 20.0 to 60.0).
void setFormat(const QgsTextFormat &format)
Sets the label text formatting settings, e.g., font settings, buffer settings, etc.
double xOffset
Horizontal offset of label.
Qgis::LabelPlacement placement
Label placement mode.
double angleOffset
Label rotation, in degrees clockwise.
double maxCurvedCharAngleOut
Maximum angle between outside curved label characters (valid range -20.0 to -95.0).
Qgis::RenderUnit offsetUnits
Units for offsets of label.
bool isExpression
true if this label is made from a expression string, e.g., FieldName || 'mm'
const QgsLabelLineSettings & lineSettings() const
Returns the label line settings, which contain settings related to how the label engine places and fo...
Qgis::UpsideDownLabelHandling upsidedownLabels
Controls whether upside down labels are displayed and how they are handled.
QString fieldName
Name of field (or an expression) to use for label text.
Represents a 2D point.
Definition qgspointxy.h:62
Point geometry type, with support for z-dimension and m-values.
Definition qgspoint.h:53
Encapsulates properties and constraints relating to fetching elevation profiles from different source...
virtual QString translate(const QString &context, const QString &sourceText, const char *disambiguation=nullptr, int n=-1) const =0
Translates a string using the Qt QTranslator mechanism.
Encapsulates a QGIS project, including sets of map layers and their styles, layouts,...
Definition qgsproject.h:114
QgsRelationManager * relationManager
Definition qgsproject.h:125
bool commitChanges(QStringList &commitErrors, bool stopEditing=true, QgsVectorLayer *vectorLayer=nullptr)
Attempts to commit to the underlying data provider any buffered changes made since the last to call t...
static QgsProject * instance()
Returns the QgsProject singleton instance.
bool rollBack(QStringList &rollbackErrors, bool stopEditing=true, QgsVectorLayer *vectorLayer=nullptr)
Stops a current editing operation on vectorLayer and discards any uncommitted edits.
bool startEditing(QgsVectorLayer *vectorLayer=nullptr)
Makes the layer editable.
QMap< QString, QgsMapLayer * > mapLayers(const bool validOnly=false) const
Returns a map of all registered layers by layer ID.
A grouped map of multiple QgsProperty objects, each referenced by an integer key value.
void setProperty(int key, const QgsProperty &property)
Adds a property to the collection and takes ownership of it.
Definition for a property.
Definition qgsproperty.h:47
@ Double
Double value (including negative values).
Definition qgsproperty.h:56
@ Boolean
Boolean value.
Definition qgsproperty.h:52
static QgsProperty fromField(const QString &fieldName, bool isActive=true)
Returns a new FieldBasedProperty created from the specified field name.
QString absoluteToRelativeUri(const QString &providerKey, const QString &uri, const QgsReadWriteContext &context) const
Converts absolute path(s) to relative path(s) in the given provider-specific URI.
static QgsProviderRegistry * instance(const QString &pluginPath=QString())
Means of accessing canonical single instance.
QString relativeToAbsoluteUri(const QString &providerKey, const QString &uri, const QgsReadWriteContext &context) const
Converts relative path(s) to absolute path(s) in the given provider-specific URI.
Allows entering a context category and takes care of leaving this category on deletion of the class.
A container for the context for various read/write operations on objects.
QgsReadWriteContextCategoryPopper enterCategory(const QString &category, const QString &details=QString()) const
Push a category to the stack.
const QgsProjectTranslator * projectTranslator() const
Returns the project translator.
QgsCoordinateTransformContext transformContext() const
Returns data provider coordinate transform context.
const QgsPathResolver & pathResolver() const
Returns path resolver for conversion between relative and absolute paths.
A rectangle specified with double values.
Q_INVOKABLE QString toString(int precision=16) const
Returns a string representation of form xmin,ymin : xmax,ymax Coordinates will be rounded to the spec...
double xMinimum
double yMinimum
double xMaximum
void set(const QgsPointXY &p1, const QgsPointXY &p2, bool normalize=true)
Sets the rectangle from two QgsPoints.
double yMaximum
void combineExtentWith(const QgsRectangle &rect)
Expands the rectangle so that it covers both the original rectangle and the given rectangle.
void normalize()
Normalize the rectangle so it has non-negative width/height.
void setNull()
Mark a rectangle as being null (holding no spatial information).
QList< QgsRelation > referencedRelations(const QgsVectorLayer *layer=nullptr) const
Gets all relations where this layer is the referenced part (i.e.
void relationsLoaded()
Emitted when the relations were loaded after reading a project.
Represents a relationship between two vector layers.
Definition qgsrelation.h:42
Contains information about the context of a rendering operation.
double rendererScale() const
Returns the renderer map scale.
bool useRenderingOptimization() const
Returns true if the rendering optimization (geometry simplification) can be executed.
void appendChild(QgsRuleBasedLabeling::Rule *rule)
add child rule, take ownership, sets this as parent
A boolean settings entry.
A double settings entry.
A template class for enum and flag settings entry.
static QgsSettingsTreeNode * sTreeQgis
Renders the diagrams for all features with the same settings.
Holds SLD export options and other information related to SLD export of a QGIS layer style.
void setExtraProperties(const QVariantMap &properties)
Sets the open ended set of properties that can drive/inform the SLD encoding.
QVariantMap extraProperties() const
Returns the open ended set of properties that can drive/inform the SLD encoding.
Renders diagrams using mixed diagram render types.
Manages stored expressions regarding creation, modification and storing in the project.
An interface for classes which can visit style entity (e.g.
static double rendererFrameRate(const QgsFeatureRenderer *renderer)
Calculates the frame rate (in frames per second) at which the given renderer must be redrawn.
static QgsStringMap getSvgParameterList(QDomElement &element)
static void mergeScaleDependencies(double mScaleMinDenom, double mScaleMaxDenom, QVariantMap &props)
Merges the local scale limits, if any, with the ones already in the map, if any.
static bool fillFromSld(QDomElement &element, Qt::BrushStyle &brushStyle, QColor &color)
static Qgis::RenderUnit decodeSldUom(const QString &str, double *scaleFactor=nullptr)
Decodes a SLD unit of measure string to a render unit.
long addTask(QgsTask *task, int priority=0)
Adds a task to the manager.
void taskCompleted()
Will be emitted by task to indicate its successful completion.
void taskTerminated()
Will be emitted by task if it has terminated for any reason other then completion (e....
void setColor(const QColor &color)
Sets the color for the buffer.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units used for the buffer size.
void setEnabled(bool enabled)
Sets whether the text buffer will be drawn.
void setSize(double size)
Sets the size of the buffer.
void setColor(const QColor &color)
Sets the color that text will be rendered in.
void setSize(double size)
Sets the size for rendered text.
void setFont(const QFont &font)
Sets the font used for rendering text.
void setSizeUnit(Qgis::RenderUnit unit)
Sets the units for the size of rendered text.
void setBuffer(const QgsTextBufferSettings &bufferSettings)
Sets the text's buffer settings.
static QString threadDescription(QThread *thread)
Returns a descriptive identifier for a thread.
Allows creation of a multi-layer database-side transaction.
void dirtied(const QString &sql, const QString &name)
Emitted if a sql query is executed and the underlying data is modified.
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.
static const int EditingCapabilities
Bitmask of all provider's editing capabilities.
virtual QString geometryColumnName() const
Returns the name of the column storing geometry, if applicable.
void raiseError(const QString &msg) const
Signals an error in this provider.
virtual void handlePostCloneOperations(QgsVectorDataProvider *source)
Handles any post-clone operations required after this vector data provider was cloned from the source...
virtual QgsTransaction * transaction() const
Returns the transaction this data provider is included in, if any.
void committedAttributesDeleted(const QString &layerId, const QgsAttributeList &deletedAttributes)
Emitted after attribute deletion has been committed to the layer.
void committedAttributeValuesChanges(const QString &layerId, const QgsChangedAttributesMap &changedAttributesValues)
Emitted after feature attribute value changes have been committed to the layer.
void geometryChanged(QgsFeatureId fid, const QgsGeometry &geom)
Emitted when a feature's geometry is changed.
void committedAttributesAdded(const QString &layerId, const QList< QgsField > &addedAttributes)
Emitted after attribute addition has been committed to the layer.
void committedFeaturesAdded(const QString &layerId, const QgsFeatureList &addedFeatures)
Emitted after feature addition has been committed to the layer.
void featureDeleted(QgsFeatureId fid)
Emitted when a feature was deleted from the buffer.
void attributeAdded(int idx)
Emitted when an attribute was added to the buffer.
void committedGeometriesChanges(const QString &layerId, const QgsGeometryMap &changedGeometries)
Emitted after feature geometry changes have been committed to the layer.
void attributeValueChanged(QgsFeatureId fid, int idx, const QVariant &value)
Emitted when a feature's attribute value has been changed.
void attributeDeleted(int idx)
Emitted when an attribute was deleted from the buffer.
void featureAdded(QgsFeatureId fid)
Emitted when a feature has been added to the buffer.
void layerModified()
Emitted when modifications has been done on layer.
void committedFeaturesRemoved(const QString &layerId, const QgsFeatureIds &deletedFeatureIds)
Emitted after feature removal has been committed to the layer.
Contains utility functions for editing vector layers.
int translateFeature(QgsFeatureId featureId, double dx, double dy)
Translates feature by dx, dy.
bool insertVertex(double x, double y, QgsFeatureId atFeatureId, int beforeVertex)
Insert a new vertex before the given vertex number, in the given ring, item (first number is index 0)...
Q_DECL_DEPRECATED Qgis::GeometryOperationResult addPart(const QVector< QgsPointXY > &ring, QgsFeatureId featureId)
Adds a new part polygon to a multipart feature.
Qgis::VectorEditResult deleteVertex(QgsFeatureId featureId, int vertex)
Deletes a vertex from a feature.
int addTopologicalPoints(const QgsGeometry &geom)
Adds topological points for every vertex of the geometry.
Q_DECL_DEPRECATED Qgis::GeometryOperationResult splitParts(const QVector< QgsPointXY > &splitLine, bool topologicalEditing=false)
Splits parts cut by the given line.
Q_DECL_DEPRECATED Qgis::GeometryOperationResult splitFeatures(const QVector< QgsPointXY > &splitLine, bool topologicalEditing=false)
Splits features cut by the given line.
bool moveVertex(double x, double y, QgsFeatureId atFeatureId, int atVertex)
Moves the vertex at the given position number, ring and item (first number is index 0),...
Q_DECL_DEPRECATED Qgis::GeometryOperationResult addRing(const QVector< QgsPointXY > &ring, const QgsFeatureIds &targetFeatureIds=QgsFeatureIds(), QgsFeatureId *modifiedFeatureId=nullptr)
Adds a ring to polygon/multipolygon features.
Vector layer specific subclass of QgsMapLayerElevationProperties.
QgsVectorLayerElevationProperties * clone() const override
Creates a clone of the properties.
Counts the features in a QgsVectorLayer in task.
A feature iterator which iterates over features from a QgsVectorLayer.
Manages joined fields for a vector layer.
bool containsJoins() const
Quick way to test if there is any join at all.
void joinedFieldsChanged()
Emitted whenever the list of joined fields changes (e.g.
Defines left outer join from our vector layer to some other vector layer.
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.
Implementation of QgsAbstractProfileGenerator for vector layers.
Implementation of threaded rendering for vector layers.
Implementation of layer selection properties for vector layers.
QgsVectorLayerSelectionProperties * clone() const override
Creates a clone of the properties.
Basic implementation of the labeling interface.
Implementation of map layer temporal properties for vector layers.
Contains settings which reflect the context in which vector layer tool operations should be considere...
QgsExpressionContext * expressionContext() const
Returns the optional expression context used by the vector layer tools.
static QString guessFriendlyIdentifierField(const QgsFields &fields, bool *foundFriendly=nullptr)
Given a set of fields, attempts to pick the "most useful" field for user-friendly identification of f...
static QgsFeatureIds filterValidFeatureIds(const QgsVectorLayer *layer, const QgsFeatureIds &featureIds)
Filters a set of feature IDs to only include those that exist in the layer.
Represents a vector layer which manages a vector based dataset.
void setLabeling(QgsAbstractVectorLayerLabeling *labeling)
Sets labeling configuration.
Q_INVOKABLE QString attributeDisplayName(int index) const
Convenience function that returns the attribute alias if defined or the field name else.
bool writeSymbology(QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories) const final
Write the style for the layer into the document provided.
QSet< QgsMapLayerDependency > dependencies() const final
Gets the list of dependencies.
int addExpressionField(const QString &exp, const QgsField &fld)
Add a new field which is calculated by the expression specified.
Q_INVOKABLE void selectByRect(const QgsRectangle &rect, Qgis::SelectBehavior behavior=Qgis::SelectBehavior::SetSelection)
Selects features found within the search rectangle (in layer's coordinates).
void committedFeaturesAdded(const QString &layerId, const QgsFeatureList &addedFeatures)
Emitted when features are added to the provider if not in transaction mode.
Q_DECL_DEPRECATED Qgis::GeometryOperationResult addPart(const QList< QgsPointXY > &ring)
Adds a new part polygon to a multipart feature.
static const QgsSettingsEntryEnumFlag< Qgis::VectorRenderingSimplificationFlags > * settingsSimplifyDrawingHints
void featureBlendModeChanged(QPainter::CompositionMode blendMode)
Signal emitted when setFeatureBlendMode() is called.
bool labelsEnabled() const
Returns whether the layer contains labels which are enabled and should be drawn.
bool isModified() const override
Returns true if the provider has been modified since the last commit.
bool isEditable() const final
Returns true if the provider is in editing mode.
void addFeatureRendererGenerator(QgsFeatureRendererGenerator *generator)
Adds a new feature renderer generator to the layer.
Q_DECL_DEPRECATED void setExcludeAttributesWfs(const QSet< QString > &att)
A set of attributes that are not advertised in WFS requests with QGIS server.
Q_INVOKABLE bool deleteSelectedFeatures(int *deletedCount=nullptr, QgsVectorLayer::DeleteContext *context=nullptr)
Deletes the selected features.
Q_INVOKABLE void removeFieldAlias(int index)
Removes an alias (a display name) for attributes to display in dialogs.
void setAuxiliaryLayer(QgsAuxiliaryLayer *layer=nullptr)
Sets the current auxiliary layer.
Q_INVOKABLE QVariant minimumValue(int index) const final
Returns the minimum value for an attribute column or an invalid variant in case of error.
void beforeRemovingExpressionField(int idx)
Will be emitted, when an expression field is going to be deleted from this vector layer.
Q_INVOKABLE bool deleteFeatures(const QgsFeatureIds &fids, QgsVectorLayer::DeleteContext *context=nullptr)
Deletes a set of features from the layer (but does not commit it).
void committedGeometriesChanges(const QString &layerId, const QgsGeometryMap &changedGeometries)
Emitted when geometry changes are saved to the provider if not in transaction mode.
void beforeCommitChanges(bool stopEditing)
Emitted before changes are committed to the data provider.
Q_INVOKABLE bool startEditing()
Makes the layer editable.
void setFieldConfigurationFlags(int index, Qgis::FieldConfigurationFlags flags)
Sets the configuration flags of the field at given index.
Q_INVOKABLE QVariant maximumValue(int index) const final
Returns the maximum value for an attribute column or an invalid variant in case of error.
QMap< QgsFieldConstraints::Constraint, QgsFieldConstraints::ConstraintStrength > fieldConstraintsAndStrength(int fieldIndex) const
Returns a map of constraint with their strength for a specific field of the layer.
bool addJoin(const QgsVectorLayerJoinInfo &joinInfo)
Joins another vector layer to this layer.
QgsVectorLayer(const QString &path=QString(), const QString &baseName=QString(), const QString &providerLib="ogr", const QgsVectorLayer::LayerOptions &options=QgsVectorLayer::LayerOptions())
Constructor - creates a vector layer.
QgsMapLayerTemporalProperties * temporalProperties() override
Returns the layer's temporal properties.
QgsExpressionContext createExpressionContext() const final
This method needs to be reimplemented in all classes which implement this interface and return an exp...
Q_INVOKABLE bool changeAttributeValue(QgsFeatureId fid, int field, const QVariant &newValue, const QVariant &oldValue=QVariant(), bool skipDefaultValues=false, QgsVectorLayerToolsContext *context=nullptr)
Changes an attribute value for a feature (but does not immediately commit the changes).
Q_DECL_DEPRECATED Qgis::GeometryOperationResult splitFeatures(const QVector< QgsPointXY > &splitLine, bool topologicalEditing=false)
Splits features cut by the given line.
QgsDefaultValue defaultValueDefinition(int index) const
Returns the definition of the expression used when calculating the default value for a field.
QgsVectorLayerFeatureCounter * countSymbolFeatures(bool storeSymbolFids=false)
Count features for symbols.
Q_INVOKABLE QString attributeCustomComment(int index) const
Returns the custom comment for the field.
QPainter::CompositionMode featureBlendMode() const
Returns the current blending mode for features.
QString constraintExpression(int index) const
Returns the constraint expression for for a specified field index, if set.
Q_INVOKABLE bool addAttribute(const QgsField &field)
Add an attribute field (but does not commit it) returns true if the field was added.
void attributeAdded(int idx)
Will be emitted, when a new attribute has been added to this vector layer.
QString capabilitiesString() const
Capabilities for this layer, comma separated and translated.
void deselect(QgsFeatureId featureId)
Deselects feature by its ID.
void allowCommitChanged()
Emitted whenever the allowCommit() property of this layer changes.
friend class QgsVectorLayerEditBuffer
void editCommandStarted(const QString &text)
Signal emitted when a new edit command has been started.
void updateFields()
Will regenerate the fields property of this layer by obtaining all fields from the dataProvider,...
QgsBox3D extent3D() const final
Returns the 3D extent of the layer.
const QgsDiagramLayerSettings * diagramLayerSettings() const
void setFieldConstraint(int index, QgsFieldConstraints::Constraint constraint, QgsFieldConstraints::ConstraintStrength strength=QgsFieldConstraints::ConstraintStrengthHard)
Sets a constraint for a specified field index.
Q_INVOKABLE void invertSelectionInRectangle(const QgsRectangle &rect)
Inverts selection of features found within the search rectangle (in layer's coordinates).
bool loadAuxiliaryLayer(const QgsAuxiliaryStorage &storage, const QString &key=QString())
Loads the auxiliary layer for this vector layer.
bool insertVertex(double x, double y, QgsFeatureId atFeatureId, int beforeVertex)
Inserts a new vertex before the given vertex number, in the given ring, item (first number is index 0...
QgsAbstractProfileGenerator * createProfileGenerator(const QgsProfileRequest &request) override
Given a profile request, returns a new profile generator ready for generating elevation profiles.
bool readSymbology(const QDomNode &layerNode, QString &errorMessage, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories) final
Read the symbology for the current layer from the DOM node supplied.
Q_INVOKABLE QgsRectangle boundingBoxOfSelected() const
Returns the bounding box of the selected features. If there is no selection, QgsRectangle(0,...
bool isSpatial() const final
Returns true if this is a geometry layer and false in case of NoGeometry (table only) or UnknownGeome...
Q_INVOKABLE QgsFeatureList selectedFeatures() const
Returns a copy of the user-selected features.
QString expressionField(int index) const
Returns the expression used for a given expression field.
void removeFeatureRendererGenerator(const QString &id)
Removes the feature renderer with matching id from the layer.
Q_INVOKABLE bool deleteFeature(QgsFeatureId fid, QgsVectorLayer::DeleteContext *context=nullptr)
Deletes a feature from the layer (but does not commit it).
friend class QgsVectorLayerEditPassthrough
void setSimplifyMethod(const QgsVectorSimplifyMethod &simplifyMethod)
Sets the simplification settings for fast rendering of features.
void editCommandDestroyed()
Signal emitted, when an edit command is destroyed.
QVariant aggregate(Qgis::Aggregate aggregate, const QString &fieldOrExpression, const QgsAggregateCalculator::AggregateParameters &parameters=QgsAggregateCalculator::AggregateParameters(), QgsExpressionContext *context=nullptr, bool *ok=nullptr, QgsFeatureIds *fids=nullptr, QgsFeedback *feedback=nullptr, QString *error=nullptr) const
Calculates an aggregated value from the layer's features.
QgsRectangle sourceExtent() const final
Returns the extent of all geometries from the source.
QgsFieldConstraints::Constraints fieldConstraints(int fieldIndex) const
Returns any constraints which are present for a specified field index.
static const QgsSettingsEntryEnumFlag< Qgis::VectorSimplificationAlgorithm > * settingsSimplifyAlgorithm
Q_DECL_DEPRECATED QSet< QString > excludeAttributesWms() const
A set of attributes that are not advertised in WMS requests with QGIS server.
QgsFeatureIds symbolFeatureIds(const QString &legendKey) const
Ids of features rendered with specified legend key.
void removeFieldConstraint(int index, QgsFieldConstraints::Constraint constraint)
Removes a constraint for a specified field index.
int selectedFeatureCount() const
Returns the number of features that are selected in this layer.
void featuresDeleted(const QgsFeatureIds &fids)
Emitted when features have been deleted.
Qgis::VectorLayerTypeFlags vectorLayerTypeFlags() const
Returns the vector layer type flags.
void setLabelsEnabled(bool enabled)
Sets whether labels should be enabled for the layer.
void subsetStringChanged()
Emitted when the layer's subset string has changed.
bool setDependencies(const QSet< QgsMapLayerDependency > &layers) final
Sets the list of dependencies.
QgsAuxiliaryLayer * auxiliaryLayer()
Returns the current auxiliary layer.
void setCoordinateSystem()
Setup the coordinate system transformation for the layer.
void committedFeaturesRemoved(const QString &layerId, const QgsFeatureIds &deletedFeatureIds)
Emitted when features are deleted from the provider if not in transaction mode.
void setFieldMergePolicy(int index, Qgis::FieldDomainMergePolicy policy)
Sets a merge policy for the field with the specified index.
void updateExpressionField(int index, const QString &exp)
Changes the expression used to define an expression based (virtual) field.
Q_INVOKABLE void selectByExpression(const QString &expression, Qgis::SelectBehavior behavior=Qgis::SelectBehavior::SetSelection, QgsExpressionContext *context=nullptr)
Selects matching features using an expression.
static const QgsSettingsEntryDouble * settingsSimplifyMaxScale
void reload() final
Synchronises with changes in the datasource.
long long featureCount() const final
Returns feature count including changes which have not yet been committed If you need only the count ...
~QgsVectorLayer() override
QgsRectangle extent() const final
Returns the extent of the layer.
void endEditCommand()
Finish edit command and add it to undo/redo stack.
void destroyEditCommand()
Destroy active command and reverts all changes in it.
bool isAuxiliaryField(int index, int &srcIndex) const
Returns true if the field comes from the auxiliary layer, false otherwise.
bool hasMapTips() const final
Returns true if the layer contains map tips.
Q_INVOKABLE Qgis::WkbType wkbType() const final
Returns the WKBType or WKBUnknown in case of error.
void setExtent(const QgsRectangle &rect) final
Sets the extent.
QList< QgsRelation > referencingRelations(int idx) const
Returns the layer's relations, where the foreign key is on this layer.
Q_DECL_DEPRECATED QSet< QString > excludeAttributesWfs() const
A set of attributes that are not advertised in WFS requests with QGIS server.
Q_DECL_DEPRECATED Qgis::GeometryOperationResult splitParts(const QVector< QgsPointXY > &splitLine, bool topologicalEditing=false)
Splits parts cut by the given line.
void setDefaultValueDefinition(int index, const QgsDefaultValue &definition)
Sets the definition of the expression to use when calculating the default value for a field.
bool diagramsEnabled() const
Returns whether the layer contains diagrams which are enabled and should be drawn.
void setAllowCommit(bool allowCommit)
Controls, if the layer is allowed to commit changes.
QgsBox3D sourceExtent3D() const final
Returns the 3D extent of all geometries from the source.
void symbolFeatureCountMapChanged()
Emitted when the feature count for symbols on this layer has been recalculated.
Q_INVOKABLE const QgsFeatureIds & selectedFeatureIds() const
Returns a list of the selected features IDs in this layer.
Qgis::VectorEditResult deleteVertex(QgsFeatureId featureId, int vertex)
Deletes a vertex from a feature.
Qgis::FeatureAvailability hasFeatures() const final
Determines if this vector layer has features.
void setFeatureBlendMode(QPainter::CompositionMode blendMode)
Sets the blending mode used for rendering each feature.
QString constraintDescription(int index) const
Returns the descriptive name for the constraint expression for a specified field index.
void writeCustomSymbology(QDomElement &element, QDomDocument &doc, QString &errorMessage) const
Signal emitted whenever the symbology (QML-file) for this layer is being written.
const QgsAbstractVectorLayerLabeling * labeling() const
Access to const labeling configuration.
bool writeStyle(QDomNode &node, QDomDocument &doc, QString &errorMessage, const QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories) const final
Write just the symbology information for the layer into the document.
void setProviderEncoding(const QString &encoding)
Sets the text encoding of the data provider.
Q_DECL_DEPRECATED bool writeSld(QDomNode &node, QDomDocument &doc, QString &errorMessage, const QVariantMap &props=QVariantMap()) const
Writes the symbology of the layer into the document provided in SLD 1.1 format.
void setDisplayExpression(const QString &displayExpression)
Set the preview expression, used to create a human readable preview string.
virtual Q_INVOKABLE bool deleteAttribute(int attr)
Deletes an attribute field (but does not commit it).
static const QgsSettingsEntryBool * settingsSimplifyLocal
QString loadDefaultStyle(bool &resultFlag) final
Retrieve the default style for this layer if one exists (either as a .qml file on disk or as a record...
bool simplifyDrawingCanbeApplied(const QgsRenderContext &renderContext, Qgis::VectorRenderingSimplificationFlag simplifyHint) const
Returns whether the VectorLayer can apply the specified simplification hint.
QString htmlMetadata() const final
Obtain a formatted HTML string containing assorted metadata for this layer.
QgsMapLayerElevationProperties * elevationProperties() override
Returns the layer's elevation properties.
bool removeJoin(const QString &joinLayerId)
Removes a vector layer join.
void setRenderer(QgsFeatureRenderer *r)
Sets the feature renderer which will be invoked to represent this layer in 2D map views.
Q_INVOKABLE void selectAll()
Select all the features.
QStringList commitErrors() const
Returns a list containing any error messages generated when attempting to commit changes to the layer...
QString decodedSource(const QString &source, const QString &provider, const QgsReadWriteContext &context) const final
Called by readLayerXML(), used by derived classes to decode provider's specific data source from proj...
QString storageType() const
Returns the permanent storage type for this layer as a friendly name.
QString encodedSource(const QString &source, const QgsReadWriteContext &context) const final
Called by writeLayerXML(), used by derived classes to encode provider's specific data source to proje...
bool readExtentFromXml() const
Returns true if the extent is read from the XML document when data source has no metadata,...
QString dataComment() const
Returns a description for this layer as defined in the data provider.
bool accept(QgsStyleEntityVisitorInterface *visitor) const override
QgsExpressionContextScope * createExpressionContextScope() const final
This method needs to be reimplemented in all classes which implement this interface and return an exp...
QgsGeometryOptions * geometryOptions() const
Configuration and logic to apply automatically on any edit happening on this layer.
QgsStringMap attributeAliases() const
Returns a map of field name to attribute alias.
Q_INVOKABLE int translateFeature(QgsFeatureId featureId, double dx, double dy)
Translates feature by dx, dy.
virtual void updateExtents(bool force=false)
Update the extents for the layer.
void attributeDeleted(int idx)
Will be emitted, when an attribute has been deleted from this vector layer.
QgsFeatureRenderer * renderer()
Returns the feature renderer used for rendering the features in the layer in 2D map views.
void beforeEditingStarted()
Emitted before editing on this layer is started.
void committedAttributeValuesChanges(const QString &layerId, const QgsChangedAttributesMap &changedAttributesValues)
Emitted when attribute value changes are saved to the provider if not in transaction mode.
QgsStringMap attributeCustomComments() const
Returns a map of all the custom comments.
void committedAttributesAdded(const QString &layerId, const QList< QgsField > &addedAttributes)
Emitted when attributes are added to the provider if not in transaction mode.
void setEditFormConfig(const QgsEditFormConfig &editFormConfig)
Sets the editFormConfig (configuration) of the form used to represent this vector layer.
Qgis::FieldConfigurationFlags fieldConfigurationFlags(int index) const
Returns the configuration flags of the field at given index.
void committedAttributesDeleted(const QString &layerId, const QgsAttributeList &deletedAttributes)
Emitted when attributes are deleted from the provider if not in transaction mode.
QString displayExpression
void displayExpressionChanged()
Emitted when the display expression changes.
QgsFeatureIterator getSelectedFeatures(QgsFeatureRequest request=QgsFeatureRequest()) const
Returns an iterator of the selected features.
void setEditorWidgetSetup(int index, const QgsEditorWidgetSetup &setup)
Sets the editor widget setup for the field at the specified index.
void setConstraintExpression(int index, const QString &expression, const QString &description=QString())
Sets the constraint expression for the specified field index.
Q_INVOKABLE bool rollBack(bool deleteBuffer=true)
Stops a current editing operation and discards any uncommitted edits.
QString sourceName() const final
Returns a friendly display name for the source.
bool readStyle(const QDomNode &node, QString &errorMessage, QgsReadWriteContext &context, QgsMapLayer::StyleCategories categories=QgsMapLayer::AllStyleCategories) final
Read the style for the current layer from the DOM node supplied.
bool updateFeature(QgsFeature &feature, bool skipDefaultValues=false)
Updates an existing feature in the layer, replacing the attributes and geometry for the feature with ...
Q_INVOKABLE bool commitChanges(bool stopEditing=true)
Attempts to commit to the underlying data provider any buffered changes made since the last to call t...
void setFieldConfigurationFlag(int index, Qgis::FieldConfigurationFlag flag, bool active)
Sets the given configuration flag for the field at given index to be active or not.
void setFieldDuplicatePolicy(int index, Qgis::FieldDuplicatePolicy policy)
Sets a duplicate policy for the field with the specified index.
bool setReadOnly(bool readonly=true)
Makes layer read-only (editing disabled) or not.
void editFormConfigChanged()
Will be emitted whenever the edit form configuration of this layer changes.
Q_INVOKABLE void modifySelection(const QgsFeatureIds &selectIds, const QgsFeatureIds &deselectIds)
Modifies the current selection on this layer.
void setWeakRelations(const QList< QgsWeakRelation > &relations)
Sets the layer's weak relations.
void resolveReferences(QgsProject *project) final
Resolves references to other layers (kept as layer IDs after reading XML) into layer objects.
void reselect()
Reselects the previous set of selected features.
void select(QgsFeatureId featureId)
Selects feature by its ID.
QgsEditorWidgetSetup editorWidgetSetup(int index) const
Returns the editor widget setup for the field at the specified index.
bool readSld(const QDomNode &node, QString &errorMessage) final
QgsMapLayerRenderer * createMapRenderer(QgsRenderContext &rendererContext) final
Returns new instance of QgsMapLayerRenderer that will be used for rendering of given context.
Q_INVOKABLE void setFieldCustomComment(int index, const QString &customCommentString)
Sets the custom comment for the field.
QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const final
Queries the layer for features specified in request.
void setReadExtentFromXml(bool readExtentFromXml)
Flag allowing to indicate if the extent has to be read from the XML document when data source has no ...
void afterCommitChanges()
Emitted after changes are committed to the data provider.
QgsVectorLayer * clone() const override
Returns a new instance equivalent to this one.
QgsAttributeTableConfig attributeTableConfig() const
Returns the attribute table configuration object.
QgsActionManager * actions()
Returns all layer actions defined on this layer.
QStringList uniqueStringsMatching(int index, const QString &substring, int limit=-1, QgsFeedback *feedback=nullptr) const
Returns unique string values of an attribute which contain a specified subset string.
void raiseError(const QString &msg)
Signals an error related to this vector layer.
void editCommandEnded()
Signal emitted, when an edit command successfully ended.
void supportsEditingChanged()
Emitted when the read only state or the data provider of this layer is changed.
QgsCoordinateReferenceSystem sourceCrs() const final
Returns the coordinate reference system for features in the source.
void readOnlyChanged()
Emitted when the read only state of this layer is changed.
void removeExpressionField(int index)
Removes an expression field.
void setTransformContext(const QgsCoordinateTransformContext &transformContext) override
Sets the coordinate transform context to transformContext.
void attributeValueChanged(QgsFeatureId fid, int idx, const QVariant &value)
Emitted whenever an attribute value change is done in the edit buffer.
static Q_DECL_DEPRECATED void drawVertexMarker(double x, double y, QPainter &p, Qgis::VertexMarkerType type, int vertexSize)
Draws a vertex symbol at (screen) coordinates x, y.
Q_INVOKABLE void selectByIds(const QgsFeatureIds &ids, Qgis::SelectBehavior behavior=Qgis::SelectBehavior::SetSelection, bool validateIds=false)
Selects matching features using a list of feature IDs.
Q_INVOKABLE void setFieldAlias(int index, const QString &aliasString)
Sets an alias (a display name) for attributes to display in dialogs.
friend class QgsVectorLayerFeatureSource
void minimumAndMaximumValue(int index, QVariant &minimum, QVariant &maximum) const
Calculates both the minimum and maximum value for an attribute column.
Q_INVOKABLE Qgis::GeometryType geometryType() const
Returns point, line or polygon.
Q_DECL_DEPRECATED void setExcludeAttributesWms(const QSet< QString > &att)
A set of attributes that are not advertised in WMS requests with QGIS server.
void setAttributeTableConfig(const QgsAttributeTableConfig &attributeTableConfig)
Sets the attribute table configuration object.
virtual bool setSubsetString(const QString &subset)
Sets the string (typically sql) used to define a subset of the layer.
void afterRollBack()
Emitted after changes are rolled back.
bool writeXml(QDomNode &layer_node, QDomDocument &doc, const QgsReadWriteContext &context) const final
Writes vector layer specific state to project file Dom node.
void setDiagramLayerSettings(const QgsDiagramLayerSettings &s)
bool readXml(const QDomNode &layer_node, QgsReadWriteContext &context) final
Reads vector layer specific state from project file Dom node.
QList< QgsWeakRelation > weakRelations() const
Returns the layer's weak relations as specified in the layer's style.
const QgsVectorSimplifyMethod & simplifyMethod() const
Returns the simplification settings for fast rendering of features.
void selectionChanged(const QgsFeatureIds &selected, const QgsFeatureIds &deselected, bool clearAndSelect)
Emitted when selection was changed.
void beforeAddingExpressionField(const QString &fieldName)
Will be emitted, when an expression field is going to be added to this vector layer.
Q_INVOKABLE bool deleteAttributes(const QList< int > &attrs)
Deletes a list of attribute fields (but does not commit it).
void updatedFields()
Emitted whenever the fields available from this layer have been changed.
QVariant defaultValue(int index, const QgsFeature &feature=QgsFeature(), QgsExpressionContext *context=nullptr) const
Returns the calculated default value for the specified field index.
void featureAdded(QgsFeatureId fid)
Emitted when a new feature has been added to the layer.
Q_INVOKABLE QString attributeAlias(int index) const
Returns the alias of an attribute name or a null string if there is no alias.
void featureDeleted(QgsFeatureId fid)
Emitted when a feature has been deleted.
Q_INVOKABLE void removeSelection()
Clear selection.
bool allowCommit() const
Controls, if the layer is allowed to commit changes.
QgsConditionalLayerStyles * conditionalStyles() const
Returns the conditional styles that are set for this layer.
void readCustomSymbology(const QDomElement &element, QString &errorMessage)
Signal emitted whenever the symbology (QML-file) for this layer is being read.
void setExtent3D(const QgsBox3D &rect) final
Sets the extent.
const QList< QgsVectorLayerJoinInfo > vectorJoins() const
Q_INVOKABLE bool renameAttribute(int index, const QString &newName)
Renames an attribute field (but does not commit it).
bool isSqlQuery() const
Returns true if the layer is a query (SQL) layer.
void beforeRollBack()
Emitted before changes are rolled back.
QgsAttributeList primaryKeyAttributes() const
Returns the list of attributes which make up the layer's primary keys.
void beginEditCommand(const QString &text)
Create edit command for undo/redo operations.
QString displayField() const
This is a shorthand for accessing the displayExpression if it is a simple field.
Q_DECL_DEPRECATED Qgis::GeometryOperationResult addRing(const QVector< QgsPointXY > &ring, QgsFeatureId *featureId=nullptr)
Adds a ring to polygon/multipolygon features.
bool addFeatures(QgsFeatureList &features, QgsFeatureSink::Flags flags=QgsFeatureSink::Flags()) final
Adds a list of features to the sink.
void setDiagramRenderer(QgsDiagramRenderer *r)
Sets diagram rendering object (takes ownership).
void geometryChanged(QgsFeatureId fid, const QgsGeometry &geometry)
Emitted whenever a geometry change is done in the edit buffer.
QgsEditFormConfig editFormConfig
QList< const QgsFeatureRendererGenerator * > featureRendererGenerators() const
Returns a list of the feature renderer generators owned by the layer.
Q_INVOKABLE QSet< QVariant > uniqueValues(int fieldIndex, int limit=-1) const final
Calculates a list of unique values contained within an attribute in the layer.
Q_INVOKABLE void removeFieldCustomComment(int index)
Removes the custom comment for the field.
bool moveVertex(double x, double y, QgsFeatureId atFeatureId, int atVertex)
Moves the vertex at the given position number, ring and item (first number is index 0),...
QgsGeometry getGeometry(QgsFeatureId fid) const
Queries the layer for the geometry at the given id.
int addTopologicalPoints(const QgsGeometry &geom)
Adds topological points for every vertex of the geometry.
bool addFeature(QgsFeature &feature, QgsFeatureSink::Flags flags=QgsFeatureSink::Flags()) final
Adds a single feature to the sink.
void beforeModifiedCheck() const
Emitted when the layer is checked for modifications. Use for last-minute additions.
Q_INVOKABLE QgsFeature getFeature(QgsFeatureId fid) const
Queries the layer for the feature with the given id.
Q_INVOKABLE void invertSelection()
Selects not selected features and deselects selected ones.
const QgsDiagramRenderer * diagramRenderer() const
Q_INVOKABLE bool changeAttributeValues(QgsFeatureId fid, const QgsAttributeMap &newValues, const QgsAttributeMap &oldValues=QgsAttributeMap(), bool skipDefaultValues=false, QgsVectorLayerToolsContext *context=nullptr)
Changes attributes' values for a feature (but does not immediately commit the changes).
QgsMapLayerSelectionProperties * selectionProperties() override
Returns the layer's selection properties.
QgsVectorDataProvider * dataProvider() final
Returns the layer's data provider, it may be nullptr.
bool changeGeometry(QgsFeatureId fid, QgsGeometry &geometry, bool skipDefaultValue=false)
Changes a feature's geometry within the layer's edit buffer (but does not immediately commit the chan...
static const QgsSettingsEntryDouble * settingsSimplifyDrawingTol
Qgis::SpatialIndexPresence hasSpatialIndex() const override
void setFieldSplitPolicy(int index, Qgis::FieldDomainSplitPolicy policy)
Sets a split policy for the field with the specified index.
@ Referencing
The layer is referencing (or the "child" / "right" layer in the relationship).
@ Referenced
The layer is referenced (or the "parent" / "left" left in the relationship).
static void writeXml(const QgsVectorLayer *layer, WeakRelationType type, const QgsRelation &relation, QDomNode &node, QDomDocument &doc)
Writes a weak relation infoto an XML structure.
static QgsWeakRelation readXml(const QgsVectorLayer *layer, WeakRelationType type, const QDomNode &node, const QgsPathResolver resolver)
Returns a weak relation for the given layer.
static Qgis::GeometryType geometryType(Qgis::WkbType type)
Returns the geometry type for a WKB type, e.g., both MultiPolygon and CurvePolygon would have a Polyg...
static Q_INVOKABLE QString displayString(Qgis::WkbType type)
Returns a non-translated display string type for a WKB type, e.g., the geometry name used in WKT geom...
static Q_INVOKABLE QString geometryDisplayString(Qgis::GeometryType type)
Returns a display string for a geometry type.
static QDomElement writeVariant(const QVariant &value, QDomDocument &doc)
Write a QVariant to a QDomElement.
static QgsBox3D readBox3D(const QDomElement &element)
Decodes a DOM element to a 3D box.
static QVariant readVariant(const QDomElement &element)
Read a QVariant from a QDomElement.
static QgsRectangle readRectangle(const QDomElement &element)
@ UnknownCount
Provider returned an unknown feature count.
Definition qgis.h:574
double ANALYSIS_EXPORT angle(QgsPoint *p1, QgsPoint *p2, QgsPoint *p3, QgsPoint *p4)
Calculates the angle between two segments (in 2 dimension, z-values are ignored).
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
bool qgsVariantEqual(const QVariant &lhs, const QVariant &rhs)
Compares two QVariant values and returns whether they are equal, two NULL values are always treated a...
Definition qgis.cpp:657
bool qgsVariantLessThan(const QVariant &lhs, const QVariant &rhs)
Compares two QVariant values and returns whether the first is less than the second.
Definition qgis.cpp:596
bool qgsVariantGreaterThan(const QVariant &lhs, const QVariant &rhs)
Compares two QVariant values and returns whether the first is greater than the second.
Definition qgis.cpp:601
T qgsEnumKeyToValue(const QString &key, const T &defaultValue, bool tryValueAsKey=true, bool *returnOk=nullptr)
Returns the value corresponding to the given key of an enum.
Definition qgis.h:7335
QString qgsEnumValueToKey(const T &value, bool *returnOk=nullptr)
Returns the value for the given key of an enum.
Definition qgis.h:7316
QString qgsFlagValueToKeys(const T &value, bool *returnOk=nullptr)
Returns the value for the given keys of a flag.
Definition qgis.h:7374
T qgsFlagKeysToValue(const QString &keys, const T &defaultValue, bool tryValueAsKey=true, bool *returnOk=nullptr)
Returns the value corresponding to the given keys of a flag.
Definition qgis.h:7403
QMap< QString, QString > QgsStringMap
Definition qgis.h:7649
QVector< QgsPoint > QgsPointSequence
QMap< int, QVariant > QgsAttributeMap
QList< QgsFeature > QgsFeatureList
QSet< QgsFeatureId > QgsFeatureIds
qint64 QgsFeatureId
64 bit feature ids negative numbers are used for uncommitted/newly added features
#define QgsDebugMsgLevel(str, level)
Definition qgslogger.h:63
#define QgsDebugError(str)
Definition qgslogger.h:59
QMap< int, QgsPropertyDefinition > QgsPropertiesDefinition
Definition of available properties.
#define RENDERER_TAG_NAME
Definition qgsrenderer.h:57
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS_NON_FATAL
#define QGIS_PROTECT_QOBJECT_THREAD_ACCESS
bool saveStyle_t(const QString &uri, const QString &qmlStyle, const QString &sldStyle, const QString &styleName, const QString &styleDescription, const QString &uiFileContent, bool useAsDefault, QString &errCause)
int listStyles_t(const QString &uri, QStringList &ids, QStringList &names, QStringList &descriptions, QString &errCause)
QString getStyleById_t(const QString &uri, QString styleID, QString &errCause)
bool deleteStyleById_t(const QString &uri, QString styleID, QString &errCause)
QString loadStyle_t(const QString &uri, QString &errCause)
QList< int > QgsAttributeList
QMap< QgsFeatureId, QgsFeature > QgsFeatureMap
A bundle of parameters controlling aggregate calculation.
Setting options for creating vector data providers.
Context for cascade delete features.
QList< QgsVectorLayer * > handledLayers(bool includeAuxiliaryLayers=true) const
Returns a list of all layers affected by the delete operation.
QMap< QgsVectorLayer *, QgsFeatureIds > mHandledFeatures
QgsFeatureIds handledFeatures(QgsVectorLayer *layer) const
Returns a list of feature IDs from the specified layer affected by the delete operation.
Setting options for loading vector layers.
bool skipCrsValidation
Controls whether the layer is allowed to have an invalid/unknown CRS.
bool forceReadOnly
Controls whether the layer is forced to be load as Read Only.
bool loadDefaultStyle
Set to true if the default layer style should be loaded.
QgsCoordinateTransformContext transformContext
Coordinate transform context.
QgsCoordinateReferenceSystem fallbackCrs
Fallback layer coordinate reference system.
Qgis::WkbType fallbackWkbType
Fallback geometry type.
bool loadAllStoredStyles
Controls whether the stored styles will be all loaded.