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