LCOV - code coverage report
Current view: top level - analysis/processing - qgsalgorithmdrape.cpp (source / functions) Hit Total Coverage
Test: coverage.info.cleaned Lines: 0 153 0.0 %
Date: 2021-03-26 12:19:53 Functions: 0 0 -
Branches: 0 0 -

           Branch data     Line data    Source code
       1                 :            : /***************************************************************************
       2                 :            :                          qgsalgorithmdrape.cpp
       3                 :            :                          ---------------------
       4                 :            :     begin                : November 2017
       5                 :            :     copyright            : (C) 2017 by Nyall Dawson
       6                 :            :     email                : nyall dot dawson at gmail dot com
       7                 :            :  ***************************************************************************/
       8                 :            : 
       9                 :            : /***************************************************************************
      10                 :            :  *                                                                         *
      11                 :            :  *   This program is free software; you can redistribute it and/or modify  *
      12                 :            :  *   it under the terms of the GNU General Public License as published by  *
      13                 :            :  *   the Free Software Foundation; either version 2 of the License, or     *
      14                 :            :  *   (at your option) any later version.                                   *
      15                 :            :  *                                                                         *
      16                 :            :  ***************************************************************************/
      17                 :            : 
      18                 :            : #include "qgsalgorithmdrape.h"
      19                 :            : #include "qgsvectorlayer.h"
      20                 :            : 
      21                 :            : ///@cond PRIVATE
      22                 :            : 
      23                 :            : 
      24                 :          0 : QString QgsDrapeAlgorithmBase::group() const
      25                 :            : {
      26                 :          0 :   return QObject::tr( "Vector geometry" );
      27                 :            : }
      28                 :            : 
      29                 :          0 : QString QgsDrapeAlgorithmBase::groupId() const
      30                 :            : {
      31                 :          0 :   return QStringLiteral( "vectorgeometry" );
      32                 :            : }
      33                 :            : 
      34                 :          0 : QString QgsDrapeAlgorithmBase::outputName() const
      35                 :            : {
      36                 :          0 :   return QObject::tr( "Draped" );
      37                 :            : }
      38                 :            : 
      39                 :          0 : void QgsDrapeAlgorithmBase::initParameters( const QVariantMap & )
      40                 :            : {
      41                 :          0 :   addParameter( new QgsProcessingParameterRasterLayer( QStringLiteral( "RASTER" ),
      42                 :          0 :                 QObject::tr( "Raster layer" ) ) );
      43                 :          0 :   addParameter( new QgsProcessingParameterBand( QStringLiteral( "BAND" ),
      44                 :          0 :                 QObject::tr( "Band number" ), 1, QStringLiteral( "RASTER" ) ) );
      45                 :            : 
      46                 :            :   // nodata value
      47                 :          0 :   std::unique_ptr< QgsProcessingParameterNumber > nodata = std::make_unique< QgsProcessingParameterNumber >( QStringLiteral( "NODATA" ),
      48                 :          0 :       QObject::tr( "Value for nodata or non-intersecting vertices" ), QgsProcessingParameterNumber::Double,
      49                 :          0 :       0.0 );
      50                 :          0 :   nodata->setIsDynamic( true );
      51                 :          0 :   nodata->setDynamicPropertyDefinition( QgsPropertyDefinition( QStringLiteral( "NODATA" ), QObject::tr( "Value for nodata or non-intersecting vertices" ), QgsPropertyDefinition::Double ) );
      52                 :          0 :   nodata->setDynamicLayerParameterName( QStringLiteral( "INPUT" ) );
      53                 :          0 :   addParameter( nodata.release() );
      54                 :            : 
      55                 :          0 :   auto scaleParam = std::make_unique< QgsProcessingParameterNumber >( QStringLiteral( "SCALE" ), QObject::tr( "Scale factor" ), QgsProcessingParameterNumber::Double, 1.0, false, 0.0 );
      56                 :          0 :   scaleParam->setIsDynamic( true );
      57                 :          0 :   scaleParam->setDynamicPropertyDefinition( QgsPropertyDefinition( QStringLiteral( "SCALE" ), QObject::tr( "Scale factor" ), QgsPropertyDefinition::Double ) );
      58                 :          0 :   scaleParam->setDynamicLayerParameterName( QStringLiteral( "INPUT" ) );
      59                 :          0 :   addParameter( scaleParam.release() );
      60                 :          0 : }
      61                 :            : 
      62                 :          0 : bool QgsDrapeAlgorithmBase::prepareAlgorithm( const QVariantMap &parameters, QgsProcessingContext &context, QgsProcessingFeedback * )
      63                 :            : {
      64                 :          0 :   mNoData = parameterAsDouble( parameters, QStringLiteral( "NODATA" ), context );
      65                 :          0 :   mDynamicNoData = QgsProcessingParameters::isDynamic( parameters, QStringLiteral( "NODATA" ) );
      66                 :          0 :   if ( mDynamicNoData )
      67                 :          0 :     mNoDataProperty = parameters.value( QStringLiteral( "NODATA" ) ).value< QgsProperty >();
      68                 :            : 
      69                 :          0 :   mScale = parameterAsDouble( parameters, QStringLiteral( "SCALE" ), context );
      70                 :          0 :   mDynamicScale = QgsProcessingParameters::isDynamic( parameters, QStringLiteral( "SCALE" ) );
      71                 :          0 :   if ( mDynamicScale )
      72                 :          0 :     mScaleProperty = parameters.value( QStringLiteral( "SCALE" ) ).value< QgsProperty >();
      73                 :            : 
      74                 :          0 :   QgsRasterLayer *layer = parameterAsRasterLayer( parameters, QStringLiteral( "RASTER" ), context );
      75                 :            : 
      76                 :          0 :   if ( !layer )
      77                 :          0 :     throw QgsProcessingException( invalidRasterError( parameters, QStringLiteral( "RASTER" ) ) );
      78                 :            : 
      79                 :          0 :   mBand = parameterAsInt( parameters, QStringLiteral( "BAND" ), context );
      80                 :          0 :   if ( mBand < 1 || mBand > layer->bandCount() )
      81                 :          0 :     throw QgsProcessingException( QObject::tr( "Invalid band number for BAND (%1): Valid values for input raster are 1 to %2" ).arg( mBand )
      82                 :          0 :                                   .arg( layer->bandCount() ) );
      83                 :          0 :   mRasterExtent = layer->extent();
      84                 :            : 
      85                 :          0 :   std::unique_ptr< QgsRasterInterface > provider( layer->dataProvider()->clone() );
      86                 :          0 :   QgsRasterDataProvider *dp = dynamic_cast< QgsRasterDataProvider * >( provider.get() );
      87                 :          0 :   if ( !dp )
      88                 :          0 :     throw QgsProcessingException( invalidRasterError( parameters, QStringLiteral( "RASTER" ) ) );
      89                 :            : 
      90                 :          0 :   mRasterProvider.reset( dp );
      91                 :          0 :   provider.release();
      92                 :            : 
      93                 :            :   return true;
      94                 :          0 : }
      95                 :            : 
      96                 :          0 : QgsFeatureList QgsDrapeAlgorithmBase::processFeature( const QgsFeature &feature, QgsProcessingContext &context, QgsProcessingFeedback *feedback )
      97                 :            : {
      98                 :          0 :   if ( !mCreatedTransform )
      99                 :            :   {
     100                 :          0 :     mCreatedTransform = true;
     101                 :          0 :     mTransform = QgsCoordinateTransform( sourceCrs(), mRasterProvider->crs(), context.transformContext() );
     102                 :            : 
     103                 :            :     // transform the raster extent back to the vector's crs, so that we can test
     104                 :            :     // whether individual vector geometries are actually covered by the raster
     105                 :            :     try
     106                 :            :     {
     107                 :          0 :       mRasterExtent = mTransform.transform( mRasterExtent, QgsCoordinateTransform::ReverseTransform );
     108                 :          0 :     }
     109                 :            :     catch ( QgsCsException & )
     110                 :            :     {
     111                 :          0 :       mRasterExtent = QgsRectangle();
     112                 :          0 :     }
     113                 :          0 :   }
     114                 :            : 
     115                 :          0 :   QgsFeature f = feature;
     116                 :          0 :   if ( f.hasGeometry() )
     117                 :            :   {
     118                 :          0 :     QgsGeometry geometry = f.geometry();
     119                 :            : 
     120                 :          0 :     double nodata = mNoData;
     121                 :          0 :     if ( mDynamicNoData )
     122                 :          0 :       nodata = mNoDataProperty.valueAsDouble( context.expressionContext(), nodata );
     123                 :            : 
     124                 :          0 :     double scale = mScale;
     125                 :          0 :     if ( mDynamicScale )
     126                 :          0 :       scale = mScaleProperty.valueAsDouble( context.expressionContext(), scale );
     127                 :            : 
     128                 :          0 :     prepareGeometry( geometry, nodata );
     129                 :            : 
     130                 :            :     // only do the "draping" if the geometry intersects the raster - otherwise skip
     131                 :            :     // a pointless iteration over all vertices
     132                 :          0 :     if ( !mRasterExtent.isNull() && geometry.boundingBoxIntersects( mRasterExtent ) )
     133                 :            :     {
     134                 :          0 :       geometry.transformVertices( [ = ]( const QgsPoint & p )->QgsPoint
     135                 :            :       {
     136                 :          0 :         QgsPointXY t;
     137                 :          0 :         double val = nodata;
     138                 :            :         try
     139                 :            :         {
     140                 :          0 :           t = mTransform.transform( p );
     141                 :          0 :           bool ok = false;
     142                 :          0 :           val = mRasterProvider->sample( t, mBand, &ok );
     143                 :          0 :           if ( !ok )
     144                 :          0 :             val = nodata;
     145                 :            :           else
     146                 :          0 :             val *= scale;
     147                 :          0 :         }
     148                 :            :         catch ( QgsCsException & )
     149                 :            :         {
     150                 :          0 :           feedback->reportError( QObject::tr( "Transform error while reprojecting feature {}" ).arg( f.id() ) );
     151                 :          0 :         }
     152                 :            : 
     153                 :          0 :         return drapeVertex( p, val );
     154                 :          0 :       } );
     155                 :          0 :     }
     156                 :            : 
     157                 :          0 :     f.setGeometry( geometry );
     158                 :          0 :   }
     159                 :          0 :   return QgsFeatureList() << f;
     160                 :          0 : }
     161                 :            : 
     162                 :            : 
     163                 :            : //
     164                 :            : // QgsDrapeToZAlgorithm
     165                 :            : //
     166                 :            : 
     167                 :          0 : QString QgsDrapeToZAlgorithm::name() const
     168                 :            : {
     169                 :          0 :   return QStringLiteral( "setzfromraster" );
     170                 :            : }
     171                 :            : 
     172                 :          0 : QString QgsDrapeToZAlgorithm::displayName() const
     173                 :            : {
     174                 :          0 :   return QObject::tr( "Drape (set Z value from raster)" );
     175                 :            : }
     176                 :            : 
     177                 :          0 : QStringList QgsDrapeToZAlgorithm::tags() const
     178                 :            : {
     179                 :          0 :   return QObject::tr( "3d,vertex,vertices,elevation,height,sample,dem,update,feature" ).split( ',' );
     180                 :          0 : }
     181                 :            : 
     182                 :          0 : QString QgsDrapeToZAlgorithm::shortHelpString() const
     183                 :            : {
     184                 :          0 :   return QObject::tr( "This algorithm sets the z value of every vertex in the feature geometry to a value sampled from a band within a raster layer." )
     185                 :          0 :          + QStringLiteral( "\n\n" )
     186                 :          0 :          + QObject::tr( "The raster values can optionally be scaled by a preset amount." );
     187                 :          0 : }
     188                 :            : 
     189                 :          0 : QString QgsDrapeToZAlgorithm::shortDescription() const
     190                 :            : {
     191                 :          0 :   return QObject::tr( "Sets the z value for vertices to values sampled from a raster layer." );
     192                 :            : }
     193                 :            : 
     194                 :          0 : QgsDrapeToZAlgorithm *QgsDrapeToZAlgorithm::createInstance() const
     195                 :            : {
     196                 :          0 :   return new QgsDrapeToZAlgorithm();
     197                 :          0 : }
     198                 :            : 
     199                 :          0 : bool QgsDrapeToZAlgorithm::supportInPlaceEdit( const QgsMapLayer *l ) const
     200                 :            : {
     201                 :          0 :   const QgsVectorLayer *layer = qobject_cast< const QgsVectorLayer * >( l );
     202                 :          0 :   if ( !layer )
     203                 :          0 :     return false;
     204                 :            : 
     205                 :          0 :   if ( ! QgsDrapeAlgorithmBase::supportInPlaceEdit( layer ) )
     206                 :          0 :     return false;
     207                 :          0 :   return QgsWkbTypes::hasZ( layer->wkbType() );
     208                 :          0 : }
     209                 :            : 
     210                 :          0 : QgsWkbTypes::Type QgsDrapeToZAlgorithm::outputWkbType( QgsWkbTypes::Type inputWkbType ) const
     211                 :            : {
     212                 :          0 :   QgsWkbTypes::Type wkb = inputWkbType;
     213                 :          0 :   return QgsWkbTypes::addZ( wkb );
     214                 :            : }
     215                 :            : 
     216                 :          0 : void QgsDrapeToZAlgorithm::prepareGeometry( QgsGeometry &geometry, double defaultVal ) const
     217                 :            : {
     218                 :          0 :   geometry.get()->addZValue( defaultVal );
     219                 :          0 : }
     220                 :            : 
     221                 :          0 : QgsPoint QgsDrapeToZAlgorithm::drapeVertex( const QgsPoint &p, double rasterVal ) const
     222                 :            : {
     223                 :          0 :   return QgsPoint( p.wkbType(), p.x(), p.y(), rasterVal, p.m() );
     224                 :            : }
     225                 :            : 
     226                 :            : //
     227                 :            : // QgsDrapeToMAlgorithm
     228                 :            : //
     229                 :            : 
     230                 :          0 : QString QgsDrapeToMAlgorithm::name() const
     231                 :            : {
     232                 :          0 :   return QStringLiteral( "setmfromraster" );
     233                 :            : }
     234                 :            : 
     235                 :          0 : QString QgsDrapeToMAlgorithm::displayName() const
     236                 :            : {
     237                 :          0 :   return QObject::tr( "Set M value from raster" );
     238                 :            : }
     239                 :            : 
     240                 :          0 : QStringList QgsDrapeToMAlgorithm::tags() const
     241                 :            : {
     242                 :          0 :   return QObject::tr( "drape,vertex,vertices,sample,dem,update,feature,measure" ).split( ',' );
     243                 :          0 : }
     244                 :            : 
     245                 :          0 : QString QgsDrapeToMAlgorithm::shortHelpString() const
     246                 :            : {
     247                 :          0 :   return QObject::tr( "This algorithm sets the M value for every vertex in the feature geometry to a value sampled from a band within a raster layer." )
     248                 :          0 :          + QStringLiteral( "\n\n" )
     249                 :          0 :          + QObject::tr( "The raster values can optionally be scaled by a preset amount." );
     250                 :          0 : }
     251                 :            : 
     252                 :          0 : QString QgsDrapeToMAlgorithm::shortDescription() const
     253                 :            : {
     254                 :          0 :   return QObject::tr( "Sets the M value for vertices to values sampled from a raster layer." );
     255                 :            : }
     256                 :            : 
     257                 :          0 : QgsDrapeToMAlgorithm *QgsDrapeToMAlgorithm::createInstance() const
     258                 :            : {
     259                 :          0 :   return new QgsDrapeToMAlgorithm();
     260                 :          0 : }
     261                 :            : 
     262                 :          0 : bool QgsDrapeToMAlgorithm::supportInPlaceEdit( const QgsMapLayer *l ) const
     263                 :            : {
     264                 :          0 :   const QgsVectorLayer *layer = qobject_cast< const QgsVectorLayer * >( l );
     265                 :          0 :   if ( !layer )
     266                 :          0 :     return false;
     267                 :            : 
     268                 :          0 :   if ( ! QgsDrapeAlgorithmBase::supportInPlaceEdit( layer ) )
     269                 :          0 :     return false;
     270                 :          0 :   return QgsWkbTypes::hasM( layer->wkbType() );
     271                 :          0 : }
     272                 :            : 
     273                 :          0 : QgsWkbTypes::Type QgsDrapeToMAlgorithm::outputWkbType( QgsWkbTypes::Type inputWkbType ) const
     274                 :            : {
     275                 :          0 :   QgsWkbTypes::Type wkb = inputWkbType;
     276                 :          0 :   return QgsWkbTypes::addM( wkb );
     277                 :            : }
     278                 :            : 
     279                 :          0 : void QgsDrapeToMAlgorithm::prepareGeometry( QgsGeometry &geometry, double defaultVal ) const
     280                 :            : {
     281                 :          0 :   geometry.get()->addMValue( defaultVal );
     282                 :          0 : }
     283                 :            : 
     284                 :          0 : QgsPoint QgsDrapeToMAlgorithm::drapeVertex( const QgsPoint &p, double rasterVal ) const
     285                 :            : {
     286                 :          0 :   return QgsPoint( p.wkbType(), p.x(), p.y(), p.z(), rasterVal );
     287                 :            : }
     288                 :            : 
     289                 :            : 
     290                 :            : ///@endcond
     291                 :            : 
     292                 :            : 

Generated by: LCOV version 1.14