From 5762304eae56184f12e2e5fdfa3051db4e5bfc89 Mon Sep 17 00:00:00 2001 From: Emmanuel Christophe <emmanuel.christophe@orfeo-toolbox.org> Date: Tue, 2 Dec 2008 16:19:52 +0800 Subject: [PATCH] ENH: Adding ImageFittingPolygonList class --- .../otbImageFittingPolygonListFilter.h | 114 ++++++++ .../otbImageFittingPolygonListFilter.txx | 273 ++++++++++++++++++ 2 files changed, 387 insertions(+) create mode 100644 Code/FeatureExtraction/otbImageFittingPolygonListFilter.h create mode 100644 Code/FeatureExtraction/otbImageFittingPolygonListFilter.txx diff --git a/Code/FeatureExtraction/otbImageFittingPolygonListFilter.h b/Code/FeatureExtraction/otbImageFittingPolygonListFilter.h new file mode 100644 index 0000000000..39ba6869f4 --- /dev/null +++ b/Code/FeatureExtraction/otbImageFittingPolygonListFilter.h @@ -0,0 +1,114 @@ +/*========================================================================= + +Program: ORFEO Toolbox +Language: C++ +Date: $Date$ +Version: $Revision$ + + +Copyright (c) Centre National d'Etudes Spatiales. All rights reserved. +See OTBCopyright.txt for details. + + +This software is distributed WITHOUT ANY WARRANTY; without even +the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR +PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#ifndef __otbImagePerturbationPolygonListFilter_h +#define __otbImagePerturbationPolygonListFilter_h + +#include "otbPathListToPathListFilter.h" +#include "otbMacro.h" +#include "itkLineConstIterator.h" + +namespace otb +{ +/** \class ImagePerturbationPolygonListFilter + * \brief Slightly deform polygon to reach higher enery from the image + * + * <br>Limitations:</br> This filter is currently working with integer position + * for the polygon vertices. It should be optimized for continuous positions. + * + */ +template <class TPath, class TImage> +class ITK_EXPORT ImagePerturbationPolygonListFilter + : public PathListToPathListFilter<TPath> +{ + public: + /** Standard typedefs */ + typedef ImagePerturbationPolygonListFilter Self; + typedef PathListToPathListFilter<TPath> Superclass; + typedef itk::SmartPointer<Self> Pointer; + typedef itk::SmartPointer<const Self> ConstPointer; + + /** Type macro */ + itkNewMacro(Self); + + /** Creation through object factory macro */ + itkTypeMacro(ImagePerturbationPolygonListFilter, PathListToPathListFilter); + + /** Template parameters typedefs */ + typedef typename Superclass::PathType PathType; + typedef typename Superclass::PathListType PathListType; + typedef typename Superclass::PathPointerType PathPointerType; + typedef typename PathListType::Pointer PathListPointerType; + typedef typename PathListType::ConstIterator IteratorType; + typedef typename PathType::VertexType VertexType; + typedef typename PathType::VertexListType VertexListType; + typedef typename VertexListType::ConstIterator VertexListConstIteratorType; + typedef double RealType; + + typedef TImage ImageType; + typedef typename ImageType::Pointer ImagePointerType; + typedef typename ImageType::ConstPointer ImageConstPointerType; + + typedef itk::LineConstIterator<ImageType> LineConstIteratorType; + + /** + * Set the input Likelihood image. + * \param image The Likelihood image. + */ + void SetInputImage(const ImageType * image); + + /** + * Get the input Likelihood image. + * \return The input Likelihood image. + */ + const ImageType * GetInputImage(void); + + /** Set/Get the search radius. */ + itkSetMacro(Radius,unsigned int); + itkGetMacro(Radius,unsigned int); + + /** Set/Get the number of iteration. */ + itkSetMacro(NumberOfIterations,unsigned int); + itkGetMacro(NumberOfIterations,unsigned int); + +protected: + /** Constructor */ + ImagePerturbationPolygonListFilter(); + /** Destructor */ + virtual ~ImagePerturbationPolygonListFilter() {}; + /** GenerateData method */ + virtual void GenerateData(); + /** PrintSelf method */ + virtual void PrintSelf(std::ostream& os, itk::Indent indent) const; + + virtual double computeValue(ImageConstPointerType image, VertexType middlePoint, VertexType previousPoint, VertexType nextPoint) const; + + unsigned int m_Radius; + unsigned int m_NumberOfIterations; + +private: + ImagePerturbationPolygonListFilter(const Self&); //purposely not implemented + void operator=(const Self&); //purposely not implemented + + +}; +}// End namespace otb +#ifndef OTB_MANUAL_INSTANTIATION +#include "otbImagePerturbationPolygonListFilter.txx" +#endif + +#endif diff --git a/Code/FeatureExtraction/otbImageFittingPolygonListFilter.txx b/Code/FeatureExtraction/otbImageFittingPolygonListFilter.txx new file mode 100644 index 0000000000..2d054ca12d --- /dev/null +++ b/Code/FeatureExtraction/otbImageFittingPolygonListFilter.txx @@ -0,0 +1,273 @@ +/*========================================================================= + +Program: ORFEO Toolbox +Language: C++ +Date: $Date$ +Version: $Revision$ + + +Copyright (c) Centre National d'Etudes Spatiales. All rights reserved. +See OTBCopyright.txt for details. + + +This software is distributed WITHOUT ANY WARRANTY; without even +the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR +PURPOSE. See the above copyright notices for more information. + +=========================================================================*/ +#ifndef __otbImagePerturbationPolygonListFilter_txx +#define __otbImagePerturbationPolygonListFilter_txx + +#include "otbImagePerturbationPolygonListFilter.h" +#include "otbPolyLineImageConstIterator.h" +#include "otbMacro.h" + + +namespace otb +{ +/** + * Constructor + */ +template <class TPath, class TImage> +ImagePerturbationPolygonListFilter<TPath, TImage> +::ImagePerturbationPolygonListFilter() +{ + this->SetNumberOfRequiredInputs(2); + this->SetNumberOfInputs(2); + m_Radius=1; + m_NumberOfIterations=1; +} + +template <class TPath, class TImage> +void +ImagePerturbationPolygonListFilter<TPath, TImage> +::SetInputImage(const ImageType * image) +{ + this->itk::ProcessObject::SetNthInput(1,const_cast<ImageType *>(image)); +} + +template <class TPath, class TImage> +const typename ImagePerturbationPolygonListFilter<TPath, TImage> +::ImageType * +ImagePerturbationPolygonListFilter<TPath, TImage> +::GetInputImage(void) +{ + if(this->GetNumberOfInputs()<1) + { + return 0; + } + return static_cast<const ImageType *>(this->itk::ProcessObject::GetInput(1)); +} + + +//FIXME +//There is an issue here with integer and continous indexes +//maybe we should use the itk::LineConstIterator +template <class TPath, class TImage> +void +ImagePerturbationPolygonListFilter<TPath, TImage> +::GenerateData() +{ + // I/O wiring + ImageConstPointerType inputImagePtr = this->GetInputImage(); + const PathListType * inputPtr = this->GetInput(); + PathListType * outputPtr = this->GetOutput(); + + + typename ImageType::RegionType regionLargest=inputImagePtr->GetLargestPossibleRegion(); + + typedef itk::ImageRegionConstIteratorWithIndex<ImageType> NeighborhoodIteratorType; + + typename ImageType::SizeType size; + size[0]= 2*m_Radius+1; + size[1]= 2*m_Radius+1; + typename ImageType::RegionType region; + region.SetSize(size); + typename ImageType::IndexType start; + + //go through all the polygons in the list + for(IteratorType it = inputPtr->Begin(); it != inputPtr->End(); ++it) + { + PathPointerType polygon = it.Get(); + + if(polygon->GetVertexList()->Size()>2) + { + for (int iteration=0;iteration < m_NumberOfIterations;++iteration) + { + PathPointerType newPolygon = PathType::New(); + VertexListConstIteratorType vertexIt = polygon->GetVertexList()->Begin(); + //We are now going to go through all the vertex, we won't start to process + // first as we need to know the last one for that. + VertexType firstPoint = vertexIt.Value(); + VertexType previousPoint = vertexIt.Value(); + ++vertexIt; + VertexType currentPoint = vertexIt.Value(); + ++vertexIt; + while (vertexIt != polygon->GetVertexList()->End()) + { + VertexType nextPoint=vertexIt.Value(); + + /** try all the possible neighbor for the current point + * to factorize + * */ + { + + start[0] = static_cast<long int>(currentPoint[0]-m_Radius); + start[1] = static_cast<long int>(currentPoint[1]-m_Radius); + region.SetIndex(start); + + NeighborhoodIteratorType nIt(inputImagePtr, region); + double maxValue=0.0; + VertexType maxPoint = currentPoint; + for (nIt.GoToBegin();!nIt.IsAtEnd();++nIt) + { + if(regionLargest.IsInside(nIt.GetIndex())) + { + VertexType middlePoint=static_cast<VertexType>(nIt.GetIndex()); + double currentValue = computeValue(inputImagePtr, middlePoint, previousPoint, nextPoint); + if (currentValue > maxValue) + { + maxValue=currentValue; + maxPoint=middlePoint; + } + } + } + currentPoint=maxPoint; + newPolygon->AddVertex(maxPoint); + } + /** End 'to factorize' */ + + ++vertexIt; + previousPoint=currentPoint; + currentPoint=nextPoint; + + } + //We now need to process the last and the first point + + VertexType nextPoint = firstPoint; + /** try all the possible neighbor for the current point + * to factorize + * */ + { + start[0] = static_cast<long int>(currentPoint[0]-m_Radius); + start[1] = static_cast<long int>(currentPoint[1]-m_Radius); + region.SetIndex(start); + + NeighborhoodIteratorType nIt(inputImagePtr, region); + double maxValue=0.0; + VertexType maxPoint = currentPoint; + for (nIt.GoToBegin();!nIt.IsAtEnd();++nIt) + { + if(regionLargest.IsInside(nIt.GetIndex())) + { + VertexType middlePoint=static_cast<VertexType>(nIt.GetIndex()); + double currentValue = computeValue(inputImagePtr, middlePoint, previousPoint, nextPoint); + if (currentValue > maxValue) + { + maxValue=currentValue; + maxPoint=middlePoint; + } + } + } + currentPoint=maxPoint; + newPolygon->AddVertex(maxPoint); + } + /** End 'to factorize' */ + + previousPoint = currentPoint; + currentPoint= firstPoint; + vertexIt = newPolygon->GetVertexList()->Begin(); + nextPoint=vertexIt.Value(); + + /** try all the possible neighbor for the current point + * to factorize + * */ + { + + start[0] = static_cast<long int>(currentPoint[0]-m_Radius); + start[1] = static_cast<long int>(currentPoint[1]-m_Radius); + region.SetIndex(start); + + NeighborhoodIteratorType nIt(inputImagePtr, region); + double maxValue=0.0; + VertexType maxPoint = currentPoint; + for (nIt.GoToBegin();!nIt.IsAtEnd();++nIt) + { + if(regionLargest.IsInside(nIt.GetIndex())) + { + VertexType middlePoint=static_cast<VertexType>(nIt.GetIndex()); + double currentValue = computeValue(inputImagePtr, middlePoint, previousPoint, nextPoint); + if (currentValue > maxValue) + { + maxValue=currentValue; + maxPoint=middlePoint; + } + } + } + currentPoint=maxPoint; + newPolygon->AddVertex(maxPoint); + } + /** End 'to factorize' */ + + polygon = newPolygon;//prepare the next iteration + } + } + + outputPtr->PushBack(polygon); + + }//going through the polygon list + + +} + +template <class TPath, class TImage> + double + ImagePerturbationPolygonListFilter<TPath, TImage> + ::computeValue(ImageConstPointerType image, VertexType middlePoint, VertexType previousPoint, VertexType nextPoint) const +{ + typedef typename ImageType::IndexType IndexType; + IndexType middleIndex; + IndexType previousIndex; + IndexType nextIndex; + middleIndex[0]=static_cast<long int>(middlePoint[0]); + middleIndex[1]=static_cast<long int>(middlePoint[1]); + previousIndex[0]=static_cast<long int>(previousPoint[0]); + previousIndex[1]=static_cast<long int>(previousPoint[1]); + nextIndex[0]=static_cast<long int>(nextPoint[0]); + nextIndex[1]=static_cast<long int>(nextPoint[1]); + double currentValue = 0.0; + unsigned int count = 0; + {//compute for first segment + LineConstIteratorType itLine(image, previousIndex, middleIndex); + while (!itLine.IsAtEnd()) + { + currentValue += itLine.Get(); + ++count; + ++itLine; + } + } + {//compute for second segment + LineConstIteratorType itLine(image, nextIndex, middleIndex); + while (!itLine.IsAtEnd()) + { + currentValue += itLine.Get(); + ++count; + ++itLine; + } + } + return currentValue/count; +} + +/** + * PrintSelf Method + */ +template <class TPath, class TImage> +void +ImagePerturbationPolygonListFilter<TPath, TImage> +::PrintSelf(std::ostream& os, itk::Indent indent) const +{ + Superclass::PrintSelf(os, indent); +} + +} // End namespace otb +#endif -- GitLab