26using namespace Qt::StringLiterals;
34 std::vector<QgsFeatureId> allFeats;
35 allFeats.reserve( count );
37 feedback->
pushInfo( QObject::tr(
"Building list of all features..." ) );
42 allFeats.push_back( f.
id() );
46 std::random_device randomDevice;
47 std::mt19937 mersenneTwister( randomDevice() );
48 std::uniform_int_distribution<size_t> fidsDistribution;
52 const std::size_t actualFeatureCount = allFeats.size();
53 std::size_t shuffledFeatureCount = count;
54 bool invertSelection =
static_cast<std::size_t
>( count ) > actualFeatureCount / 2;
55 if ( invertSelection )
56 shuffledFeatureCount = actualFeatureCount - count;
58 std::size_t nb = actualFeatureCount;
61 feedback->
pushInfo( QObject::tr(
"Randomly selecting %1 features" ).arg( count ) );
62 auto cursor = allFeats.begin();
63 using difference_type = std::vector<QgsFeatureId>::difference_type;
64 while ( shuffledFeatureCount-- )
70 fidsDistribution.param( std::uniform_int_distribution<size_t>::param_type( 0, nb - 1 ) );
72 std::swap( *cursor, *( cursor +
static_cast<difference_type
>( fidsDistribution( mersenneTwister ) ) ) );
81 if ( invertSelection )
82 for (
auto it = cursor; it != allFeats.end(); ++it )
83 mSelectedFeatureIds.insert( *it );
85 for (
auto it = allFeats.begin(); it != cursor; ++it )
86 mSelectedFeatureIds.insert( *it );
89QString QgsRandomExtractSelectAlgorithmBase::group()
const
91 return QObject::tr(
"Vector selection" );
94QString QgsRandomExtractSelectAlgorithmBase::groupId()
const
96 return u
"vectorselection"_s;
101QString QgsRandomExtractAlgorithm::name()
const
103 return u
"randomextract"_s;
106QString QgsRandomExtractAlgorithm::displayName()
const
108 return QObject::tr(
"Random extract" );
111QStringList QgsRandomExtractAlgorithm::tags()
const
113 return QObject::tr(
"extract,filter,random,number,percentage" ).split(
',' );
116QString QgsRandomExtractAlgorithm::shortDescription()
const
118 return QObject::tr(
"Generates a vector layer that contains only a random subset of the features in an input layer." );
121QString QgsRandomExtractAlgorithm::shortHelpString()
const
124 "This algorithm takes a vector layer and generates a new one that contains only a subset "
125 "of the features in the input layer.\n\n"
126 "The subset is defined randomly, using a percentage or count value to define the total number "
127 "of features in the subset."
136QgsRandomExtractAlgorithm *QgsRandomExtractAlgorithm::createInstance()
const
138 return new QgsRandomExtractAlgorithm();
141void QgsRandomExtractAlgorithm::initAlgorithm(
const QVariantMap & )
144 addParameter(
new QgsProcessingParameterEnum( u
"METHOD"_s, QObject::tr(
"Method" ), QStringList() << QObject::tr(
"Number of features" ) << QObject::tr(
"Percentage of features" ),
false, 0 ) );
152 std::unique_ptr<QgsProcessingFeatureSource> source( parameterAsSource( parameters, u
"INPUT"_s, context ) );
161 const int method = parameterAsEnum( parameters, u
"METHOD"_s, context );
162 double number = parameterAsDouble( parameters, u
"NUMBER"_s, context );
168 if ( number >
static_cast<double>( count ) )
169 throw QgsProcessingException( QObject::tr(
"Selected number is greater than feature count. Choose a lower value and try again." ) );
175 throw QgsProcessingException( QObject::tr(
"Percentage can't be greater than 100. Choose a lower value and try again." ) );
177 number = std::ceil( number *
static_cast<double>( count ) / 100 );
180 sampleFeatureIds( source.get(),
static_cast<long long>( number ), feedback );
182 feedback->
pushInfo( QObject::tr(
"Adding selected features" ) );
188 return QVariantMap();
197 outputs.insert( u
"OUTPUT"_s, dest );
203QString QgsRandomSelectionAlgorithm::name()
const
205 return u
"randomselection"_s;
208QString QgsRandomSelectionAlgorithm::displayName()
const
210 return QObject::tr(
"Random selection" );
213QStringList QgsRandomSelectionAlgorithm::tags()
const
215 return QObject::tr(
"select,random,number,percentage" ).split(
',' );
218QString QgsRandomSelectionAlgorithm::shortDescription()
const
220 return QObject::tr(
"Randomly selects features from a vector layer." );
223QString QgsRandomSelectionAlgorithm::shortHelpString()
const
226 "This algorithm takes a vector layer and selects a subset of its features. "
227 "No new layer is generated by this algorithm.\n\n"
228 "The subset is defined randomly, using a percentage or count value to define "
229 "the total number of features in the subset."
233QgsRandomSelectionAlgorithm *QgsRandomSelectionAlgorithm::createInstance()
const
235 return new QgsRandomSelectionAlgorithm();
238void QgsRandomSelectionAlgorithm::initAlgorithm(
const QVariantMap & )
241 addParameter(
new QgsProcessingParameterEnum( u
"METHOD"_s, QObject::tr(
"Method" ), QStringList() << QObject::tr(
"Number of features" ) << QObject::tr(
"Percentage of features" ),
false, 0 ) );
249 mInput = parameters.value( u
"INPUT"_s );
250 mTargetLayer = parameterAsVectorLayer( parameters, u
"INPUT"_s, context );
255 const int method = parameterAsEnum( parameters, u
"METHOD"_s, context );
256 double number = parameterAsDouble( parameters, u
"NUMBER"_s, context );
257 const long long count = mTargetLayer->featureCount();
262 if ( number >
static_cast<double>( count ) )
263 throw QgsProcessingException( QObject::tr(
"Selected number is greater than feature count. Choose a lower value and try again." ) );
269 throw QgsProcessingException( QObject::tr(
"Percentage can't be greater than 100. Choose a lower value and try again." ) );
271 number = std::ceil( number *
static_cast<double>( count ) / 100 );
275 sampleFeatureIds( mTargetLayer,
static_cast<long long>( number ), feedback );
277 return QVariantMap();
282 mTargetLayer->selectByIds( mSelectedFeatureIds );
285 outputs.insert( u
"OUTPUT"_s, mInput );
@ Vector
Tables (i.e. vector layers with or without geometry). When used for a sink this indicates the sink ha...
@ NoGeometry
Geometry is not required. It may still be returned if e.g. required for a filter condition.
@ RegeneratesPrimaryKey
Algorithm always drops any existing primary keys or FID values and regenerates them in outputs.
QFlags< ProcessingAlgorithmDocumentationFlag > ProcessingAlgorithmDocumentationFlags
Flags describing algorithm behavior for documentation purposes.
@ SkipGeometryValidityChecks
Invalid geometry checks should always be skipped. This flag can be useful for algorithms which always...
@ Double
Double/float values.
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.
Wraps a request for features to a vector layer (or directly its vector data provider).
@ FastInsert
Use faster inserts, at the cost of updating the passed features to reflect changes made at the provid...
@ RegeneratePrimaryKey
This flag indicates, that a primary key field cannot be guaranteed to be unique and the sink should i...
An interface for objects which provide features via a getFeatures method.
virtual QgsFields fields() const =0
Returns the fields associated with features in the source.
virtual QgsCoordinateReferenceSystem sourceCrs() const =0
Returns the coordinate reference system for features in the source.
virtual Qgis::WkbType wkbType() const =0
Returns the geometry type for features returned by this source.
virtual QgsFeatureIterator getFeatures(const QgsFeatureRequest &request=QgsFeatureRequest()) const =0
Returns an iterator for the features in the source.
virtual long long featureCount() const =0
Returns the number of features contained in the source, or -1 if the feature count is unknown.
The feature class encapsulates a single feature including its unique ID, geometry and a list of field...
bool isCanceled() const
Tells whether the operation has been canceled already.
Contains information about the context in which a processing algorithm is executed.
Custom exception class for processing related exceptions.
Base class for providing feedback from a processing algorithm.
virtual void pushInfo(const QString &info)
Pushes a general informational message from the algorithm.
A vector layer output for processing algorithms.
An enum based parameter for processing algorithms, allowing for selection from predefined values.
A feature sink output for processing algorithms.
An input feature source (such as vector layers) parameter for processing algorithms.
A numeric parameter for processing algorithms.
A vector layer (with or without geometry) parameter for processing algorithms.