SFSExample.cxx¶
Example usage:
./SFSExample Input/suburb2.jpeg \
Output/SFSLengthOutput.tif \
Output/SFSWidthOutput.tif \
Output/SFSMeanOutput.tif \
Output/SFSRatioOutput.tif \
Output/SFSSDOutput.tif \
Output/SFSPsiOutput.tif \
Output/SFSLengthPrettyOutput.tif \
Output/SFSWidthPrettyOutput.tif \
Output/SFSMeanPrettyOutput.tif \
Output/SFSRatioPrettyOutput.tif \
Output/SFSSDPrettyOutput.tif \
Output/SFSPsiPrettyOutput.tif \
20 \
50 \
8 \
4 \
0.6
Example source code (SFSExample.cxx):
#include "itkMacro.h"
#include "otbImage.h"
#include "otbImageFileReader.h"
#include "otbImageFileWriter.h"
#include "itkUnaryFunctorImageFilter.h"
#include "itkRescaleIntensityImageFilter.h"
// This example illustrates the use of the
// \doxygen{otb}{SFSTexturesImageFilter}.
// This filter computes the Structural Feature Set as described in
// \cite{SFS}. These features are textural parameters which give
// information about the structure of lines passing through each pixel
// of the image.
//
// The first step required to use this filter is to include its header file.
#include "otbSFSTexturesImageFilter.h"
int main(int itkNotUsed(argc), char* argv[])
{
using PixelType = double;
const unsigned int Dimension = 2;
std::string inName = argv[1];
std::string outNameLength = argv[2];
std::string outNameWidth = argv[3];
std::string outNameWMean = argv[4];
std::string outNameRatio = argv[5];
std::string outNameSD = argv[6];
std::string outNamePsi = argv[7];
std::string lengthprettyfname = argv[8];
std::string widthprettyfname = argv[9];
std::string wmeanprettyfname = argv[10];
std::string ratioprettyfname = argv[11];
std::string sdprettyfname = argv[12];
std::string psiprettyfname = argv[13];
PixelType spectThresh = atof(argv[14]);
unsigned int spatialThresh = atoi(argv[15]);
unsigned int dirNb = atoi(argv[16]);
unsigned int maxConsideration = atoi(argv[17]);
double alpha = atof(argv[18]);
// As with every OTB program, we start by defining the types for the
// images, the readers and the writers.
using ImageType = otb::Image<PixelType, Dimension>;
using ReaderType = otb::ImageFileReader<ImageType>;
using WriterType = otb::ImageFileWriter<ImageType>;
// The we can instantiate the type for the SFS filter, which is
// templated over the input and output pixel types.
using SFSFilterType = otb::SFSTexturesImageFilter<ImageType, ImageType>;
// After that, we can instantiate the filter. We will also instantiate
// the reader and one writer for each output image, since the SFS
// filter generates 6 different features.
SFSFilterType::Pointer filter = SFSFilterType::New();
ReaderType::Pointer reader = ReaderType::New();
WriterType::Pointer writerLength = WriterType::New();
WriterType::Pointer writerWidth = WriterType::New();
WriterType::Pointer writerWMean = WriterType::New();
WriterType::Pointer writerRatio = WriterType::New();
WriterType::Pointer writerSD = WriterType::New();
WriterType::Pointer writerPsi = WriterType::New();
reader->SetFileName(inName);
// The SFS filter has several parameters which have to be
// selected. They are:
// \begin{enumerate}
// \item a spectral threshold to decide if 2 neighboring pixels are
// connected;
//\item a spatial threshold defining the maximum length for an
// extracted line;
//\item the number of directions which will be analyzed (the first
// one is to the right and they are equally distributed between 0 and
// $2\pi$);
// \item the $\alpha$ parameter fort the $\omega-mean$ feature;
// \item the RatioMax parameter fort the $\omega-mean$ feature.
// \end{enumerate}
filter->SetSpectralThreshold(spectThresh);
filter->SetSpatialThreshold(spatialThresh);
filter->SetNumberOfDirections(dirNb);
filter->SetRatioMaxConsiderationNumber(maxConsideration);
filter->SetAlpha(alpha);
// In order to disable the computation of a feature, the
// \code{SetFeatureStatus} parameter can be used. The $true$ value
// enables the feature (default behavior) and the $false$ value
// disables the computation. Therefore, the following line is useless,
// but is given here as an example.
filter->SetFeatureStatus(SFSFilterType::PSI, true);
// Now, we plug the pipeline using all the writers.
filter->SetInput(reader->GetOutput());
writerLength->SetFileName(outNameLength);
writerLength->SetInput(filter->GetLengthOutput());
writerLength->Update();
writerWidth->SetFileName(outNameWidth);
writerWidth->SetInput(filter->GetWidthOutput());
writerWidth->Update();
writerWMean->SetFileName(outNameWMean);
writerWMean->SetInput(filter->GetWMeanOutput());
writerWMean->Update();
writerRatio->SetFileName(outNameRatio);
writerRatio->SetInput(filter->GetRatioOutput());
writerRatio->Update();
writerSD->SetFileName(outNameSD);
writerSD->SetInput(filter->GetSDOutput());
writerSD->Update();
writerPsi->SetFileName(outNamePsi);
writerPsi->SetInput(filter->GetPSIOutput());
writerPsi->Update();
// Figure~\ref{fig:SFS_FILTER} shows the result of applying
// the SFS computation to an image
// \begin{figure}
// \center
// \includegraphics[width=0.25\textwidth]{suburb2.eps}
// \includegraphics[width=0.25\textwidth]{SFSLengthPrettyOutput.eps}
// \includegraphics[width=0.25\textwidth]{SFSWidthPrettyOutput.eps}
// \includegraphics[width=0.25\textwidth]{SFSMeanPrettyOutput.eps}
// \includegraphics[width=0.25\textwidth]{SFSRatioPrettyOutput.eps}
// \includegraphics[width=0.25\textwidth]{SFSSDPrettyOutput.eps}
// \includegraphics[width=0.25\textwidth]{SFSPsiPrettyOutput.eps}
// \itkcaption[Right Angle Detection Filter]{Result of applying the
// \doxygen{otb}{SFSTexturesImageFilter} to an image. From left to
// right and top to bottom: original image, length, width,
// $\omega$-mean, ratio, SD and Psi structural features.
// original image, .}
// \label{fig:SFS_FILTER}
// \end{figure}
/************** pretty images for printing *********/
using OutputImageType = otb::Image<unsigned char, 2>;
using RescalerType = itk::RescaleIntensityImageFilter<ImageType, OutputImageType>;
using OutputWriterType = otb::ImageFileWriter<OutputImageType>;
RescalerType::Pointer rescaler = RescalerType::New();
rescaler->SetOutputMinimum(0);
rescaler->SetOutputMaximum(255);
OutputWriterType::Pointer outWriter = OutputWriterType::New();
outWriter->SetInput(rescaler->GetOutput());
rescaler->SetInput(filter->GetLengthOutput());
outWriter->SetFileName(lengthprettyfname);
outWriter->Update();
rescaler->SetInput(filter->GetWidthOutput());
outWriter->SetFileName(widthprettyfname);
outWriter->Update();
rescaler->SetInput(filter->GetWMeanOutput());
outWriter->SetFileName(wmeanprettyfname);
outWriter->Update();
rescaler->SetInput(filter->GetRatioOutput());
outWriter->SetFileName(ratioprettyfname);
outWriter->Update();
rescaler->SetInput(filter->GetSDOutput());
outWriter->SetFileName(sdprettyfname);
outWriter->Update();
rescaler->SetInput(filter->GetPSIOutput());
outWriter->SetFileName(psiprettyfname);
outWriter->Update();
return EXIT_SUCCESS;
}