-
-
Notifications
You must be signed in to change notification settings - Fork 675
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
ENH: Add frequency iterators and band pass filter.
Initially developed for ITKIsotropicWavelets module [1]. The associated Insight Journal article contains a descriptions of these iterators [2]. The frequency iterators are ImageIterators with GetFrequency and GetFrequencyBins methods to abstract and ease development of filters in the frequency domain. Images in the frequency domain have layouts that depend on the algorithm that generates them, for example after a FFT, the zero frequency is at origin, max positive frequency is at index size/2 - 1. The min negative frequencies is size/2, and the least negative frequency is at index: size - 1. This layout change slightly if image is even or odd in size. (Nyquist bands appear). The best example is itkFrequencyFFTLayoutImageRegionConstIteratorWithIndex.h that contains the layout after applying the standard Fast Fourier Transform to an image. Also the Shifted version, that cover the common case of shifting the zero frequency to the center of the image. These iterators try to abstract this complexity from the developer, for them to only manipulate frequency values. The best example is the FrequencyBandFilter, that acts as BandPass or BandStop filter given two boundary frequencies. The base classes of this hierarchy of iterators handle the case where images are taken in frequency space (scattering experiments for example), and they are just regular ImageIterator with the GetFrequency functions. Note: The frequency iterator for hermitian images is not included (FFT of real images can be chopped by half because they contain redundant information) Also at the time of developing I triggered this question in the mail-list about the lack of meaning of image metadata after performing a FFT: [3] [1]: https://github.com/phcerdan/ITKIsotropicWavelets/ [2]: http://www.insight-journal.org/browse/publication/986 [3]: https://public.kitware.com/pipermail/community/2017-May/013055.html Change-Id: Ie29319eb66cd0dd9f95545396faa7df9226a9364
- Loading branch information
Showing
22 changed files
with
2,239 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,2 @@ | ||
project(ITKImageFrequency) | ||
itk_module_impl() |
221 changes: 221 additions & 0 deletions
221
Modules/Filtering/ImageFrequency/include/itkFrequencyBandImageFilter.h
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,221 @@ | ||
/*========================================================================= | ||
* | ||
* Copyright Insight Software Consortium | ||
* | ||
* 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 | ||
* | ||
* http://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. | ||
* | ||
*=========================================================================*/ | ||
#ifndef itkFrequencyBandImageFilter_h | ||
#define itkFrequencyBandImageFilter_h | ||
|
||
#include <itkInPlaceImageFilter.h> | ||
#include <itkFrequencyFFTLayoutImageRegionIteratorWithIndex.h> | ||
|
||
namespace itk | ||
{ | ||
/** \class FrequencyBandImageFilter | ||
* \brief Performs a frequency band filter in the range LowFrequencyThreshold and | ||
* HighFrequencyThreshold in an input image in the frequency domain. | ||
* | ||
* The default is a pass band between threshold frequencies [0,0.5] Hz or [0, pi] radians, | ||
* where both boundary values also pass (equivalent to SetPassBand(true,true)). | ||
* A pass band sets to zero any value outside the defined range, and let pass without modification the input image inside the band. | ||
* | ||
* Instead, a stop band can be set between the threshold values. | ||
* In this case, the delimited band acts as a stop band, setting values to zero in this range, | ||
* and does not modify input image values outside this range. | ||
* Set a stop band using SetPassBand(false), but it is clearer to use | ||
* SetStopBand(bool, bool) that also control behaviour at band boundaries. | ||
* | ||
* The boundaries of the bands are controlled with | ||
* SetPassLow(High)FrequencyThreshold(bool). The default is to let pass low and high boundaries. | ||
* Also, SetPassBand(true, false), will let pass low boundary/threshold, and stop the high value. | ||
* | ||
* Filters in the module ITKImageFrequency work with input images in the frequency domain. | ||
* This filter is templated over a TFrequencyIterator depending on the | ||
* frequency layout of the input image. | ||
* Images in the dual space can be acquired experimentally, from scattering exaperiments or other techniques. | ||
* The layout of these images is the same as spatial domain images. Use \ref FrequencyImageRegionIteratorWithIndex | ||
* | ||
* Frequency-domain images can be computed from any spatial-domain applying a Fourier Transform \ref ForwardFFTImageFilter. | ||
* Use \ref FrequencyFFTLayoutImageRegionIteratorWithIndex. | ||
* Please note that \ref FrequencyFFTLayoutImageRegionIteratorWithIndex requires a full FFT, | ||
* and is not compatible with the hermitian optimization. | ||
* | ||
* If the output of the FFT is shifted, for example after applying \ref FFTShiftImageFilter, | ||
* use \ref FrequencyFFTShiftedLayoutImageRegionIteratorWithIndex. | ||
* | ||
* | ||
* \ingroup ITKImageFrequency | ||
*/ | ||
template< typename TImageType, | ||
typename TFrequencyIterator = FrequencyFFTLayoutImageRegionIteratorWithIndex<TImageType> > | ||
class FrequencyBandImageFilter: | ||
public InPlaceImageFilter<TImageType, TImageType> | ||
{ | ||
public: | ||
ITK_DISALLOW_COPY_AND_ASSIGN(FrequencyBandImageFilter); | ||
|
||
/** Standard class type alias. */ | ||
using Self = FrequencyBandImageFilter; | ||
using Superclass = InPlaceImageFilter<TImageType, TImageType>; | ||
using Pointer = SmartPointer<Self>; | ||
using ConstPointer = SmartPointer<const Self>; | ||
|
||
/** Method for creation through the object factory. */ | ||
itkNewMacro(Self); | ||
|
||
/** Run-time type information (and related methods). */ | ||
itkTypeMacro(FrequencyBandImageFilter, InPlaceImageFilter); | ||
|
||
/** Typedef to images */ | ||
using ImageType = TImageType; | ||
using ImagePointer = typename ImageType::Pointer; | ||
using ImageConstPointer = typename ImageType::ConstPointer; | ||
using IndexType = typename TImageType::IndexType; | ||
using PixelType = typename TImageType::PixelType; | ||
|
||
/** Typedef to describe the image region type. */ | ||
using ImageRegionType = typename TImageType::RegionType; | ||
|
||
static constexpr unsigned int ImageDimension = TImageType::ImageDimension; | ||
|
||
#ifdef ITK_USE_CONCEPT_CHECKING | ||
/** Begin concept checking */ | ||
itkConceptMacro( ImageTypeHasNumericTraitsCheck, | ||
( Concept::HasNumericTraits< typename TImageType::PixelType > ) ); | ||
/** End concept checking */ | ||
#endif | ||
|
||
/** Frequency Iterator types */ | ||
using FrequencyIteratorType = TFrequencyIterator; | ||
using FrequencyValueType = typename FrequencyIteratorType::FrequencyValueType; | ||
|
||
/****** Frequency Threshold Getters/Setters *****/ | ||
/** Band range: Low threshold/boundary in Hertz */ | ||
itkGetConstReferenceMacro(LowFrequencyThreshold, FrequencyValueType); | ||
itkSetMacro(LowFrequencyThreshold, FrequencyValueType); | ||
/** | ||
* Set low frequency threshold when input frequency is in radians. | ||
* @param freqLowInRadians low freq in radians. | ||
*/ | ||
void SetLowFrequencyThresholdInRadians(const FrequencyValueType& freqLowInRadians); | ||
|
||
/** Band range: High threshold/boundary in Hertz */ | ||
itkGetConstReferenceMacro(HighFrequencyThreshold, FrequencyValueType); | ||
itkSetMacro(HighFrequencyThreshold, FrequencyValueType); | ||
void SetHighFrequencyThresholdInRadians(const FrequencyValueType& freqHighInRadians); | ||
|
||
/** | ||
* Set LowFrequencyThreshold and HighFrequencyThreshold at the same time, | ||
* input frequencies in Hertz. | ||
* | ||
* @param freqLow in hertz. | ||
* @param freqHigh in hertz. | ||
*/ | ||
void SetFrequencyThresholds( const FrequencyValueType& freqLow, | ||
const FrequencyValueType& freqHigh); | ||
|
||
/** | ||
* Set LowFrequencyThreshold and HighFrequencyThreshold at the same time, | ||
* input frequencies in Radians. 1Hz = (1/2pi) rad | ||
* | ||
* @param freqLowInRadians low freq in radians. | ||
* @param freqHighInRadians high freq in radians. | ||
*/ | ||
void SetFrequencyThresholdsInRadians( const FrequencyValueType& freqLowInRadians, | ||
const FrequencyValueType& freqHighInRadians); | ||
|
||
/** The pixel values that correspond to m_LowFrequencyThreshold are passed to the output image, | ||
* independent of m_PassBand */ | ||
itkSetMacro( PassLowFrequencyThreshold, bool ); | ||
itkGetConstReferenceMacro(PassLowFrequencyThreshold, bool); | ||
itkBooleanMacro( PassLowFrequencyThreshold ); | ||
|
||
/** The pixel values that correspond to m_HighFrequencyThreshold are passed to the output image, | ||
* independent of m_PassBand */ | ||
itkSetMacro( PassHighFrequencyThreshold, bool ); | ||
itkGetConstReferenceMacro(PassHighFrequencyThreshold, bool); | ||
itkBooleanMacro( PassHighFrequencyThreshold ); | ||
|
||
/** True: the band is a PassBand. False: StopBand **/ | ||
itkSetMacro( PassBand, bool ); | ||
itkGetConstReferenceMacro(PassBand, bool); | ||
itkBooleanMacro( PassBand ); | ||
|
||
/** | ||
* Utility method equivalent to: | ||
* SetPassBand(true) | ||
* SetPassLowFrequencyThreshold(pass_low_threshold) | ||
* SetPassHighFrequencyThreshold(pass_high_threshold) | ||
* | ||
* @param pass_low_threshold flag to let pass or not low boundary | ||
* @param pass_high_threshold flag to let pass or not high boundary | ||
*/ | ||
void SetPassBand(const bool pass_low_threshold, const bool pass_high_threshold); | ||
|
||
/** | ||
* Utility method equivalent to: | ||
* SetPassBand(false) | ||
* SetPassLowFrequencyThreshold(pass_low_threshold) | ||
* SetPassHighFrequencyThreshold(pass_high_threshold) | ||
* | ||
* @param pass_low_threshold flag to let pass or not low boundary | ||
* @param pass_high_threshold flag to let pass or not high boundary | ||
*/ | ||
void SetStopBand(const bool pass_low_threshold, const bool pass_high_threshold); | ||
|
||
/** If true the frequency cut-off uses the radius of the frequency vector. If false, it uses the max absolute value of the frequency vector. */ | ||
itkSetMacro( RadialBand, bool ); | ||
itkGetConstReferenceMacro(RadialBand, bool); | ||
itkBooleanMacro( RadialBand ); | ||
|
||
/** Control if negative frequencies with absolute value equal to low frequency threshold are passing. | ||
* Only effective when RadialBand is false **/ | ||
itkSetMacro( PassNegativeLowFrequencyThreshold, bool ); | ||
itkGetConstReferenceMacro(PassNegativeLowFrequencyThreshold, bool); | ||
itkBooleanMacro( PassNegativeLowFrequencyThreshold ); | ||
|
||
/** Control if negative frequencies with absolute value equal to high frequency threshold are passing. | ||
* Only effective when RadialBand is false **/ | ||
itkSetMacro( PassNegativeHighFrequencyThreshold, bool ); | ||
itkGetConstReferenceMacro(PassNegativeHighFrequencyThreshold, bool); | ||
itkBooleanMacro( PassNegativeHighFrequencyThreshold ); | ||
|
||
protected: | ||
FrequencyBandImageFilter(); | ||
void PrintSelf(std::ostream & os, Indent indent) const override; | ||
|
||
/* Checks the logic of FrequencyThresholds. */ | ||
void VerifyPreconditions() override; | ||
|
||
void DynamicThreadedGenerateData(const ImageRegionType & outputRegionForThread) override; | ||
|
||
private: | ||
FrequencyValueType m_LowFrequencyThreshold; | ||
FrequencyValueType m_HighFrequencyThreshold; | ||
|
||
bool m_PassBand; | ||
bool m_PassLowFrequencyThreshold; | ||
bool m_PassHighFrequencyThreshold; | ||
bool m_RadialBand; | ||
bool m_PassNegativeLowFrequencyThreshold; | ||
bool m_PassNegativeHighFrequencyThreshold; | ||
}; | ||
} // end namespace itk | ||
|
||
#ifndef ITK_MANUAL_INSTANTIATION | ||
#include "itkFrequencyBandImageFilter.hxx" | ||
#endif | ||
|
||
#endif |
Oops, something went wrong.