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