Branch data Line data Source code
1 : : /*************************************************************************** 2 : : qgsrasternuller.cpp 3 : : --------------------- 4 : : begin : August 2012 5 : : copyright : (C) 2012 by Radim Blazek 6 : : email : radim dot blazek 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 "qgsrasterdataprovider.h" 19 : : #include "qgsrasternuller.h" 20 : : 21 : 0 : QgsRasterNuller::QgsRasterNuller( QgsRasterInterface *input ) 22 : 0 : : QgsRasterInterface( input ) 23 : 0 : { 24 : 0 : } 25 : : 26 : 0 : QgsRasterNuller *QgsRasterNuller::clone() const 27 : : { 28 : 0 : QgsDebugMsgLevel( QStringLiteral( "Entered" ), 4 ); 29 : 0 : QgsRasterNuller *nuller = new QgsRasterNuller( nullptr ); 30 : 0 : nuller->mNoData = mNoData; 31 : 0 : nuller->mOutputNoData = mOutputNoData; 32 : 0 : nuller->mHasOutputNoData = mHasOutputNoData; 33 : 0 : return nuller; 34 : 0 : } 35 : : 36 : 0 : void QgsRasterNuller::setOutputNoDataValue( int bandNo, double noData ) 37 : : { 38 : 0 : if ( bandNo > mOutputNoData.size() ) 39 : : { 40 : 0 : mOutputNoData.resize( bandNo ); 41 : 0 : mHasOutputNoData.resize( bandNo ); 42 : 0 : } 43 : 0 : mOutputNoData[bandNo - 1] = noData; 44 : 0 : mHasOutputNoData[bandNo - 1] = true; 45 : 0 : } 46 : : 47 : 0 : void QgsRasterNuller::setNoData( int bandNo, const QgsRasterRangeList &noData ) 48 : : { 49 : 0 : if ( bandNo > mNoData.size() ) 50 : : { 51 : 0 : mNoData.resize( bandNo ); 52 : 0 : } 53 : 0 : mNoData[bandNo - 1] = noData; 54 : 0 : } 55 : : 56 : 0 : int QgsRasterNuller::bandCount() const 57 : : { 58 : 0 : if ( mInput ) return mInput->bandCount(); 59 : 0 : return 0; 60 : 0 : } 61 : : 62 : 0 : Qgis::DataType QgsRasterNuller::dataType( int bandNo ) const 63 : : { 64 : 0 : if ( mInput ) return mInput->dataType( bandNo ); 65 : 0 : return Qgis::UnknownDataType; 66 : 0 : } 67 : : 68 : 0 : QgsRasterBlock *QgsRasterNuller::block( int bandNo, QgsRectangle const &extent, int width, int height, QgsRasterBlockFeedback *feedback ) 69 : : { 70 : 0 : QgsDebugMsgLevel( QStringLiteral( "Entered" ), 4 ); 71 : 0 : if ( !mInput ) 72 : : { 73 : 0 : return new QgsRasterBlock(); 74 : : } 75 : : 76 : 0 : std::unique_ptr< QgsRasterBlock > inputBlock( mInput->block( bandNo, extent, width, height, feedback ) ); 77 : 0 : if ( !inputBlock ) 78 : : { 79 : 0 : return new QgsRasterBlock(); 80 : : } 81 : : 82 : : // We don't support nuller for color types 83 : 0 : if ( QgsRasterBlock::typeIsColor( inputBlock->dataType() ) ) 84 : : { 85 : 0 : return inputBlock.release(); 86 : : } 87 : : 88 : 0 : std::unique_ptr< QgsRasterBlock > outputBlock( new QgsRasterBlock( inputBlock->dataType(), width, height ) ); 89 : 0 : if ( mHasOutputNoData.value( bandNo - 1 ) || inputBlock->hasNoDataValue() ) 90 : : { 91 : : double noDataValue; 92 : 0 : if ( mHasOutputNoData.value( bandNo - 1 ) ) 93 : : { 94 : 0 : noDataValue = mOutputNoData.value( bandNo - 1 ); 95 : 0 : } 96 : : else 97 : : { 98 : 0 : noDataValue = inputBlock->noDataValue(); 99 : : } 100 : 0 : outputBlock->setNoDataValue( noDataValue ); 101 : 0 : } 102 : : 103 : 0 : bool isNoData = false; 104 : 0 : for ( int i = 0; i < height; i++ ) 105 : : { 106 : 0 : for ( int j = 0; j < width; j++ ) 107 : : { 108 : 0 : double value = inputBlock->valueAndNoData( i, j, isNoData ); 109 : : 110 : 0 : if ( QgsRasterRange::contains( value, mNoData.value( bandNo - 1 ) ) ) 111 : : { 112 : 0 : isNoData = true; 113 : 0 : } 114 : 0 : outputBlock->setValue( i, j, inputBlock->value( i, j ) ); 115 : 0 : if ( isNoData ) 116 : : { 117 : 0 : outputBlock->setIsNoData( i, j ); 118 : 0 : } 119 : : else 120 : : { 121 : 0 : outputBlock->setValue( i, j, value ); 122 : : } 123 : 0 : } 124 : 0 : } 125 : 0 : return outputBlock.release(); 126 : 0 : } 127 : :