/*========================================================================= * * 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. * *=========================================================================*/ #ifndef itkSpatialNeighborSubsampler_hxx #define itkSpatialNeighborSubsampler_hxx #include "itkImageRegionConstIteratorWithIndex.h" namespace itk { namespace Statistics { template SpatialNeighborSubsampler::SpatialNeighborSubsampler() { this->m_RadiusInitialized = false; this->m_Radius.Fill(1); } template typename LightObject::Pointer SpatialNeighborSubsampler::InternalClone() const { typename LightObject::Pointer loPtr = Superclass::InternalClone(); typename Self::Pointer rval = dynamic_cast(loPtr.GetPointer()); if (rval.IsNull()) { itkExceptionMacro("downcast to type " << this->GetNameOfClass() << " failed."); } if (this->GetRadiusInitialized()) { rval->SetRadius(this->GetRadius()); } else { rval->m_RadiusInitialized = false; } return loPtr; } template void SpatialNeighborSubsampler::SetRadius(const RadiusType & radius) { itkDebugMacro("Setting Radius to " << radius); if (!this->m_RadiusInitialized || this->m_Radius != radius) { this->m_Radius = radius; this->m_RadiusInitialized = true; this->Modified(); } } template void SpatialNeighborSubsampler::SetRadius(unsigned int radius) { RadiusType radiusND; radiusND.Fill(radius); this->SetRadius(radiusND); } template void SpatialNeighborSubsampler::Search(const InstanceIdentifier & query, SubsamplePointer & results) { if (!m_RadiusInitialized) { itkExceptionMacro("Radius not set."); } if (!this->m_SampleRegionInitialized) { itkExceptionMacro("Sample region not set."); } if (!this->GetRegionConstraintInitialized()) { this->SetRegionConstraint(this->m_SampleRegion); } results->Clear(); results->SetSample(this->m_Sample); RegionType searchRegion; IndexType searchIndex; SizeType searchSize; IndexType endIndex; IndexType constraintIndex = this->m_RegionConstraint.GetIndex(); SizeType constraintSize = this->m_RegionConstraint.GetSize(); IndexType queryIndex; typename RegionType::OffsetTableType offsetTable; this->m_SampleRegion.ComputeOffsetTable(offsetTable); ImageHelperType::ComputeIndex(this->m_SampleRegion.GetIndex(), query, offsetTable, queryIndex); for (unsigned int dim = 0; dim < RegionType::ImageDimension; ++dim) { if (queryIndex[dim] < static_cast(m_Radius[dim])) { searchIndex[dim] = std::max(IndexValueType{}, constraintIndex[dim]); } else { searchIndex[dim] = std::max(static_cast(queryIndex[dim] - m_Radius[dim]), constraintIndex[dim]); } if (queryIndex[dim] + m_Radius[dim] < constraintIndex[dim] + constraintSize[dim]) { searchSize[dim] = queryIndex[dim] + m_Radius[dim] - searchIndex[dim] + 1; } else { searchSize[dim] = (constraintIndex[dim] + constraintSize[dim]) - searchIndex[dim]; } endIndex[dim] = searchIndex[dim] + searchSize[dim]; } searchRegion.SetIndex(searchIndex); searchRegion.SetSize(searchSize); if (!this->m_RegionConstraint.IsInside(queryIndex)) { itkWarningMacro("query point (" << query << ") corresponding to index (" << queryIndex << ") is not inside the given region constraint (" << this->m_RegionConstraint << "). No matching points found."); return; } IndexType positionIndex = searchIndex; bool someRemaining = true; typename RegionType::OffsetValueType offset = 0; // The trouble with decoupling the region from the sample is that // there is an implicit assumption that the instance identifiers in the sample // are ordered as if someone was iterating forward through the region // TODO Is this a safe assumption to make? if (this->m_CanSelectQuery || (positionIndex != queryIndex)) { offset = 0; ImageHelperType::ComputeOffset(this->m_SampleRegion.GetIndex(), positionIndex, offsetTable, offset); results->AddInstance(static_cast(offset)); } while (someRemaining) { someRemaining = false; for (unsigned int dim = 0; dim < ImageDimension; ++dim) { positionIndex[dim]++; if (positionIndex[dim] < endIndex[dim]) { offset += offsetTable[dim]; if (this->m_CanSelectQuery || (static_cast(offset) != query)) { results->AddInstance(static_cast(offset)); } someRemaining = true; break; } else { offset -= offsetTable[dim] * (static_cast(searchSize[dim]) - 1); positionIndex[dim] = searchIndex[dim]; } } } } // end Search method template void SpatialNeighborSubsampler::PrintSelf(std::ostream & os, Indent indent) const { Superclass::PrintSelf(os, indent); if (m_RadiusInitialized) { os << indent << "Radius initialized as: " << m_Radius << std::endl; } else { os << indent << "Radius not initialized yet." << std::endl; } os << std::endl; } } // end namespace Statistics } // end namespace itk #endif