/*========================================================================= * * Copyright NumFOCUS * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * https://www.apache.org/licenses/LICENSE-2.0.txt * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * *=========================================================================*/ #include "itkImageFileReader.h" #include "itkImageFileWriter.h" #include "itkDiscreteGradientMagnitudeGaussianImageFunction.h" #include "itkRescaleIntensityImageFilter.h" #include "itkTestingMacros.h" template int itkDiscreteGradientMagnitudeGaussianImageFunctionTestND(int argc, char * argv[]) { const unsigned int Dimension = VDimension; using PixelType = float; using ImageType = itk::Image; // Read input image using ReaderType = itk::ImageFileReader; auto reader = ReaderType::New(); reader->SetFileName(argv[1]); ITK_TRY_EXPECT_NO_EXCEPTION(reader->Update()); ImageType * inputImage = reader->GetOutput(); // Create the itk::DiscreteGradientMagnitudeGaussianImageFunction using DiscreteGradientMagnitudeGaussianFunctionType = itk::DiscreteGradientMagnitudeGaussianImageFunction; typename DiscreteGradientMagnitudeGaussianFunctionType::Pointer function = DiscreteGradientMagnitudeGaussianFunctionType::New(); function->SetInputImage(inputImage); // Set up operator parameters double sigma = std::stod(argv[3]); double maxError = 0.001; unsigned int maxKernelWidth = 100; typename DiscreteGradientMagnitudeGaussianFunctionType::InterpolationModeEnum interpolationMode = DiscreteGradientMagnitudeGaussianFunctionType::InterpolationModeEnum::NearestNeighbourInterpolation; bool useImageSpacing = true; if (argc > 4) { maxError = std::stod(argv[4]); } if (argc > 5) { maxKernelWidth = std::stoi(argv[5]); } if (argc > 6) { interpolationMode = static_cast(std::stoi(argv[6])); } if (argc > 7) { useImageSpacing = static_cast(std::stoi(argv[7])); } double varianceValue = sigma * sigma; typename DiscreteGradientMagnitudeGaussianFunctionType::VarianceArrayType varianceArray; varianceArray.Fill(varianceValue); function->SetVariance(varianceArray); ITK_TEST_SET_GET_VALUE(varianceArray, function->GetVariance()); // Increase code coverage calling other variations of the SetVariance method function->SetVariance(varianceValue); ITK_TEST_SET_GET_VALUE(varianceArray, function->GetVariance()); // Test itkSetVectorMacro double varianceVector[DiscreteGradientMagnitudeGaussianFunctionType::VarianceArrayType::Length]; for (unsigned int i = 0; i < DiscreteGradientMagnitudeGaussianFunctionType::VarianceArrayType::Length; ++i) { varianceVector[i] = varianceValue; } function->SetVariance(varianceVector); ITK_TEST_SET_GET_VALUE(varianceArray, function->GetVariance()); function->SetMaximumError(maxError); ITK_TEST_SET_GET_VALUE(maxError, function->GetMaximumError()); function->SetMaximumKernelWidth(maxKernelWidth); ITK_TEST_SET_GET_VALUE(maxKernelWidth, function->GetMaximumKernelWidth()); bool normalizeAcrossScale = true; ITK_TEST_SET_GET_BOOLEAN(function, NormalizeAcrossScale, normalizeAcrossScale); ITK_TEST_SET_GET_BOOLEAN(function, UseImageSpacing, useImageSpacing); function->SetInterpolationMode(interpolationMode); ITK_TEST_SET_GET_VALUE(interpolationMode, function->GetInterpolationMode()); function->Initialize(); // Create image for storing result auto output = ImageType::New(); output->SetSpacing(inputImage->GetSpacing()); output->SetOrigin(inputImage->GetOrigin()); output->SetDirection(inputImage->GetDirection()); output->SetLargestPossibleRegion(inputImage->GetLargestPossibleRegion()); output->SetRequestedRegion(inputImage->GetRequestedRegion()); output->SetBufferedRegion(inputImage->GetBufferedRegion()); output->Allocate(); output->FillBuffer(PixelType{}); // Step over input and output images using ConstIteratorType = itk::ImageRegionConstIterator; using IteratorType = itk::ImageRegionIterator; ConstIteratorType it(inputImage, inputImage->GetRequestedRegion()); it.GoToBegin(); IteratorType out(output, output->GetRequestedRegion()); out.GoToBegin(); using PointType = typename DiscreteGradientMagnitudeGaussianFunctionType::PointType; PointType point; const unsigned long nop = inputImage->GetRequestedRegion().GetNumberOfPixels(); unsigned long pixelNumber = 0; while (!it.IsAtEnd()) { // To test all available Evaluate functions, we split it in three parts. if (pixelNumber < nop / 3) { out.Set(function->EvaluateAtIndex(it.GetIndex())); } else if (pixelNumber < nop * 2 / 3) { inputImage->TransformIndexToPhysicalPoint(it.GetIndex(), point); out.Set(function->Evaluate(point)); } else { using ContinuousIndexType = typename DiscreteGradientMagnitudeGaussianFunctionType::ContinuousIndexType; using ContinuousValueIndexType = typename ContinuousIndexType::ValueType; inputImage->TransformIndexToPhysicalPoint(it.GetIndex(), point); const ContinuousIndexType cindex = inputImage->template TransformPhysicalPointToContinuousIndex(point); out.Set(function->EvaluateAtContinuousIndex(cindex)); } ++it; ++out; ++pixelNumber; } // Rescale output using OutputPixelType = unsigned char; using OutputImageType = itk::Image; using RescaleType = itk::RescaleIntensityImageFilter; auto rescaler = RescaleType::New(); rescaler->SetInput(output); rescaler->SetOutputMinimum(itk::NumericTraits::min()); rescaler->SetOutputMaximum(itk::NumericTraits::max()); // Write the output image using WriterType = itk::ImageFileWriter; auto writer = WriterType::New(); writer->SetFileName(argv[2]); writer->SetInput(rescaler->GetOutput()); ITK_TRY_EXPECT_NO_EXCEPTION(writer->Update()); std::cout << "Test finished." << std::endl; return EXIT_SUCCESS; } int itkDiscreteGradientMagnitudeGaussianImageFunctionTest(int argc, char * argv[]) { if (argc < 4) { std::cerr << "Missing parameters." << std::endl; std::cerr << "Usage: " << itkNameOfTestExecutableMacro(argv); std::cerr << "inputFileName" " outputFileName" " sigma" " [maximumError]" " [maximumKernelWidth]" " [interpolator]: 0: NearestNeighbourInterpolation; 1: LinearInterpolation" " useImageSpacing" << std::endl; return EXIT_FAILURE; } // Exercise basic object methods // Done outside the helper function in the test because GCC is limited // when calling overloaded base class functions. constexpr unsigned int Dimension = 2; using PixelType = float; using ImageType = itk::Image; using DiscreteGradientMagnitudeGaussianFunctionType = itk::DiscreteGradientMagnitudeGaussianImageFunction; DiscreteGradientMagnitudeGaussianFunctionType::Pointer function = DiscreteGradientMagnitudeGaussianFunctionType::New(); ITK_EXERCISE_BASIC_OBJECT_METHODS(function, DiscreteGradientMagnitudeGaussianImageFunction, ImageFunction); return itkDiscreteGradientMagnitudeGaussianImageFunctionTestND(argc, argv); }