OTB  10.0.0
Orfeo Toolbox
otbScalarImageToPanTexTextureFilter.hxx
Go to the documentation of this file.
1 /*
2  * Copyright (C) 2005-2024 Centre National d'Etudes Spatiales (CNES)
3  *
4  * This file is part of Orfeo Toolbox
5  *
6  * https://www.orfeo-toolbox.org/
7  *
8  * Licensed under the Apache License, Version 2.0 (the "License");
9  * you may not use this file except in compliance with the License.
10  * You may obtain a copy of the License at
11  *
12  * http://www.apache.org/licenses/LICENSE-2.0
13  *
14  * Unless required by applicable law or agreed to in writing, software
15  * distributed under the License is distributed on an "AS IS" BASIS,
16  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
17  * See the License for the specific language governing permissions and
18  * limitations under the License.
19  */
20 
21 #ifndef otbScalarImageToPanTexTextureFilter_hxx
22 #define otbScalarImageToPanTexTextureFilter_hxx
23 
25 #include "itkImageRegionIteratorWithIndex.h"
26 #include "itkConstNeighborhoodIterator.h"
27 #include "itkImageRegionIterator.h"
28 #include "itkProgressReporter.h"
29 #include "itkNumericTraits.h"
30 
31 namespace otb
32 {
33 template <class TInputImage, class TOutputImage>
35  : m_Radius(), m_NumberOfBinsPerAxis(8), m_InputImageMinimum(0), m_InputImageMaximum(255)
36 {
37  this->DynamicMultiThreadingOn();
38  // There are 1 output corresponding to the Pan Tex texture indice
39  this->SetNumberOfRequiredOutputs(1);
40 
41  // Ten offsets are selected for contrast computation (2 pixels displacement grid, and given that the
42  // co-occurance matrix is symmetric
43  m_OffsetList = { {0, 1}, {0, 2}, {1, -2}, {1, -1}, {1, 0}, {1, 1}, {1, 2}, {2, -1}, {2, 0}, {2, 1} };
44 }
45 
46 template <class TInputImage, class TOutputImage>
48 {
49  // First, call superclass implementation
50  Superclass::GenerateInputRequestedRegion();
51 
52  // Retrieve the input and output pointers
53  InputImagePointerType inputPtr = const_cast<InputImageType*>(this->GetInput());
54  OutputImagePointerType outputPtr = this->GetOutput();
55 
56  if (!inputPtr || !outputPtr)
57  {
58  return;
59  }
60 
61  // Retrieve the output requested region
62  // We use only the first output since requested regions for all outputs are enforced to be equal
63  // by the default GenerateOutputRequestedRegiont() implementation
64  OutputRegionType outputRequestedRegion = outputPtr->GetRequestedRegion();
65 
66  // Build the input requested region
67  InputRegionType inputRequestedRegion = outputRequestedRegion;
68 
69  // Apply the radius
70  SizeType maxOffsetSize = {2, 2};
71  inputRequestedRegion.PadByRadius(m_Radius + maxOffsetSize);
72 
73  // Try to apply the requested region to the input image
74  if (inputRequestedRegion.Crop(inputPtr->GetLargestPossibleRegion()))
75  {
76  inputPtr->SetRequestedRegion(inputRequestedRegion);
77  }
78  else
79  {
80  // Build an exception
81  itk::InvalidRequestedRegionError e(__FILE__, __LINE__);
82  e.SetLocation(ITK_LOCATION);
83  e.SetDescription("Requested region is (at least partially) outside the largest possible region.");
84  e.SetDataObject(inputPtr);
85  throw e;
86  }
87 }
88 
89 template <class TInputImage, class TOutputImage>
91 {
92  // Retrieve the input and output pointers
93  InputImagePointerType inputPtr = const_cast<InputImageType*>(this->GetInput());
94  OutputImagePointerType outputPtr = this->GetOutput();
95 
96  itk::ImageRegionIteratorWithIndex<OutputImageType> outputIt(outputPtr, outputRegionForThread);
97  outputIt.GoToBegin();
98  // Iterate on outputs to compute textures
99  while (!outputIt.IsAtEnd())
100  {
101  // Initialise output value
102  double out = itk::NumericTraits<double>::max();
103 
104  // For each offset
105  typename OffsetListType::const_iterator offIt;
106  for (offIt = m_OffsetList.begin(); offIt != m_OffsetList.end(); ++offIt)
107  {
108  OffsetType currentOffset = *offIt;
109 
110  // Compute the region on which co-occurence will be estimated
111  typename InputRegionType::IndexType inputIndex;
112  typename InputRegionType::SizeType inputSize;
113 
114  // First, create an window for neighborhood iterator based on m_Radius
115  // For example, if xradius and yradius is 2. window size is 5x5 (2 *
116  // radius + 1).
117  for (unsigned int dim = 0; dim < InputImageType::ImageDimension; ++dim)
118  {
119  inputIndex[dim] = outputIt.GetIndex()[dim] - m_Radius[dim];
120  inputSize[dim] = 2 * m_Radius[dim] + 1;
121  }
122 
123  // Build the input region
124  InputRegionType inputRegion = {inputIndex, inputSize};
125  inputRegion.Crop(inputPtr->GetRequestedRegion());
126 
127  SizeType neighborhoodRadius;
129  unsigned int minRadius = 0;
130  for (unsigned int i = 0; i < currentOffset.GetOffsetDimension(); i++)
131  {
132  unsigned int distance = std::abs(currentOffset[i]);
133  if (distance > minRadius)
134  {
135  minRadius = distance;
136  }
137  }
138  neighborhoodRadius.Fill(minRadius);
140 
141  CooccurrenceIndexedListPointerType GLCIList = CooccurrenceIndexedListType::New();
142  GLCIList->Initialize(m_NumberOfBinsPerAxis, m_InputImageMinimum, m_InputImageMaximum);
143 
144  typedef itk::ConstNeighborhoodIterator<InputImageType> NeighborhoodIteratorType;
145  NeighborhoodIteratorType neighborIt = NeighborhoodIteratorType(neighborhoodRadius, inputPtr, inputRegion);
146 
147  for (neighborIt.GoToBegin(); !neighborIt.IsAtEnd(); ++neighborIt)
148  {
149  const InputPixelType centerPixelIntensity = neighborIt.GetCenterPixel();
150  bool pixelInBounds;
151  const InputPixelType pixelIntensity = neighborIt.GetPixel(currentOffset, pixelInBounds);
152  if (!pixelInBounds)
153  {
154  continue; // don't put a pixel in the histogram if it's out-of-bounds.
155  }
156  GLCIList->AddPixelPair(centerPixelIntensity, pixelIntensity);
157  }
158 
159  VectorConstIteratorType constVectorIt;
160  VectorType glcVector = GLCIList->GetVector();
161  double totalFrequency = static_cast<double>(GLCIList->GetTotalFrequency());
162 
163  // Compute inertia aka contrast
164  double inertia = 0;
165  constVectorIt = glcVector.begin();
166  while (constVectorIt != glcVector.end())
167  {
168  CooccurrenceIndexType index = constVectorIt->first;
169  RelativeFrequencyType frequency = constVectorIt->second / totalFrequency;
170  inertia += (index[0] - index[1]) * (index[0] - index[1]) * frequency;
171  ++constVectorIt;
172  }
173 
174  if (inertia < out)
175  {
176  out = inertia;
177  }
178  }
179 
180  outputIt.Set(out);
181  ++outputIt;
182  }
183 }
184 
185 } // End namespace otb
186 
187 #endif
CooccurrenceIndexedListType::RelativeFrequencyType RelativeFrequencyType
CooccurrenceIndexedListType::Pointer CooccurrenceIndexedListPointerType
CooccurrenceIndexedListType::VectorType VectorType
void DynamicThreadedGenerateData(const OutputRegionType &outputRegion) override
CooccurrenceIndexedListType::IndexType CooccurrenceIndexType
The "otb" namespace contains all Orfeo Toolbox (OTB) classes.